WTL Developer's Guide

User Manual:

Open the PDF directly: View PDF PDF.
Page Count: 225

DownloadWTL Developer's Guide
Open PDF In BrowserView PDF
WTL
Developer’s Guide

By Eamon O’Tuathail (www.clipcode.biz)

This document is available under the
Creative Commons’ Public Domain Dedication
http://creativecommons.org/licenses/publicdomain/

WTL Developer’s Guide

Table Of Contents
CHAPTER 1
OVERVIEW OF WTL...........................................................................................................................................4
OBJECTIVES.............................................................................................................................................................4
OVERVIEW..............................................................................................................................................................4
AIMS OF WTL........................................................................................................................................................5
RELATIONSHIP BETWEEN WTL, ATL AND WIN32/WIN64 API..................................................................................7
ALTERNATIVES TO WTL – ATL, MFC, VB, JAVA, DHTML OR ….......................................................................10
INSTALLATION OF WTL..........................................................................................................................................12
CONTENTS OF WTL’S INSTALLATION......................................................................................................................15
POSSIBLE PROBLEMS WITH THE INSTALLATION............................................................................................................15
RESOURCES...........................................................................................................................................................19
TARGET AUDIENCE FOR THIS DEVELOPER’S GUIDE....................................................................................................20
CHAPTER CONTENTS...............................................................................................................................................20
CHAPTER 2
WIN32 SDK WINDOWING...............................................................................................................................22
OBJECTIVES...........................................................................................................................................................22
FUNDAMENTAL WINDOWING CONCEPTS.....................................................................................................................22
MESSAGES............................................................................................................................................................35
THREADS AND WINDOWING.....................................................................................................................................37
CHAPTER 3
ATL WINDOWING.............................................................................................................................................41
OBJECTIVES...........................................................................................................................................................41
OVERVIEW............................................................................................................................................................41
WINDOWING WITH ATL.........................................................................................................................................42
WINDOW CONSTRUCTION .......................................................................................................................................44
MESSAGE MAPS....................................................................................................................................................51
SUBCLASSING AND SUPERCLASSING...........................................................................................................................64
CCONTAINEDWINDOW...........................................................................................................................................66
HIGHER-LEVEL UI.................................................................................................................................................68
DIALOG BOXES......................................................................................................................................................69
WINDOWING FOR ACTIVEX CONTROLS.....................................................................................................................73
ACTIVEX CONTROL CONTAINMENT..........................................................................................................................77
CHAPTER 4
WTL QUICK TOUR............................................................................................................................................86
OBJECTIVES...........................................................................................................................................................86
THE WTL DISTRIBUTION.......................................................................................................................................86
TEMPLATES AND CLASSES ......................................................................................................................................87
WHAT IS NOT IN WTL........................................................................................................................................95
DEVELOPMENT ISSUES............................................................................................................................................98
COMPLETE LIST OF MACROS USED IN WTL.............................................................................................................103
DEBUGGING WITH WTL.......................................................................................................................................105
DETAILED COMPARISON BETWEEN MFC AND WTL................................................................................................107
WTL’S CSTRING................................................................................................................................................116
CHAPTER 5
THE WTL APPWIZARD..................................................................................................................................119
OBJECTIVES.........................................................................................................................................................119
JUST SAY “HELLOWORLD”...................................................................................................................................119
MODAL DIALOG-BASED APPLICATION....................................................................................................................119

2

WTL Developer’s Guide

DEFAULT PROJECT SETTINGS.................................................................................................................................122
SDI APPLICATION................................................................................................................................................124
MDI APPLICATION..............................................................................................................................................130
MULTIPLE THREADS SDI......................................................................................................................................132
MODELESS DIALOG-BASED APPLICATION................................................................................................................140
APPLICATION FEATURES - ACTIVEX CONTROL HOSTING...........................................................................................141
APPLICATION FEATURES - ACT AS A COM SERVER.................................................................................................142
STEP 2 OF THE WTL APPWIZARD........................................................................................................................152
THREADS WITH COM AND WINDOWING.................................................................................................................161
CHAPTER 6
DIALOG BOXES AND CONTROLS..............................................................................................................164
OBJECTIVES.........................................................................................................................................................164
INTRODUCTION.....................................................................................................................................................164
DIALOG BOXES IN WTL......................................................................................................................................166
DDX.................................................................................................................................................................171
WTL WRAPPERS FOR THE STANDARD CONTROLS..................................................................................................189
WTL WRAPPERS FOR COMMON CONTROLS............................................................................................................194
CHAPTER 7
GRAPHICAL PRIMITIVES............................................................................................................................196
OBJECTIVES.........................................................................................................................................................196
OVERVIEW..........................................................................................................................................................196
GETTING STARTED WITH WTL GRAPHICS...............................................................................................................201
WTL HELPER CLASSES – CSIZE, CPOINT AND CRECT...........................................................................................202
GDI OBJECTS AND HANDLE MANAGEMENT ...........................................................................................................206
DEVICE CONTEXT TEMPLATES...............................................................................................................................207
MANAGING ATTRIBUTES OBJECTS..........................................................................................................................209
LINES AND PENS..................................................................................................................................................209
FILLED SHAPES AND BRUSHES...............................................................................................................................213
TEXT AND FONTS.................................................................................................................................................215
CHAPTER 8
INTERNALS OF WTL......................................................................................................................................219
OBJECTIVES.........................................................................................................................................................219
OVERVIEW..........................................................................................................................................................219
HEADER FILES ....................................................................................................................................................219
CONTENTS OF EACH HEADER FILE.........................................................................................................................220

3

WTL Developer’s Guide

Chapter 1
Overview of WTL

Objectives
The objectives of this chapter are to:
•

See where WTL fits into the “big picture” of VC++ development

•

Examine its aims

•

Contrast WTL with other UI development techniques

•

Explain the installation of WTL

•

Describe what is installed with WTL

•

List the available development resources

•

Describe the following chapters

Overview
In the past Visual C++ developers often choose MFC as it provided functionality in a
very wide range of technologies, while the template libraries covered limited areas.
More recently the range and quality of the template libraries have been improving and
now the template answer is more and more often chosen.
The design goal of ATL is to provide fast compact COM components. ISO C++’s STL
provides collections. The VC++ OLE DB Data Consumer and Provider Templates
provide database support. In contrast, MFC provides a single library of C++ classes,
which provide a reasonable range of features in COM creation, collection classes,
database classes and user interface.
Most advanced developers prefer the newer template-route to development. MFC is
monolithic, bulky, not very thread-friendly and, well, is basically old-fashioned! The
template approach is fast (when designed correctly), flexible, covers all the latest
techniques and for new development is definitely the way to go.
Up to now the one major problem for template enthusiasts was how to create the
graphical user interface. ATL does provide lightweight wrappers around Win32/64
windowing, but it certainly does not cover all UI needs. In the past, for anything other
than trivial UI, ATL developers had to resort to MFC UI programming or developing a
VB front-end to their ATL COM components, neither of which was totally satisfactory.
Enter the Windows Template Library. The WTL is an advanced set of wrappers and
productivity enhancement templates, which sit above the Win32/64 API, and provide

4

WTL Developer’s Guide

comprehensive support for a wide variety of graphical user interface requirements.
Keeping true to the template library tradition, WTL is small, ultra-fast and nonintrusive. It covers the latest UI concepts. It works well with multithreading. You can
use WTL on its own or along with any combination of ATL, STL, VC++ Data
Templates, third-party template libraries and your own custom templates libraries,
incorporating just the features you need in each application. WTL usually has no
dependencies on external DLLs – you only need to ship to end-users the EXE for the
WTL application.

Aims of WTL
WTL is to the user interface what ATL is to COM or STL is to collections. Like its
cousins, WTL takes a little while to learn, but when mastered there is no better way of
developing the most advanced applications.
Essentially, WTL accomplishes three significant tasks:
•

Providing an application framework

•

Aggregating UI functionality

•

Wrapping windowing controls, system dialogs and GDI objects

Let’s examine each of these.
Providing An Application Framework
WTL provides a lightweight yet comprehensive application framework, which
automatically furnishes applications based on it with many desirable facilities. The goal
is something less than the impenetrable MFC framework, and something easier than
starting to code WinMain manually.
WTL comes in the form of a series of C++ templates and a WTL AppWizard. The
AppWizard asks the application developer a few questions and then generates a VC++
project and application source code, much of which instantiates or derives from the
WTL templates and classes. In general, what application developers might wish to
change is in the AppWizard-generated application source code, and the “boiler-plate”
style code, which rarely needs to be changed, is inside the WTL templates. Where
necessary these templates may be derived from and custom functionality provided.
WTL has a class to manage a module (a DLL or EXE). This class is instantiated in the
AppWizard-generated application code and within the generated WinMain it is
initialized and terminated. A WTL application can optionally act as a COM server, thus
supporting programmable applications. WTL provides message-handling support and
includes message filtering and on-idle functionality. Change notifications from
WM_SETTINGCHANGE messages may be handled.
MFC provides a document-frame-view architecture. WTL does provide frames (i.e. a
top-level window, containing a menu bar, toolbar and statusbar) and views, but does not
support documents at all. The MFC approach often got in the way of more advanced
developers, and its documents were based on serialized binary data. In the modern
Internet-world, document formats based on XML/XSL are becoming popular and there

5

WTL Developer’s Guide

are demands for flexible storage mechanisms – it might not be the local hard disk;
documents might need to be stored using a variety of remoting architectures, such as
WebDAV or FTP. WTL provides no functionality regarding data formats or storage
mechanisms. Application developers will have to write their own code for these tasks,
e.g. using the fast Win32 APIs WriteFileGather / ReadFileScatter or an
XML parser.
WTL supports applications based on dialog-box, single document interface (SDI) or a
multi-document interface (MDI). For SDI, it also supports multi-threaded SDI, with one
thread per SDI window. It is expected that most advanced applications will use this
architecture (e.g. following the Word 2000 approach). WTL does not support a multithreaded MDI architecture with one thread per MDI child window. This is sensible, as
this avoids a variety of problems with the interaction between threads and MDI parentchild windows.
Aggregating UI Functionality
WTL provides a range of “must-have” features for applications requiring a modern user
interface.
A frame window is provided which manages a rebar control containing a command bar
(enhanced menubar) and a toolbar, a status bar and one or more views. The views may
be based on a generic empty window, or on controls such as richedit, listview or
treeview. If more than one view is used, then the WTL splitter functionality can be
employed to render multiple views with a moveable splitter divider. The views can also
be scrolled. The status bar can be single-paned or multi-paned.
WTL does support traditional menu bars and toolbars, but its main menu presentation
concept is based around command bars. A command bar contains menus and toolbar
icons, and can be displayed and moved around within a rebar control. If a menu item
has the same command id as a toolbar icon, then when the menu is rendered the
equivalent icon is rendered next to each item. WTL supports neither docking nor
floating command bars. WTL’s command-bars are similar to those in Internet Explorer,
and not like the (much more desirable!) command-bars in Word 2000 or Visual Studio.
WTL does not support shortened menus based on usage-data (e.g. WORD-2000).
The concept of dynamic data exchange allows the transfer in both directions of values
between on-screen user interface controls and C++ data members. This functions quite
similarly to MFC’s DDX.
Dialog boxes may be created using ATL, including those, which support ActiveX,
control containment. Visual C++’s ResourceView can be used to layout standard
controls, common controls and ActiveX controls in the dialog resource template. The
ATL Window Message Handler Wizard can be used to map incoming messages for
these controls to message handlers.
WTL provides templates to manage property sheets and property page construction and
wizard construction.
Printing is supported through a printer device context, print preview, devmode handling
and print job management functionality.

6

WTL Developer’s Guide

Enhanced metafiles are supported using a special device context, file management and
enhanced metafile information classes.
Wrapping Windowing Controls, System Dialogs and GDI objects
ATL provides access to generic window functionality, but provides no special support
for windows based on different windows classes. Developers had to resort to manually
coding SendMessage calls as needed. So whether communicating with an edit or a
treeview control, they had to use ATL’s CWindow and call Win32’s SendMessage
with EM_LIMITTEXT or VM_SETITEM and ensure that the parameters were correct
for each message (no type checking was performed). When applications received them
they were in the raw format – wParam and lParam as sent on the message queue.
Applications needed code to convert these to appropriate data types; again making sure
the correct conversion was made for each message type.
Some ATL developers were using code extracted from the ATLCON sample, which
provide some wrappers for Windows UI elements. This has evolved into a full set of
comprehensive wrappers for all the standard and common window controls, the system
dialogs and all the GDI objects, and much more. There are WTL classes for edit,
button, listbox, treeview, listview etc. There are wrappers for the common file dialog,
the color dialog, the font selection dialog, etc. There are wrappers for the device
context, pen, brush, region, font etc.
In addition, a full set of message crackers is provided. Incoming messages are mapped
to message handlers. With ATL these message handlers were passed lParam and
wParam. With WTL’s message crackers, the message handlers’ input is specific to the
incoming message. For example, the handler for WM_LBUTTONDOWN is passed in a
CPoint parameter (CPoint is a WTL wrapper for the Win32 POINT structure –
WTL also has wrappers for such common structures). Note that the ATL Windows
Message Handler Wizard uses ATL’s raw message maps, not WTL’s message crackers.
It is likely wizard support will be improved in the next version of Visual C++.
Many developers choose to use MFC solely for its string-handling support. WTL
eliminates this need by providing its own implementation of CString, which is a
comprehensive clone of MFC’s CString. The string formats WTL’s CString
supports include ASCII, MBCS, UNICODE and Automation’s BSTR. It supports
conversions between all these formats. It supports string manipulation such as
concatenation, trimming and comparison. Its supports printf-like string formatting. It
supports flexible memory management.

Relationship Between WTL, ATL and Win32/Win64 API
WTL is based on ATL and Win32/Win64, and ATL is based on Win32/Win64. When
considering their relationships, we need to examine the source code view and the binary
deliverable view.
Source Code
The Win32/Win64 API is a set of thousands of C functions, covering a vast range of
topics – including two that interest us here, COM and windowing.

7

WTL Developer’s Guide

ATL is a set of C++ templates, which mostly are dedicated to COM programming, but
also include comprehensive low-level support for windowing functionality.

Your Application Code
WTL-AppWizard generated code

WTL

ATL

Win32/Win64 API
WTL is a set of C++ templates, which focus purely on higher-level windowing
functionality. WTL is independent of COM, but can be use along with COM in an
application. Provided you do not select the “Com Server” support in the WTL
AppWizard, then CoInitialize is not actually called.
WTL uses the ATL windowing services, so therefore to use WTL you must have access
to the ATL templates.
ATL makes calls to the Win32/Win64 API. WTL makes calls to the ATL templates and
the Win32/Win64 API. Your application code makes calls to WTL, ATL and the
Win32/Win64 API.
When you run the WTL AppWizard, it produces a number of source files, which
become part of your application source code. The generated source files make calls into
WTL and sets up the framework for the application.
WTL supports Windows 2000, Windows Me, Windows NT 4, Windows 98, Windows
95 OSR 2 and “Classic” Windows 95. WTL support of these is not of the “lowest
common denominator” variety. Instead it uses some #defines (e.g.
_RICHEDIT_VER, _WIN32_IE, _WIN32_WINNT and WINVER) to determine
which Win32/Win64 features to use. WTL does not auto-detect the installed versions of
the OS or Internet Explorer (e.g. it does not use the LoadLibrary /
GetProceAddress or GetVersionEx APIs). Instead, the application developer
must specify the #defines, and the application during compilation will assume they
are available on the client machine and will not run if absent.
As an example of this, take chevrons in menus. When a command-bar is shortened due
to a window resize and chevrons are not used, then the piece of the command-bars,
which are outside the windows boundaries, are unceremoniously cut off. These two

8

WTL Developer’s Guide

screen dumps are an example of a WTL application with a window in full size and
reduced – you note the help menu title and the paste toolbar icon are chopped in half.

The chevron is a symbol comprising two greater-than signs. When pressed, it displays a
pop-down menu containing the items, which have been chopped off the end of the
command-bar and toolbar due to resizing. Chevrons are supported if Internet Explorer 5
is included, and are not supported otherwise.
In a WTL application, if _WIN32_IE is set to 0x0500 or later (e.g. in stdafx.h) then
the chevron is displayed when needed. These two screen dumps show a WTL
application with a resized window, with the chevron not pressed and pressed.

Note that if Internet Explorer 5 or later is installed, but _WIN32_IE is not set as
0x0500 or later, then chevrons are not supported in the application, even though the
installed OS does support them.
WTL does not support PocketPC or X-Box, though it would be nice to see this in future
and there are no technical reasons preventing it. It does seem that WTL is prepared for
Win64, as consideration for 64-bit programming is obvious from some of the WTL
source files.

Binary Deliverable
ATL and WTL are delivered as a set of header files containing C++ templates. ATL
does also have a small number of C++ classes – and these may be used as a separate
DLL but the vast majority of application developers also include them in each project,
and with WTL applications this is also recommended. WTL does not have separate
DLL option. So as needed, the WTL C++ classes (there are a few of them) will be
included in each application.
When you compile a WTL application you will end up with an EXE or a DLL. That is
all you have to deliver to the end-user of the application. There are no dependencies on

9

WTL Developer’s Guide

external libraries, apart from OS libraries such as Kernel32.DLL, which are always
present (if not, then Windows itself would have problems starting).
One exception to this “no-dependencies” rule is if you need to use floating-point
numbers. If this is the case, then you will need to ship the C-Runtime Library (CRT).

Alternatives to WTL – ATL, MFC, VB, Java, DHTML or …
As an application developer, you will certainly find plenty of options available when
you need to construct a graphical user interface. No matter which you choose, you will
often get other developers saying “why did you not use , it is so much better, because of XYZ”. The development platforms wars will
continue for along time. What is noticeable is that there are multiple options now, more
than there were in the past and it is extremely likely to be the case well into the future.
At any one time there are some options, which are very popular, some becoming
popular and some waning in popularity. The best option for a particular project is
constantly in flux.
Issues to consider when selecting your UI development technology include:
•

Quality of implementation (how well does it work)

•

Ease of use (what do developers need to know)

•

The development environment and third party products

•

Available journals and books

•

Industry knowledge (how many software houses use it)

•

Future prospects (will it be here in three years’ time)

•

The requirements of the project

•

Your available development resources (e.g. what do your existing developers
know)

•

Your access to new development resources (e.g. how easy is it to get developers
who knows a particular technology).

The ordering of the desirability of UI toolkits for one criterion might be the direct
opposite for another. For example, in terms of ease of programming, DHTML is easiest,
then VB and then one of the C++ options (which include use raw Win32 API, MFC,
ATL or WTL). In terms of pure functionality and performance, the C++ options are
best, then VB and then DHTML.
DHTML is the use of HTML, especially its forms capabilities, together with scripting,
to produce web-based user interface. As HTML offers fewer UI elements than Win32/
Win64, the user interface is much simpler (both in the positive and negative senses of
that word). For the general public, who has no interest in becoming techno-nerds (e.g.
they have mastered the channel selection and volume control buttons on their TV
remote control but would have difficulty explaining the other buttons), the vast array of

10

WTL Developer’s Guide

UI widgets in Win32 is a problem. They do not understand how they work. For them, it
needs to be much simpler and DHTML is ideal. There is less clutter on-screen. There
are fewer options. It is easy to navigate. However, if you need a UI to do something
more complex than just accepting name and address text fields, you can quickly get into
problems. It is possible to develop custom ActiveX controls and embed these within
DHTML pages. This makes sense in some cases but also moves back to Win32
programming, but it would not be as easy as developing complex applications fully in
Win32. DHTML does have problems when used to construct full UIs for complete
complex applications – the problems are not insurmountable – indeed it would be an
interesting project to tackle.
The latest trend is to use DHTML for desktop applications. The application hosts a
WebBrowser ActiveX control, and then it renders DHTML pages. These can for
example be stored as resources. To get remoting with UI toolkits other than DHTML,
one could consider the Windows Terminal Server option.
DHTML would be very suitable for content delivery (e.g. delivering training courses,)
but not that good at content creation (e.g. developing a training course editing suite),
which would require a more complex UI.
MFC and VB are excellent tools for user interface development if the design you
require closely matches that which these frameworks support well. However, if you
wish something even slightly out of the ordinary then using MFC or VB can involve
lots of unexpected twists and turns. It can take a considerable amount of work
manipulating MFC or VB to meet you specialist needs. MFC and VB also have the
major disadvantage of requiring a very large run-time (MFC DLLs and VBRUN).
MFC’s support for Active Document is absent from WTL.
ATL applications and components often need to display windows and dialog boxes.
Though it mainly concentrates on supporting COM, ATL does provide a good level of
windowing functionality. Windowing in ATL is not as easy to use as MFC or VB.
ATL’s approach is flexible and highly configurable, but for windowing it is too-low
level. If you are using ATL, and need windowing, then you should really move up to
WTL.
WTL is very new. Few developers know it. More mainstream development groups will
in due course follow the early adopters. Currently there are far more developers using
the other UI options. WTL documentation (apart from the document you are currently
reading) is very sketchy. All the other UI options are better documented. Any new
technology will suffer these disadvantages over established technologies.
Why Choose WTL?
WTL is much less cumbersome than MFC and the resulting applications are less
bloated. For advanced applications WTL gives you all the benefits of using the raw
Win32 APIs but saves you a lot of time.
The Win32 API can be used directly, but it has a bewildering range of seemingly
independent functions and sometimes it is difficult to determine how they fit together.
WTL provides more homogeneous access to the available functionality.

11

WTL Developer’s Guide

Template libraries in general are becoming increasingly popular because the method
that gets called is selected at compile time, rather than run-time, with benefits in terms
of efficiency and code size.

Installation of WTL
WTL is evolving rapidly and is likely to see changes in its delivery mechanism over
time.
Future Delivery Mechanism
It is likely that WTL will be folded into the Visual C++ product in the next release of
Visual Studio. Hence it probably will automatically be installed along with ATL, STL
and the Data Templates.
Current Installation Mechanism
Currently, WTL is released as part of the Microsoft Platform SDK (January 2000 or
later). Installation involves three manual steps; though it is likely these will be
automated in some future update.
First Step - Getting the Files
The first step is to get the relevant WTL files onto your hard disk. The Microsoft
Platform SDK is available as part of the MSDN Subscription (disk 3 in the
Development Platform set of CDs) or from the http://msdn.microsoft.com website. The
Platform SDK using the Microsoft Installer to put the select files on your hard disk.
You may choose to install/download the entire Platform SDK or subsections. The
Platform SDK is large – but contains lots of goodies and it is recommended that all
advanced developers have it fully installed on their hard-disks (we do all have 100GB+
hard-disks nowadays – don’t we). To get the WTL files onto your hard-disk you may
either install the whole Platform SDK or any selection of sections, provided they
include the “Windows Template Library”, which is located under the Source Code node
in the tree in the installation wizard.
The installed size of the WTL developer files is quite small – the exact size depends on
the sector sizes on your disk – on one demonstration machine it was only 1,276KB.
(Note that apps built with WTL can be tiny – even less than 25K). The WTL files will
be installed on your disk under \Src\WTL, where  is the installation path
you provided (e.g. c:\Program Files\Microsoft Platform SDK).

12

WTL Developer’s Guide

Second Step – The WTL AppWizard
The second step is to move the WTL AppWizard to the appropriate location. You need
to manually move it from \Src\WTL\AppWizard to \Common\MSDev98\
Template, where  is the installation path of Visual C++. Note that to use WTL
you must have Visual C++ 6.0 installed. It is recommended, though not required, that
you also have service pack 3 for Visual C++ 6 installed.
When you now start Visual C++ and select New from the File menu, you will see a new
entry under the “projects” tab called ATL/WTL AppWizard.

When clicked, it displays an AppWizard, which lets you select a few options, and then
produces the initial project source files for you.

13

WTL Developer’s Guide

Third Step – Updating the Path
The third step is to set the path so that the compiler can find the WTL header files.
WTL is a template library consisting of C++ header files (there are no .cpp files). When
compiling, the compiler needs to be able to find these files. The files are located under
\Src\WTL\include. One option would be to include the full pathname in your
application’s source files, to which most developers will respond in shock horror tones
(and they are quite right too!). The second option is to put them in a directory which is
already on the path – such as inside the ATL include directory. However, this is
changing directories associated with the development environment, and it is usually
best to leave them alone (but for here it will work). The third and recommended option
is to configure the Visual C++ compiler on each machine to look in the WTL directory
when searching for include files. This can be done by selecting the Options entry from
DevStudio’s Tools menu, then selecting the Directories tab, then selecting “Include
Files” from the “Show Directories For” drop-down list, and then append an entry
stating where the WTL files can be found. Do not forget to include the ‘include’ part of
the path.

14

WTL Developer’s Guide

The ordering of the WTL entry in this list of directories is not important.

Contents Of WTL’s Installation
The installation of WTL contains a top-level directory, \Src\WTL\, and this in
turn contains a readme.txt, and three subdirectories, \Include, \Examples and
\AppWizard.
The readme.txt is the only documentation on WTL available currently from Microsoft –
and it is certainly sparse. It is about 6 pages in length, and briefly introduces WTL,
provides a list of files in the installation, and provides a one-line (!) description of each
template/class in WTL, and then mentions the WTL AppWizard.
The \AppWizard directory contains a single AWX file, and as we have seen this needs
to be manually copied to the VC++’s installation path.
The \Include directory contains 15 WTL header files – this directory can be considered
the core of WTL.
The \Examples directory contains three examples showing how to use WTL. They are:
•

MTPad – Notepad using multiple threads

•

MDIDocVw - Document View sample

•

GuidGen – ATL implementation of GuidGen (calls the Win32 API
CoCreateGuid)

The MTPad sample is a SDI application that emulates Notepad.
The MDIDocVw shows how to use the MDI support and how frames and views
interact.
The GuidGen sample looks exactly like the GUIDGEN utility COM developers have
used for years. It uses a single dialog box class to manage the various UI features of
generating GUIDs.
As a test of the installation, it is recommended to load up one of the examples in
DevStudio, build it and run it.

Possible Problems with the Installation
There are a small number of common installation problems / questions. Here we will try
and provide appropriate answers.
I cannot find the WTL AppWizard.
You must first manually copy it to the appropriate location – it is not done
automatically during installation. Note that in DevStudio’s New Project dialog it is
called the ATL/WTL AppWizard.
The Examples GuidGen and MdiDocVw work fine, but MTPad compiles and links
OK, but crashes when it runs.

15

WTL Developer’s Guide

Tut, tut, tut: you are not reading the output from the Build!
When you compile stdatl.cpp, the following message is display in the Build tab of the
output window:
NOTE: WINVER has been defined as 0x0500 or
greater which enables Windows 2000
features.
Caution: When building applications with
WINVER set to 0x0500 or greater, the
resulting binary may not run as expected
on earlier platforms, such Windows NT 4.0
or Windows 95 & Windows 98
See the Platform SDK release notes for
more information.
What is happening is that stdatl.h in the MTPad example has WINVER set to 0x0500,
and this is used in various header files to determine if Windows 2000-specific material
should be included. The above comment is outputted due to some code in windows.h,
which checks WINVER and outputs the message if it is as 0x0500 or greater. This
means that MTPad will run fine on Windows 2000 but will fail on older OSes, such as
Windows NT4, where the problem will manifest itself as a failed ATLASSERT just
after a call to the Win32 API ::GetMenuItemInfo in the WTL header file
AtlCtrlw.h. You can comment out the block of code calling ::GetMenuItemInfo,
in which case the menu will not be displayed on Windows NT 4, but apart from that the
MTPad application will run correctly (it does have a toolbar for also accessing
commands). Alternatively, comment out the lines:
// #define WINVER
0x0500
// #define _WIN32_WINNT 0x0500

from stdatl.h file, and the application will then run on older OSes. Actually you really
only have to remove the WINVER define, and MTPad will then run fine, but it makes
sense to remove both: check out the MSDN article titled "Header File Conventions",
which states “In general, applications expected to run on Windows 95 should be built
without defining _WIN32_WINNT.”
In the MTPad example no Windows 2000-specific features are used, so these defines
are not needed. However, note that for some applications you will wish to use Windows
2000 features, and therefore will definitely wish to include these lines. You will need to
decide to either have a special code path in your application for older OSes, and another
for Windows 2000 and later OSes, or you will decide you will only run on Windows
2000 (for example, for other reasons it might be necessary – e.g. you are using the
thread pool functions which are exclusive to Windows 2000 or later). In the latter case
it would be a good idea at application startup to detect if the OS can run the application,
and if not to display a simple error message, rather than crashing mid-way through, as
in the case of the default MTPad on non-Windows 2000 OSes.

16

WTL Developer’s Guide

I have not added WTL’s include directory to DevStudio’s directory list, yet the
samples compile and run fine?
Yippee, it works, so what’s the problem! However, you are quite right to be concerned,
as such “magical” solutions often have a nasty way of coming back to haunt you with
difficult-to-detect problems (usually within a few hours of a your product’s release!).
The files must be coming from somewhere, but where?
Your first worry might be it is coming from either the Platform SDK’s own Include
directory or one of the include directories under VC++, such as ATL’s.
It is a worry because you might be getting an old version of the headers and you
definitely want the latest. Also, going forward, it is vital that you are sure which version
and the location of the header files you are using, as new releases of WTL are likely.
All members of a development team using code based on WTL should be building off
the same version of the WTL headers.
So you will start a search of your entire hard disk, looking for say, WTL’s “atlapp.h”
file. (All WTL’s header files are named beginning with “atl”, which provides a good
hint as to the likely packaging arrangements when Visual Studio 7 is launched).
You will not find the WTL files in any location other than \Src\WTL\include.
What has happened with the examples is that per example, they have a relative include
path configured, pointing to the location of the WTL includes relative to the WTL
examples.
This solution is fine for the WTL examples, but it is highly unlikely that all your code
will be located under \Src\WTL\examples, so you should follow the suggestion
of modifying DevStudio’s options as explained earlier. Also, it means that inside
DevStudio’s text editor, when you select a WTL header file in source code, right click,
and select Open Document , that with the ..\..\include method the file will
not be displayed, but with the DevStudio’s options techniques it does work, and the
appropriate WTL header file is displayed.

17

WTL Developer’s Guide

The GUIDGEN and MTPAD examples do not use stdafx.cpp & stdafx.h
The title AFX is a historical naming artifact, and has no coding significance.

AFX started off as the project name for the earliest MFC project, before it was
christened MFC. The AFX name has survived through later MFC versions by being
embedded in the stdafx.cpp and stdafx.h files and is part of the name of certain MFC
global functions. By MFC tradition, stdafx.cpp contains boilerplate code which only
needs to be added once and stdafx.h contains includes of other framework header files,
general application header files and definitions of versions (e.g. WINVER). For each
project these files are slightly different, and therefore the files are part of the application
source files as opposed to the framework’s. Both files tend to be quite small. Most
developers know what should go in each.

Stdafx.cpp and stdafx.h are also traditionally the files used to manage precompiled
headers. One .cpp file (in this case stdafx.cpp) is selected for creating the PCH file

18

WTL Developer’s Guide

based on a selected header file (stdafx.h), and then all other files use the PCH, based on
the same header.
When ATL got started, it also used stdafx.cpp and stdafx.h for the same purposes,
though obviously their contents were quite different and contained absolutely no MFC
material.
Now with WTL, we also see the use of stdafx.cpp and stdafx.h in AppWizard generated
code. (Is AFX a virus or what!).
So, to return to the question, why do WTL’s examples MTPad and GuidGen not use
stdafx.cpp and stdafx.h? The answer is for no particular reason, but probably just that
the example developers wanted to break with Visual C++ tradition – the WTL
AppWizard has obviously much greater respect for our coding heritage. In brief, it does
not matter. So long as the appropriate WTL header get included and the pre-compiler
header files get set up correctly, you may choose any file names (it is suggested that you
use stdafx.[h/cpp], as that is what the WTL AppWizard uses).

Resources
WTL currently has few available resources, but this is likely to change shortly.
Development Tools
The Microsoft Platform SDK contains the WTL itself, and apart from that there are
currently no other development tools available. It is likely that many tool vendors who
currently based their products on MFC will migrate to WTL.
Internet Sites
Some Internet sites with interesting WTL material are:
•

http://www.develop.com/dm/dev_resources.asp (very comprehensive)

•

http://www.idevresource.com/wtl

•

http://www.codeproject.com/wtl

•

http://www.argsoft.com/Wtl/DocView.html

•

http://www.sellsbrothers.com/tools/index.htm#wtl

Newsgroups and Discussion List
There are no newsgroups or discussion lists dedicated specifically to WTL
programming. However, those targeted at ATL have many WTL-related postings.
The best ATL discussion list is:
•

http://discuss.microsoft.com/archives/atl.html

The best ATL newsgroup is:
•

news://msnews.microsoft.com/microsoft.public.vc.atl

19

WTL Developer’s Guide

Books
Apart from the WTL Developer’s Guide which you are currently reading, there is no
other WTL book available. It is expected that most of the major software engineering
book publishers will provide books on WTL in the future as its popularity increases.
Many ATL books cover ATL windowing, and this is a crucial foundation for WTL. The
best ATL books are:
•

“Professional ATL COM Programming”, Dr. Richard Grimes, Wrox Press,
1998, ISBN: 1-861001-40-1 (he also has a beginner’s guide and a reference
manual published with Wrox)

•

“ATL Internals”, Brent Rector and Chris Sells, Addison-Wesley, 1999, ISBN:
0-201-69589-8

•

“Creating Lightweight Components with ATL”, Jonathan Bates, SAMS, 1999,
ISBN: 0-672-31535-1 [interesting discussion of adding support for Active
Documents to ATL applications]

•

“Inside ATL”, George Shepherd and Brad King, Microsoft Press, 1999,
1-57231-858-9

Journals
Visual C++ Developer Journal (http://www.vcdj.com) published an article on WTL in
its April 2000 issue.

Target Audience For This Developer’s Guide
This developer’s guide is aimed at experienced software developers who have a good
knowledge of C++ and some exposure to C++ templates, ATL, Win32 windowing and
MFC UI programming.

Chapter Contents
Chapter 2 – Win32 SDK Windowing reviews core concepts of Win32 windowing,
which must be clearly understood before moving on to ATL windowing and WTL.
Chapter 3 – ATL Windowing examines the functionality within ATL relating to
windowing. This includes constructing windows and dialogs, handling messages and
command processing through message maps and super/subclassing. WTL does not
replace this; rather it extends the range of functionality already available with ATL
windowing.
Chapter 4 – WTL Quick Tour provides an overview of developing with WTL. It
introduces the WTL build process, application architecture and describes a number of
development issues, such as CRT usage. It has a list of WTL’s templates and classes, a
description of WTL’s CString, and explains the macros used.

20

WTL Developer’s Guide

Chapter 5 – WTL AppWizard walks through the coded generated from the various
AppWizard options. This chapter examines each AppWizard option and analyses the
code changes that it effects.
Chapter 6 – Dialogs and Controls looks at how dialog boxes and their contents work
with WTL. It looks at how a variety of standard, common and ActiveX controls can be
used, along with common system dialogs and property pages. It looks at the message
crackers. It describes DDX, data validation and how to extend these.
Chapter 7 – Graphical Primitives looks at outputting graphics. It introduces the WTL
templates for GDI objects, such as CDC, CPen, CBrush, CFont, CBitmap,
CPalette and CRgn. It explains the role of CDC derivatives, for painting client-area
rendering, complete window rendering and enhanced metafiles.
Chapter 8 – Internals of WTL walks through the WTL header files and examines the
design for these templates.
Chapter 9 – Bugs and Suggestions lists known bugs in the current version of WTL and
makes some small and large suggestions above extending its capabilities.

21

WTL Developer’s Guide

Chapter 2
Win32 SDK Windowing

Objectives
The objectives of this chapter are to:
•

Review the fundamentals of windowing concepts

•

Introduce windowing terms

•

Explain how the user interface is structured in the Windows OS

•

Describe how the Windows OS handles concepts such as message queues,
subclassing, superclassing, the message loop and general windows management

•

Examine how threads and windows interact

•

Describe how graphics primitives work

Fundamental Windowing Concepts
WTL is built above ATL windowing, which in turn is built above Win32 SDK
windowing. Before exploring WTL, it is useful to review how windowing works within
the Windows family of operating systems, which we do in this chapter, and how ATL
handles windowing, which we will do in the next chapter. There are a number of
important Win32 concepts and if clearly understood they will facilitate understanding
why ATL and WTL are structured the way they are.
Types of Windows
At the most basic level, everything that appears on screen on a PC running the
Windows OS is either a window or a graphical primitive drawn within a window.
Win32 APIs are provided to create, manage and destroy windows and to render
graphics inside them. Each user interface construct lives within a separate window –
known as a control.
The end-user might think there are many types of windows – graphical output windows,
user interface standard controls (pushbutton, radio-button, etc.), common controls
(listview, treeview) and ActiveX controls. (A special case of ActiveX control, known as
windowless - can operate without its own window – it lives within the window of its
container). Fundamentally, they are all instances of windows that work according to the
same set of rules.

22

WTL Developer’s Guide

Hierarchies of Windows
Windows are arranged on screen in a hierarchy. The root desktop window is a listview
control, in large-icon mode. Applications’ top-level windows and their child windows
form nodes within this hierarchy, which is maintained by the OS.
When you create a window, you must specify its parent. The new window will be
located within the hierarchy under the node of the parent window. When a window is
destroyed, all its child windows are automatically destroyed.
Window Classes
A window class is a description of important aspects of how a window works. Each
window you create must be of a particular window class. The operating system provides
window classes for the standard controls, such as Button, Edit and List Box and the
common controls, such as listview, treeview and calendar picker.
A window class is registered using the Win32 RegisterClassEx function, and
unregistered using the UnregisterClass function.
ATOM RegisterClass( //Return is an Atom identifier for the class
CONST WNDCLASS *lpWndClass // IN: pointer to window
// class structure
);
BOOL UnregisterClass( //Return states whether it succeeded
LPCTSTR lpClassName, // lpszClassName field from WNDCLASS
HINSTANCE hInstance
// handle to application instance
);

Sometimes a window class is referred to as a “template” for a window. In programming
terms, a window class is a Win32 structure (not a C++ class). The most important
aspect of a window class is the window procedure. Other aspects include an icon when
the application is minimized, background brush, default cursor and the hInstance of
the owner (EXE or DLL containing the WndProc). A process should create a window
class and then instantiate windows based on it.
typedef struct _WNDCLASS {
UINT
style; // Bitmask containing additional settings
// (e.g. CS_NOCLOSE – Do not show CLOSE
WNDPROC
lpfnWndProc;//The windows procedure function
int
cbClsExtra; // Extra bytes for this class
int
cbWndExtra; // Extra bytes for each window
HINSTANCE hInstance;// The instance handle of the EXE/DLL
// containing the window procedure
HICON
hIcon;
// Icon for the class
HCURSOR
hCursor;
// Cursor for the class
HBRUSH
hbrBackground; // How the background should
// be drawn
LPCTSTR
lpszMenuName; // Resource name for the menu
LPCTSTR
lpszClassName; // Class name
} WNDCLASS, *PWNDCLASS;

Important styles include:

23

WTL Developer’s Guide

•

CS_SAVEBITS – copy the pixels in a window before it is obscured, and
replace the pixels when the window becomes visible again (use for small
windows only)

•

CS_NOCLOSE – Disable the close button on the window menu

•

CS_CLASSDC – use a unique device context to be shared among all windows
of this class (NOT RECOMMENDED for multithreading)

•

CS_OWNDC – use a unique device context for each window of this class

•

CS_PARENTDC – enables a child window to draw into its parent (a common
DC from the system’s cache is used)

Extra application-specific bytes can be stored with the class (cbClsExtra) and each
window of the class (cbWndExtra)
The class name is used later when creating windows – as one argument to
CreateWindow[Ex] states the name of the window class that the new window should be
based upon.
The Window Procedure
Window messages are constantly being sent to windows to be processed, and they
handle these in a window procedure. This is a function specific to the application and
class that detects messages of relevance to the application and responds, usually by
calling another function within the application. Messages not of interest are forwarded
to the default handler (DefWindowProc). It is quite normal that multiple windows
could be based on the same window class, which means that multiple windows could
have their messages handled by the same window procedure (a HWND parameter
identifies the window to which the message refers).
The Window
Calling CreateWindowEx creates a window. The name of a window class that has
already been registered must be provided (this parameter may not be NULL).
HWND CreateWindow(
LPCTSTR lpClassName,
LPCTSTR lpWindowName,
DWORD dwStyle,
int x,
int y,
int nWidth, nHeight,
HWND hWndParent,
HMENU hMenu,
HINSTANCE hInstance,
LPVOID lpParam
);

//
//
//
//
//
//
//
//
//
//

Name of window class
Name of window, to appear in titlebar
Bitmask of style of window
X&Y position of new window compared
to parent window
dimensions of window
parent window
menu handle
Module associated with the window
custom data to be sent with WM_CREATE

24

WTL Developer’s Guide

STEP 1: Application provides a WndProc function
LRESULT CALLBACK MyWndProc(HWND hwnd,
UINT uMsg,WPARAM wParam,LPARAM lParam ){
switch (uMsg) {
case WM_CREATE:
// …;
return 0;
case WM_PAINT:
// …;
return 0;
...
Window
X Application calls RegisterClassEx
STEP 2:
Application fills in the WNDCLASSEX structure, including setting the class name to a
application-defined string (e.g. “MyFirstClassName”) and setting the WNDPROC field
to point to an application defined function (e.g. MyWndProc) whose signature is
WNDPROC – then the application calls RegisterClassEx to register it.
This results in an atom getting added to an atom table based and the class name, and
the system using the atom identifier to locate the window class for it
Atom table for registered window classes
MyFirstClassName

Atom Identifier X

...

Atom Identifier Y

...

Atom Identifier Z

Window Class Stored
information for atom identfiier X
WndProc=ptr to MyWndProc
...

STEP 3: Application calls CreateWindowEx
Application calls CreateWindowEx and among it
parameters is the name of the window class to use
This results in a new window being created. Each
window has certain pieces of hidden data, one of
which is a pointer to a wndproc function. This is
initialized to point to the wndproc identified in the
window class
STEP 4: Messages are processed for window
When messages are received for a window, its ptr
to a wndproc is used to process the message

25

Ptr to
MyWndProc

WTL Developer’s Guide

Sub-Classing
When a window is created based on a window class, the window records a function
pointer to the window procedure specified in the window class. Subclassing is used to
modify the behavior of a class, without having to rewrite the class entirely, possibly
duplicating lots of code. Subclassing involves an application replacing the window
procedure of a window with a different window procedure, which can process some
messages it receives and pass them onto the original window procedure. This can be
used for window classes you write yourself or those which are provided by the
operating system for predefined controls.

STEP 1: Application calls RegisterClassEx
Application calls RegisterClassEx as before
Atom table for registered window classes
MyFirstClassName

Atom Identifier X

...

Atom Identifier Y

...

Atom Identifier Z

Window Class Stored
information for atom identfiier X
WndProc=ptr to MyWndProc
...

STEP 2A: Instance Subclassing
With instance subclassing, the window needs to be
created as before using CreateWindowEx. It
gets its wndproc from the registered wndproc value
for the window class

Ptr to
MyWndProc

STEP 3A: Instance Subclassing
Then, to carry out instance subclassing, write a
separate window procedure (e.g.
mySecondWndProc) and then call
SetWindowLongPtr(GWL_WNDPROC,
mySecondWndProc); Note that the return
value from this call will the existing wndproc in
use. In the implementation of
mySecondWndProc, for certain messages one
could call myWndProc if desired).

Ptr to
MySecondWndProc

Instance subclassing is used for a single instance of an existing window. This can be
done by calling SetWindowLong(GWL_WNDPROC) or
SetWindowLongPtr(GWL_WNDPROC) for an existing window (the difference

26

WTL Developer’s Guide

between the functions is that SetWindowLong is compatible with Win32 only,
whereas SetWindowLongPtr is compatible with both Win32 and Win64, so it is
better to use SetWindowLongPtr).
Global subclassing is to replace the window procedure in the window class, which will
come into effect for all subsequent window creation based on that window class. This is
done by calling SetClassLongPtr(GCL_WNDPROC) to replace the window
procedure stored in the window class. All new windows created after this will new the
new windows procedure. Windows created before this will continue to use the initial
window procedure.

STEP 1: Application calls RegisterClassEx
Application calls RegisterClassEx as before
Atom table for registered window classes
MyFirstClassName

Atom Identifier X

...

Atom Identifier Y

...

Atom Identifier Z

Window Class Stored
information for atom identfiier X
WndProc=ptr to MyWndProc
...

STEP 2A: Global Subclassing
With global subclassing,
SetClassLong(GCL_WNDPROC,
mySecondWndProc) is called, which replaces
the WndProc in the class information

Window Class Stored information
for atom identfiier X
WndProc=ptr to MySecondWndProc
...

STEP 3A: Creating the Window
All windows created subsequent to the call to
SetClassLongPtr(GCL_WNDPROC)
will use the new window procedure

Ptr to
MySecondWndProc

Super-Classing
Superclassing involves extending the functionality of a base window class. In the
superclass’ own window procedure, when it receives a message it may process it
directly or invoke the window procedure of the base class.

27

WTL Developer’s Guide

STEP 1: Application calls RegisterClassEx
Application calls RegisterClassEx as before
Atom table for registered window classes
MyFirstClassName

Atom Identifier X

...

Atom Identifier Y

...

Atom Identifier Z

Window Class Stored
information for atom identfiier X
WndProc=ptr to MyWndProc
...

STEP 2: Retrieve Class Info
The application calls
GetClassInfoEx to retrieve
class information about a
named class

Window Class Stored
information for atom identfiier X

STEP 3: Fill in WNDCLASSEX and call
RegisterClassEx
The application creates a new WNDCLASSEX structure,
some of whose fields are set to the values retrieved in step 2,
and some which are new. The wnd procedure is new (e.g.
MySuperWndProc). Then it calls RegisterClassEx
to create a new window class. The application will keep a
record of the information from step 2, and it might use this
within the implementation of the new window procedure.

WndProc=ptr to MyWndProc
...

Window Class Stored information
for atom identfiier Y

WndProc=ptr to MySuperWndProc

Atom table for registered window classes
MyFirstClassName

Atom Identifier X

MySuperClass

Atom Identifier Y

...

Atom Identifier Z

...
STEP 4: The application creates windows
The application calls CreateWindowEx to
create a new window. It may use the window class
name of the original class or the superclass.

Ptr to
MySuperWndProc

This is used by calling GetClassInfo to retrieve information about the window
class of the base class, modify the data returned (such as the window procedure) and
then calling RegisterClassEx to register the superclass.

28

WTL Developer’s Guide

You will often use subclassing and superclassing when trying to modify the behavior of
predefined windows controls, such as edit boxes and list boxes. Subclassing and
superclassing are allowed only within a process.
Message Queues
Each thread that invokes Win32 windowing functions has its own message queue. The
message queue of the thread in which a window is created is used to deliver messages
concerning that window to the application.
The messages for each window are queued
on the message queue of the thread in
which the call to CreateWindow was
called which created the window

Window X

Window Y

Message Queue
for thread A

Window Z

Message Queue
for thread B

Thread A

Thread B

Message loop
Each thread that has a message queue should process incoming messages by means of a
message loop. This calls the GetMessage function, which blocks until a message
becomes available, and then calls DispatchMessage, which results in the
appropriate window procedure getting called.
MSG msg;
while (GetMessage(&msg, 0, 0, 0))
DispatchMessage(&msg);

The message loop is also the basis of single-threaded apartments in COM. The idea is
that if a component is designed to be called from only has a single thread, and multiple
threads inside or outside the application wish to call it, a hidden window is created, and

29

WTL Developer’s Guide

all functions calls for a particular COM component are posted as messages to this
window. The object can then extract them one by one and process them.
Capture and Focus
Normally messages are sent to the window in which they occurred. For user input this is
the focus window. For mouse messages, many applications wish to detect the
corresponding mouse up message for every mouse down message they receive. Calling
SetCapture does this.
Window Styles (Traits)
Application developers may set a wide variety of window styles when creating windows
and these influences the look and behavior of the window. There are two types of styles
– standard styles (e.g. WS_CHILD, WS_CLIPCHILDREN, WS_VISIBLE) and
extended styles (e.g. WS_EX_PALETTEWINDOW, WS_EX_APPWINDOW,
WS_EX_CLIENTEDGE).
Another term for “window styles” used within WTL/ATL is “traits”.
Windows Properties
The operating system can maintain a set of named properties for each window. A
property name is defined by the application and the property value is a HANDLE,
usually a pointer to memory that the application allocates and fills in with data relevant
to the window.
Sample – WindowingWithSDK
This sample explores programming windowing concepts using raw Win32 calls. Later,
when we use ATL and WTL, it will be instructive to look back and see exactly how it
could be done manually with the SDK calls. The WindowingWithSDK workspace
contains five small projects.
The WindowsWithSDK project is a simple “Hello World” project, generated in
Developer Studio using the New Workspace command, then selecting Win32
Application, and then selecting “A typical “Hello World!” application”. The code is all
auto-generated by Developer Studio. It simply initializes a WNDCLASSEX structure and
calls RegisterClassEx to register the window class. Then it calls
CreateWindow to create a window based on the window class. It has a message loop
which processing messages, and forwards them to the widow procedure.
The InstanceSubclassing project was generated in the same way and modified to show
instance subclassing. An additional window procedure was written, called
MySubclassedWndProc. This processes WM_LBUTTONDOWN messages and
forwards all other calls to the original window procedure, using CallWindowProc.
LRESULT CALLBACK MySubclassedWndProc(HWND hWnd, UINT message, WPARAM
wParam, LPARAM lParam){
switch (message) {
case WM_LBUTTONDOWN:
OutputDebugString(
TEXT("Mouse Down detected in subclassed WndProc\n"));

30

WTL Developer’s Guide

break;
default:
return CallWindowProc(MyWndProc, hWnd,
message, wParam, lParam);
}
return 0;
}

The subclassing is put into effect by a call to SetWindowLongPtr, after first
creating the window.
hWnd = CreateWindow(szWindowClass, szTitle,
WS_OVERLAPPEDWINDOW,CW_USEDEFAULT, 0,
CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
SetWindowLongPtr(hWnd, GWL_WNDPROC,
(LONG_PTR)MySubclassedWndProc);

The GlobalSubclassing project changes the window procedure in the registered class.
The class is first registered as before. To change class settings with
SetClassLongPtr, an existing window based on that class must be used (which
indirectly provides access to the window class). This first window is created, and then
SetClassLongPtr is called to change the window procedure to use for all new
windows based on that class. (Note that the first window will continue to use the
original window procedure and is not affected by the call to SetClassLongPtr).
hWnd1 = CreateWindow(szWindowClass, szTitle,
WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0,
CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
SetClassLongPtr(hWnd1, GCLP_WNDPROC,
(LONG_PTR)MySubclassedWndProc);
hWnd2 = CreateWindow(szWindowClass, szTitle,
WS_OVERLAPPEDWINDOW,CW_USEDEFAULT, 0,
CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);

The SuperClassing project demonstrates superclassing of an existing window class. The
size field of the WNDCLASSEX is first correctly initialized. It extracts the window class
of an existing class by calling GetClassInfoEx. It copies certain fields to a new
WNDCLASSEX structure, and sets the window procedure to point to a new function.
Then it calls RegisterClassEx to register the new window class. In the window
procedure certain calls are forwarded to the original window procedure.
wcex_base.cbSize = sizeof(WNDCLASSEX);
GetClassInfoEx(hInstance, szWindowClass, &wcex_base);
CopyMemory(&wcex_superclass, &wcex_base, sizeof(WNDCLASSEX));
wcex_superclass.lpfnWndProc
=
(WNDPROC)MySuperclassWndProc;
wcex_superclass.lpszClassName =
TEXT("SuperClassName");
RegisterClassEx(&wcex_superclass);
hWnd = CreateWindow(TEXT("SuperClassName"), szTitle,
WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0,
CW_USEDEFAULT, 0, NULL, NULL, hInstance,
NULL);

The MessagesAndThreads project creates a window in the original thread and then
starts a new thread.

31

WTL Developer’s Guide

hWnd = CreateWindow(szWindowClass, szTitle,
WS_OVERLAPPEDWINDOW,CW_USEDEFAULT, 0,
CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
HANDLE hMyThread;
DWORD MyThreadID;
hMyThread = CreateThread(NULL, 0,
(LPTHREAD_START_ROUTINE) MyThreadStartProc, hInstance,
0, &MyThreadID);

In the new thread it creates a new window. Both threads maintain their own message
loop. The rule is that the thread in which the window was created is used to queue
messages concerning that window for the application. If one thread for example stopped
processing messages (e.g. put in a long call to Sleep), then the messages for the
windows created in that thread would not be processed.
DWORD MyThreadStartProc(LPVOID hInstance){
HWND hWnd;
MSG msg;
hWnd = CreateWindow(szWindowClass, szTitle,
WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0,
CW_USEDEFAULT, 0, NULL, NULL,
(HINSTANCE)hInstance, NULL);
ShowWindow(hWnd, SW_SHOWNORMAL);
UpdateWindow(hWnd);
// Main message loop:
while (GetMessage(&msg, NULL, 0, 0)) {
if (!TranslateAccelerator(msg.hwnd,
hAccelTable, &msg)) {
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
return 0;// thread functioned correctly
}

Messages
The messages that are placed on message queues are identified by integers, and contain
two parameters, wParam and lParam. The senders of messages could be the kernel
code that manages the desktop/keyboard/mouse, the implementation of window
controls, your application code or ActiveX controls. The recipient of messages is
usually the window procedure associated with the window in which the action that
precipitated the message occurred, but in some circumstances it can be the parent
window.
Message IDs
The Platform SDK defines a range of messages related to generic windows, which are
named WM_XXX. This will work with normal windows and the standard/common
controls, which based on normal windows. So when a mouse is moved inside a
window, the window procedure for that window gets a WM_MOUSEMOVE message. This
is the case for all messages that begin with WM_, with three exceptions. WM_COMMAND,
WM_NOTIFY and the various WM_CTLCOLORXXX messages are sent to the parent of
the window in which the action took place.

32

WTL Developer’s Guide

Applications can send custom messages for application-specific communication. Values
should be added to WM_USER to form a message id, and then the message sent to a
window. The window procedure for that window will extract the message, subtract
WM_USER and interpret the message.
To shut down an application, the WM_QUIT message is used.
Standard Controls vs. Common Controls
Windows provides a set of standard controls that are built into the kernel and are always
available. These are buttons, combo-boxes, edit, listbox, rich-edit, scroll-bar and static
control.
Note that the Rich-Edit control is a standard control, not a common control. In comes in
three versions, each with varying degrees of support for rich text. Version 1 is
supported in “classic” Windows 95 and all later OSes. Version 2 is supported with
Windows 98 and Windows NT 4 and later, and may be installed on Windows 95
manually. Version 3 is available with Windows 2000 and Windows Me.
Windows also provides more advanced controls, known as common controls, which are
available to applications if they call the Platform SDK APIs InitCommonControls
or InitCommonControlsEx and if the system DLL Comctl32.dll with the correct
version is available. “Classic” Windows 95 came with a core set of common controls,
such as treeview, listview, and property pages. Later versions of Comctl32.dll, such as
that delivered with IE 4 (version 4.71) contains additional controls, such as IPAddress,
DateTime Picker, and MonthCalendar. IE5 does not add additional common controls,
but it does add some new messages for existing controls (e.g. TVM_GETLINECOLOR
for a treeview).
Getting/Setting Information
There is no real API to these controls. Instead, application code can send message to
them to set data, get data and perform other actions. For example, to limit the number
of characters permitted in an edit-box, an application could send it an
EM_SETLIMITTEXT message, with the limit as the wParam. To retrieve this limit,
the application can send EM_GETLIMITTEXT and the limit will be the return value
from the SendMessage function.
Commands and Notifications
Applications often wish to use the standard and common controls and be informed of
some of the messages for these controls. We have seen that an application cans subclass
and superclass windows, including those used with these controls. In this way
application code, rather than the default window proc for these controls can get first
access to all the incoming messages and pass on any messages not of interest to the
default window procedure.
For specialist cases this is fine, but it is considerable amount of work when you have
many dialog boxes, and many controls inside each. In general, we wish to use the
controls “as supplied”. What would be nice is if the dialog box (the parent of theses
controls in the window hierarchy) would be told of significant actions inside the

33

WTL Developer’s Guide

controls, such as a click in a button or text entry in an edit-box. Luckily for all of use,
this is exactly how the OS behaves. It sends WM_COMMAND and WM_NOTIFY messages
to the parent window with a notification message stating what happened in the child.
For each standard and common control, the OS decides that the parent of a control
would be informed that certain ranges of actions have occurred. The control itself might
be receiving a WM_LBUTTONDOWN message, but it passes a WM_COMMAND message to
the parent, with a BN_CLICKED notification message inside it. The standard controls
and the Animation common control use WM_COMMAND. All the other common controls
use WM_NOTIFY.
Which Windows are in the Hierarchy
One good piece of advice when programming windowing code is to be clear in your
own mind the hierarchy with which you are working. It can be the case that there are
additional or fewer windows present than you actually think – and messages that are
sent to parent windows might end up going to windows that are not expecting them.
Two sample scenarios where this can occur concern ActiveX controls. Such controls
can be implemented as “windowless”, in which case the control itself does not have its
own window, rather it rendered in the window of the dialog. When controls are hosted
in a client development environment, a host window class can wrap them, which itself
manages a window for the site, and this site window is situated in the window hierarchy
below the dialog window (it is a child of the dialog window) and above the control
window (it is the control window’s parent).
Ample use of the Spy++ utility can help you clarify the hierarchy arrangements and
save many hours of debugging.

Threads and Windowing
Threads and windows interact at a variety of levels. It is important we have a clear
understanding of these interactions, to avoid a number of potential problems and to
optimize performance.
On Win32, there is one and only one type of thread. It executes user-interface and nonuser interface code alike inside a threadproc and functions called from the threadproc.
There is no such thing as a “user-interface thread” from an OS perspective. Where the
difference can arise is in application space because you as the application developer
have decided that some code with UI calls will execute in some particular threads, and
other code, without UI calls, will execute in different threads. It is a design decision you
make and the OS is not interested. (The same argument applies to COM. There is no
such thing as a COM thread, just a Win32 thread, which happens to execute COM,
calls. What is different is that the rules of COM apartments also apply. )
Win32 lets you execute UI calls on any thread. You may decide for design reasons to be
more selective and limit UI calls to certain threads. How window operations work in the
context of threads needs to be appreciated so that appropriate design decisions may be
made.
Three excellent books with coverage of threads and windowing are Richter’s
“Programming Applications for Microsoft Windows“ – ISBN: 1-57231-996-8, chapter

34

WTL Developer’s Guide

26; Berveridge’s/Wiener’s “Multithreading Applications in Win32”, ISBN:
0-201-44234-5, chapter 11 and Cohen’s/Woodring’s “Win32 Multithreaded
Programming”, ISBN: 1-56592-296-4, chapter 12.
Who Owns the Objects?
An application can create objects such as windows, pens, brushes etc. It is considered
good programming practice to delete these when they are no longer necessary. Even if
this is not done, the kernel will step in and at thread deletion time or process deletion
time will delete the objects that are still in existence. Window objects (and hook
objects) are owned by the thread in which they were created. When the thread exits then
each window created within that thread will be closed immediately. All other objects
(pens, brushes, regions, device contexts, bitmaps, fonts, etc.) are owned by the process,
and regardless of which thread upon which they were created, they remain in existence
until the object is explicitly deleted or the process (not the creator thread) is terminated.
Threads and Windows and Message Queues
The golden rule of Win32 windowing/thread interaction is that the thread in which a
window was created (via a call to Win32’s CreateWindow[Ex]) is also the thread
that processes messages for that window. Most other aspects of Win32 windows/thread
interaction flow from this rule.
There are a number of questions for which we need to find answers, to understand the
implications of this rule. If a thread is going to receive window messages, then does it
need a message queue? How does this get constructed? How are messages placed on
the message queue? If a thread has a message queue, then it needs a message pump to
extract messages from the queue and process them. How is this implemented? When a
thread dies, then the windows associated with the thread might as well also die, as no
further messages can be sent to them – how is this handled? What happens if the thread
that has a message pump also wishes to wait on OS kernel objects, such as a mutex?
Some threads need a message queue and some do not. Some threads will never be used
for windowing, and it would be inefficient to allocate memory for a message queue
when it will not be used. Therefore, when the OS creates a thread, it does not
automatically create a message queue. The very first time a call is made in a new thread
to any of the Win32 windowing APIs, the OS creates the message queue. This is done
transparently to application code. Once the message queue is in existence, it remains so
until the thread dies.
The application needs to know which threads have a message queue and hence which
threads it will need to add the familiar GetMessage / DispatchMessage
message pump functions. The application knows this because the message queue only
gets created when windowing calls are made in that thread. Therefore it is a design
issue to decide where to make such calls and if present then to add the message pump.
Sending and Posting Messages
The two main techniques, which result in the window procedure getting executed, are
known as sending and posting. Sending is done via:
LRESULT SendMessage( // return is result of message processing

35

WTL Developer’s Guide

HWND hWnd,
UINT Msg,
WPARAM wParam,
LPARAM lParam

//
//
//
//

Window to receive the message
Message to send
Parameter to message
Parameter to message

);

You can think of sending as synchronous. The caller thread will block until the message
has actually being processed, and then the next line after the SendMessage in the
thread will be executed. The hWnd parameter intrinsically identifies the thread that has
the window procedure that needs to be executed. If this is the same thread as that on
which SendMessage was called, then SendMessage internally simply calls the
window procedure directly – without a thread context switch, and without placing the
message on the message queue. If a different thread provides the window procedure,
then a message is placed on its message queue. Some time later it processes the
message, and sends back the reply, and then the thread in which SendMessage was
called awakes and continues processing. It is possible to call SendMessage from one
thread to send a message to a window created on a separate thread but the one major
problem that can arise is if that separate thread is not servicing its message pump
regularly, the first thread can block.
Posting is done via one of three functions.
BOOL PostMessage(// Return is result of posting
HWND hWnd,
// Window to receive the message
UINT Msg,
// Message to send
WPARAM wParam, // Parameter to message
LPARAM lParam
// Parameter to message
);

This places the message on the message queue associated with the thread in which the
identified window was created.
BOOL PostThreadMessage( // Return is result of posting
DWORD idThread, // Thread to receive the message
UINT Msg,
// Message to send
WPARAM wParam, // Parameter to message
LPARAM lParam
// Parameter to message
);

This places the message on the message queue associated with the identified thread.
The window parameter when received will be 0.
VOID PostQuitMessage( // no return
int nExitCode
// Exit-code
);

This places a WM_QUIT on the message queue of the thread in which this call was
made. Sometime later, when the thread retrieves the WM_QUIT message off the queue,
it should exit gracefully.
Posting is asynchronous. The message is posted on the message queue and the function
returns immediately. It does not wait and the message may or may not be correctly
processed.

36

WTL Developer’s Guide

This normally works fine and enables different threads to communicate with each other,
without requiring synchronization. The recipient thread can, at a time convenient to
itself, take the message off its queue and process it in an orderly fashion.
One point of concern is what happens if the message queue does not exist – will a call
to one of the Post functions from a different thread create it. The answer is no. The
message queue for a particular thread will only get created inside that thread, when it
calls any windowing function. This is important in the context of WTL’s Multiple
Threads SDI Application, which we will discuss further in detail in the upcoming WTL
AppWizard chapter.
Threads and Kernel Objects
The GetMessage API is a blocking call, waiting for the next message to come in on
the message queue. The WaitForSingleObject or
WaitForMultipleObjects are blocking calls, waiting until one or more kernel
objects become signaled. What happens it we wish to wait on both kernel objects and
incoming window messages? For this the Win32 APIs
MsgWaitForMultipleObjects and MsgWaitForMultipleObjectsEx can
be used.
DWORD MsgWaitForMultipleObjects(
DWORD nCount,
// handle count
CONST HANDLE pHandles, // array of handles
BOOL fWaitAll,
// what waiting needs to be done
DWORD dwMilliseconds, // Timeout
DWORD dwWakeMask
// What input to detect
);
DWORD MsgWaitForMultipleObjectsEx(
DWORD nCount,
CONST HANDLE pHandles,
DWORD dwMilliseconds,
DWORD dwWakeMask,
DWORD dwFlags
// Additional Flags
);

They will wait on an array of kernel object handles and detect incoming messages. They
also have a timeout.

37

WTL Developer’s Guide

Chapter 3
ATL Windowing

Objectives
The objectives of this chapter are to:
•

Describe ATL’s windowing classes and templates

•

Cover ATL message maps and chaining

•

Explain how subclassing and superclassing work in ATL

•

Introduce the concept of a contained window

•

Explore ActiveX control containment within ATL’s dialogs

•

Examine the use of GDI functions with ATL windows

Overview
ATL provides comprehensive support for the full range of core windowing
functionality. This includes sub-classing, super-classing, handling windows messages
and notifications in a wide variety of means, and creating modal and modeless dialog
boxes (including ActiveX control containment).
ATL does not offer a C++ class for each type of Windows standard control (button,
combo box, edit-box, etc) or common control (list view, rebar, tree view, property
sheets). Rather, it offers classes and templates to construct generic windows and allows
you to specify the class name, which can be that of one of the standard or common
controls. The concept of message maps is used to dispatch incoming messages to C++
member functions.
ATL windowing is the foundation for the Windows Template Library (WTL). WTL
provides effective MFC UI-like C++ classes for each standard and common controls,
templates for GDI drawing and higher-level user interface application features, such as
toolbars, frames, views, statusbars, splitters and printing.
ATL provides limited wizard support. The ATL Object Wizard permits the creation of
dialog boxes with support ActiveX control containment. All other window creation –
dialog boxes without ActiveX control containment and generic windows, need to be
created manually in code by instantiating the appropriate ATL templates. Dialog
resources may be constructed and edited using Developer Studio’s standard
ResourceView. Through ClassView, you can access a wizard called New Windows
Message and Event Handler that enables you to add handler functions for particular
types of windows messages.

38

WTL Developer’s Guide

Windowing with ATL
ATL windowing provides an efficient layer of classes and templates above the lowlevel Win32 windowing APIs. The aim is to provide a C++ interface to windowing that
is compatible and well integrated with other aspects of ATL. You can use ATL
windowing on its own, but more likely it will be part of an application which also
exposes COM functionality. ATL windowing is needed to display standard windows
(e.g. displaying graphics or word-processing output), for dialog boxes (including but
not limited to those which contain ActiveX controls) and for the visual aspects of an
ActiveX control and its design-mode property sheets.
CWindow
The CWindow class is the main API between the client (the user of the window) and
the implementation.
CWindow manages a HWND and provides type-safe member function wrappers to
virtually all the APIs related to HWNDs.
A typical member function of CWindow is implemented as:
BOOL SetWindowText(LPCTSTR lpszString){
ATLASSERT(::IsWindow(m_hWnd));
return ::SetWindowText(m_hWnd, lpszString);
}

The m_hWnd member needs to be initialized. This is done in one of the Create
member function, which the client must call with appropriate parameters.
HWND Create(LPCTSTR lpstrWndClass, HWND hWndParent,
RECT& rcPos, LPCTSTR szWindowName = NULL, DWORD dwStyle = 0,
DWORD dwExStyle = 0, UINT nID = 0, LPVOID lpCreateParam=NULL){
m_hWnd = ::CreateWindowEx(dwExStyle,
lpstrWndClass, szWindowName,
dwStyle, rcPos.left,rcPos.top,
rcPos.right - rcPos.left,
rcPos.bottom - rcPos.top,
hWndParent, (HMENU)nID,
_Module.GetModuleInstance(),
lpCreateParam);
return m_hWnd;
}
HWND Create(LPCTSTR lpstrWndClass, HWND hWndParent,
LPRECT lpRect = NULL,
LPCTSTR szWindowName = NULL,
DWORD dwStyle = 0,
DWORD dwExStyle = 0,
HMENU hMenu = NULL,
LPVOID lpCreateParam = NULL) {
if(lpRect == NULL)
lpRect = &rcDefault;
m_hWnd = ::CreateWindowEx(dwExStyle,
lpstrWndClass, szWindowName,
dwStyle, lpRect->left,
lpRect->top,
lpRect->right - lpRect->left,
lpRect->bottom - lpRect->top,
hWndParent, hMenu,

39

WTL Developer’s Guide

_Module.GetModuleInstance(),
lpCreateParam);
return m_hWnd;
}

Effectively, the Create member functions are very thin wrappers around the Win32
API CreateWindowEx.
If we assume the client has already registered a window class with the name
szWindowClass using RegisterClassEx, a CWindow may be instantiated
using:
CWindow wnd;
wnd.Create(szWindowClass, 0, 0,
TEXT("CWindow Demo"),
WS_OVERLAPPEDWINDOW | WS_VISIBLE);
As we have seen, this results in a call to CreateWindowEx.
Sometimes a client has an existing HWND, and to use this via the CWindow wrappers it
can pass in the hwnd as a parameter to the CWindow constructor, and not bother to call
Create (the window already exists, so there is no need for an additional call to
CreateWindowEx).
CWindow wnd(hwnd);
wnd.SetWindowText(TEXT(“New Text”));
When handling ActiveX control containment, an additional template, called
CAxWindowT, is used to provide the extra support needed. We will cover this template
later.
Note that when a CWindow goes out of scope the HWND attached to it is NOT
destroyed (The CWindow class actually does not have a destructor). If you wish to
destroy the HWND associated with a CWindow you must directly call
CWindow::DestroyWindow.
From a client’s perspective, the use of CWindow is fine as it provides us with access to
the features we require.
Sample - WindowingWithATL
The WindowingWithATL project is a simple example showing how to use CWindow.
It was created using the ATL AppWizard and “Executable” was selected in the project
type options. No COM objects were inserted as we are focusing exclusively on the
windowing features. The AppWizard generated a file WindowingWithATL.cpp, and
this contained a WinMain with a message loop.
To use windowing we first need to include atlwin.h. A good place to put this in
stdafx.h, somewhere after the inclusion of .
// stdafx.h
#include 
#include 

40

WTL Developer’s Guide

In our WinMain function we add code to register a window class using the Win32
SDK API RegisterClassEx and then we create a window based on it using
CWindow.
TCHAR szWindowClass[]=TEXT("myapp");
WNDCLASSEX wcex;
wcex.cbSize
= sizeof(WNDCLASSEX);
wcex.style
= CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = (WNDPROC)MyWndProc;
wcex.cbClsExtra
= 0;
. . .
RegisterClassEx(&wcex);
CWindow wnd;
wnd.Create(szWindowClass, 0, 0, TEXT("CWindow Demo"),
WS_OVERLAPPEDWINDOW | WS_VISIBLE);

We must create a traditional window procedure, and to access the window through the
CWindow API we first instantiate a CWindow based on the HWND passed to our
message loop.
LRESULT CALLBACK MyWndProc(HWND hWnd, UINT message, WPARAM wParam,
LPARAM lParam){
PAINTSTRUCT ps;
HDC hdc;
TCHAR szHello[]=TEXT("Hello");
switch (message) {
case WM_PAINT:
. . . ;
case WM_LBUTTONDOWN:{
CWindow wnd(hWnd);
wnd.SetWindowText(TEXT("New Text"));
}
break;
. . .
}
return 0;
}

From an implementer’s perspective (and the person who implements a window is very
often the same person who is going to use it as a client), we have two problems: the
window class and the window procedure are still handled as with SDK programming.

Window Construction
A group of classes, templates and macros based around CWindowImpl and its parent,
CWindowImplBaseT, are provided to facilitate the implementation of more
advanced windowing functionality, such as window classes and window procedures.
With CWindowImpl, the implementer may add as much or as little functionality as
s/he desires, and leave the remaining to default to that provided by CWindowImpl.
Each important piece of functionality is separately represented and may easily be
configured as needed.

41

WTL Developer’s Guide

Window Traits
When creating windows numerous styles such as WS_CHILD or
WS_OVERLAPPEDWINDOW may be specified in the call to CreateWindowEx.
Instead of having to specify the same set of styles again and again throughout your
code, ATL provides a template called CWinTraits that stores the styles to use, and
they may be passed as a parameter to CWindowImpl when needed.
template 
class CWinTraits{
public:
static DWORD GetWndStyle(DWORD dwStyle)
{
return dwStyle == 0 ? t_dwStyle : dwStyle;
}
static DWORD GetWndExStyle(DWORD dwExStyle)
{
return dwExStyle == 0 ? t_dwExStyle:dwExStyle;
}
};

You specify the set of standard and extended styles when instantiating the template. The
two functions, GetWndStyle and GetWndExStyle return these values. Note that
both values default to 0 if not specified.
There are a number of frequently used sets of styles, and for these ATL provides
predefined CWinTraits. It also provides a NULL trait.
typedef CWinTraits
CControlWinTraits;
typedef CWinTraits
CFrameWinTraits;
typedef CWinTraits
CMDIChildWinTraits;
typedef CWinTraits<0, 0> CNullTraits;

There is an extra template called CWinTraitOR that logically ORs additional styles
with those already set in a CWinTraits.
template 
class CWinTraitsOR{
public:
static DWORD GetWndStyle(DWORD dwStyle)
{
return dwStyle | t_dwStyle |
TWinTraits::GetWndStyle(dwStyle);
}
static DWORD GetWndExStyle(DWORD dwExStyle){
return dwExStyle | t_dwExStyle |
TWinTraits::GetWndExStyle(dwExStyle);
}
};

The word “trait” is used in the ISO C++ standard when modeling separately a collection
of attributes of some other class/template. An example of its use is char_traits
which is a parameter to the string template basic_string. This explains why the

42

WTL Developer’s Guide

term is used in ATL when naming CWinTraits (CWinTraits is the only use of the
term in ATL in the current version, but this could change in future). The term is also
used in the Windows Template Library.
Window Class Information
ATL provides a class to manage the window class information. This is called
_ATL_WNDCLASSINFO and is implemented as follows:
struct _ATL_WNDCLASSINFOW{
WNDCLASSEX m_wc;
LPCTSTR m_lpszOrigName;
WNDPROC pWndProc;
LPCTSTR m_lpszCursorID;
BOOL m_bSystemCursor;
ATOM m_atom;
TCHAR m_szAutoName[13];
ATOM Register(WNDPROC* p){
return AtlModuleRegisterWndClassInfoW(
&_Module, this, p);
}
};

The string CWndClassInfo maps to an ASCII or Unicode version of
_ATL_WNDCLASSINFO as needed. Note that the third member, pWndProc, is a
pointer to the window procedure to use.
The Register member function first checks to see if the named class has already
been registered, and if not, then calls the Win32 API RegisterClassEx to register a
new class. If the specified class name is NULL then a new one is generated.
Application developers normally do not use CWndClassInfo directly – rather they
use one of three macros, DECLARE_WND_CLASS, DECLARE_WND_CLASS_EX or
DECLARE_WND_SUPERCLASS. The CWindowImpl template has
DECLARE_WND_CLASS(NULL) entry, so this is used if an application’s windowing
class does not have its own CWndClassInfo.
One of these macros can be added to a window implementation class. It provides a
function called GetClassInfo, which returns a reference to a static
CWndClassInfo structure defined within the function. The reason for three macros is
to allow you to influence how the CWndClassInfo gets initialized.
DECLARE_WND_CLASS creates a data member called wc of type CWndClassInfo
and fills in all the fields in the structure with default values, and fills in the name with
the supplied parameter. This name may be NULL, in which case ATL will generate a
name by combining the string “ATL:” and the string-ified address of the m_wc field.
#define DECLARE_WND_CLASS(WndClassName) \
static CWndClassInfo& GetWndClassInfo() { \
static CWndClassInfo wc = { \
{ sizeof(WNDCLASSEX), CS_HREDRAW | CS_VREDRAW|\
CS_DBLCLKS, StartWindowProc, 0, 0, NULL,\
NULL, NULL, (HBRUSH)(COLOR_WINDOW + 1),\
NULL, WndClassName, NULL }, \
NULL, NULL, IDC_ARROW, TRUE, 0, _T("") \

43

WTL Developer’s Guide

}

}; \
return wc; \

DECLARE_WND_CLASS_EX takes additional style and background color members and
uses them when filling in the structure.
#define DECLARE_WND_CLASS_EX(
WndClassName, style, bkgnd) \
static CWndClassInfo& GetWndClassInfo() { \
static CWndClassInfo wc = { \
{ sizeof(WNDCLASSEX), style, StartWindowProc, \
0, 0, NULL, NULL, NULL, (HBRUSH)(bkgnd + 1),\
NULL, WndClassName, NULL }, \
NULL, NULL, IDC_ARROW, TRUE, 0, _T("") \
}; \
return wc; \
}

DECLARE_WND_SUPERCLASS creates a new window class which is a superclass of
an existing class.
#define DECLARE_WND_SUPERCLASS(WndClassName,
OrigWndClassName) \
static CWndClassInfo& GetWndClassInfo() { \
static CWndClassInfo wc = { \
{ sizeof(WNDCLASSEX), 0, StartWindowProc, \
0, 0, NULL, NULL, NULL, NULL, NULL, WndClassName,
NULL }, OrigWndClassName,NULL,NULL,TRUE, 0,_T("") \
}; \
return wc; \
}

The third member is the window procedure to use and it is set to
StartWindowProc.
If an application wishes to provide specific settings, and these are not supported through
the macros, then there is no problem adding a direct implementation of
GetWndClassInfo. We will see this in action in the later ATLSplitter sample,
which uses a specific class cursor for the window (IDC_SIZEALL). It defines
GetWndClassInfo as follows:
// We wish to use a specific class cursor (IDC_SIZEALL),
// so we define our own implementation of GetWndClassInfo
static CWndClassInfo& GetWndClassInfo() {
static CWndClassInfo wc = {
{ sizeof(WNDCLASSEX), CS_HREDRAW | CS_VREDRAW|\
CS_DBLCLKS, StartWindowProc, 0, 0, NULL,
NULL, NULL, (HBRUSH)(COLOR_WINDOW + 1),
NULL, TEXT("MYSPLITTER"), NULL },
NULL, NULL, IDC_SIZEALL, TRUE, 0, _T("")
};
return wc;
}

Window Procedure using CMessageMap
To facilitate the processing of messages, ATL provides an abstract class called
CMessageMap and a range of macros to set up message maps. CMessageMap
declares one pure virtual function called ProcessWindowMessage.

44

WTL Developer’s Guide

class ATL_NO_VTABLE CMessageMap{
public:
virtual BOOL ProcessWindowMessage(HWND hWnd,
UINT uMsg, WPARAM wParam, LPARAM lParam,
LRESULT& lResult, DWORD dwMsgMapID) = 0;
};

If you think ProcesssWndMessage looks similar to the declaration of a window
procedure, you are quite right. Within the implementation of this function each
windowing class will have to react to incoming window messages, such as
WM_LBUTTONDOWN and WM_CHAR. The range of messages, which a window
implementation will wish, to track and their reaction to them varies according to the
role of the window and therefore each window which derives from CMessageMap
must implement its own version of this function.
If none are needed, then the DECLARE_EMPTY_MSG_MAP macro may be used.
#define DECLARE_EMPTY_MSG_MAP() \
public: BOOL ProcessWindowMessage(HWND, UINT, \
WPARAM, LPARAM, LRESULT&, DWORD) { return FALSE; }

Most applications will wish to detect certain messages, and we will shortly examine
how ATL supports this using message map macros. For now, think of these as setting
up a large switch statement, similar to those found in normal window procedures,
which test for a particular message type (e.g. WM_MOUSEMOVE) and call a C++ member
function when the message type is received.
As we have seen from the various DECLARE_WND_XXX macros, the window
procedure is specified as StartWindowProc. So how do we get our newly created
window to call its own ProcessWindowMessage?
StartWindowProc and Thunking
The window procedure is called StartWindowProc and is defined as a static
function. C++ static functions cannot directly call instance member functions (how
could they decide which of the potentially many instances to call?), but this is exactly
what we want our window procedure to do. To get around this, the idea of thunking is
used. We have a HWND and we need to map it to the object pointer, so we can calls its
non-static ProcessWindowMessage. To understand how this works we need to
take a small tour of the innards of ATL.
The CComModule maintains a linked list of _AtlCreateWndData, which
maintains pairs of object pointers and thread ids.
struct _AtlCreateWndData{
void* m_pThis;
DWORD m_dwThreadID;
_AtlCreateWndData* m_pNext;
};

CComModule has member functions AddCreateWndData to add to this linked list
and ExtractCreateWndData to retrieve the object pointer for the current thread.
Only one value per thread may be maintained at any one time.
CWindowImplBaseT is part of the inheritance tree when we use CWindowImpl.
One of its data members is:

45

WTL Developer’s Guide

CWndProcThunk m_thunk;

It is implemented as:
class CWndProcThunk{
union {
_AtlCreateWndData cd;
_WndProcThunk thunk;
};
void Init(WNDPROC proc, void* pThis);
}

The reason a union is used here is to conserve space. At any one time either the cd or
thunk field will be needed – they are never needed together.
To create a window, the Create member function of CWindowImplBaseT is
called, and this results in a call to:
AddCreateWndData(&m_thunk.cd, this);

Note that the m_thunk.cd parameter is memory for the node that becomes part of the
linked list, and this is the data that is put into it.
Then a call to the Win32 API CreateWindowEx is made. As the window procedure
for the window class is StartWindowProc, it is called with the first window
creation message (important: this is guaranteed to occur before the call to
CreateWindowEx returns).
Within StartWindowProc it calls CComModule::ExtractCreateWndData
to retrieve the object pointer for the current thread (and remove the node from the
linked list). The application code running in this thread is in a call to
CreateWindowEx, so there is no possibility that two windows within one thread
might be in the process of being created at the same time. As a window procedure,
StartWindowProc will receive the HWND as a parameter. It stores this value in the
object’s m_hwnd field.
The thunk is initialized, so that when called it will replace the HWND parameter on the
stack with a pointer to the object instance, and jump to the
CWindowImplBaseT::WindowProc method, which is also a static member
function of CWindowImplBaseT. Then using SetWindowLong(GWL_WNDPROC)
[or SetWindowLongPtr in the 32/64-bit version of ATL] the window procedure is
changed to call the thunk. Then StartWindowProc returns. StartWindowProc
will not be called again for this instance of the window object.
For all further messages for the window, the thunk will be called, and it replaces the
HWND on the stack (we have already stored a copy of it once in the object’s m_Hwnd),
and then jumps to executing WindowProc.
CWindowImplBaseT has a (non-static) method called WindowProc, which when
called will result in the class’ ProcessWindowMessage method being called, which
as we have seen results in the message map being processed. the
CWindowImplBaseT::WindowProc looks like a window procedure, and its first
parameter is by definition a HWND. However, because of the thunking, the value on the

46

WTL Developer’s Guide

stack was replaced with the pointer to the CWindowImplBaseT object. After a
suitable cast the object’s ProcessWindowMessage is called.
CWindowImplBaseT< TBase, TWinTraits >* pThis =
(CWindowImplBaseT< TBase, TWinTraits >*)hWnd;
BOOL bRet = pThis->ProcessWindowMessage(pThis->m_hWnd, uMsg,
wParam, lParam, lRes, 0);

CWindowImpl
To help facilitate the creation of C++ classes for windowing, ATL provide the
CWindowImpl template. This derives from CWindowImplBaseT, which in turn
derives from CWindowImplRoot, which finally derives from both CMessageMap
and CWindow. Note that the ProcessWindowMessage member function in
CMessageMap is abstract and therefore your implementation class must supply this
function.
The ATL Object Wizard does not provide wizard support to use CWindowImpl (but it
does for dialog boxes). It is easy enough to use it manually. You must create a C++
class which derives from CWindowImpl, write a constructor which calls Create and
a destructor which calls DestroyWindow, and add a message map which implements
ProcessWindowMessage.
There is a small syntactical issue you should be aware of. As your class is deriving from
the template CWindowImpl, and it final parameter is itself the templated
CWinTraits, at the end of your declaration you will have two template terminators
‘>’. You must leave a space between them – C++ has a quite different understanding of
the symbols ‘>’ and ‘ >>’. If you forget don’t worry – the compiler will kindly remind
you!
class CMyATLWindow : public CWindowImpl< CMyATLWindow, CWindow,
CWinTraits < WS_CAPTION | WS_POPUPWINDOW | WS_VISIBLE, 0> >{
public:
CMyATLWindow (){
RECT r={0, 0, 510, 510};
Create(0, r, TEXT("My Name"));
}
~ CMyATLWindow (){
if (m_hWnd)
DestroyWindow();
}
BEGIN_MSG_MAP(CMyATLWindow)
MESSAGE_HANDLER(WM_CHAR, OnChar)
END_MSG_MAP()

47

WTL Developer’s Guide

};

CWindow

CMessageMap

+rcDefault:RECT
+m_hWnd:HWND

+ProcessWindowMessage():BOOL {abstract}

TBase
CWndProcThunk

CWindowImplRoot

«union » +cd : _AtlCreateWndData cd
+thunk : _WndProcThunk

+m_pCurrentMsg : *MSG
+GetCurrentMessage
+ReflectNotifications
+DefaultReflectionHandler

CWindowImplBaseT

+Init(proc WNDPROC, * pThis void)

TBase
TWinTraits

+m_pfnSuperWindowProc : WNDPROC
+GetWndStyle() +GetWndExStyle()
+StartWindowProc() +WindowProc()
+Create() +DestroyWindow()
+SubclassWindow()
+UnsubclassWindow() DefWindowProc
OnFinalMessage

CWindowImpl

T TBase
TWinTraits

+wc : CWndClassInfo
+Create() : HWND

Message Maps
Now that we have seen how our object’s ProcessWindowMessage member
function gets called, we next want to examine how it reacts to incoming messages in
this function. ATL does not “crack” messages, so the same prototype is used for all
handlers. [Note that WTL does add crackers and has an additional macro,
BEGIN_MESSAGE_MAP_EX.]
ProcessWindowMessage will receive a very large number of messages and we
wish to detect only a small number of these and pass the remainder onto

48

WTL Developer’s Guide

DefWindowProc or equivalent for default processing. Within
ProcessWindowMessage we need to build a switch statement which for each
message we in which we are interested, we call a member function we write (e.g. for
WM_CHAR we might call MyOnCharFnc).
ProcessWindowMessage(HWND hwnd, UINT uMsg, . . .){
switch (msg) {
case WM_CHAR: MyOnCharFnc (); break;
case WM_PAINT: MyOnPaintFnc (); break;
default: defWindowProc();
}
}

For organizational reasons, we often wish a C++ class representing a parent window to
react to messages that were sent to a child window. This saves us having to write
separate C++ classes for each window, and is useful for subclassing and superclassing,
which we will cover later.
ProcessWindowMessage(HWND hwnd, UINT uMsg, . .,
DWORD dwMsgMapID){
switch (dwMsgMapID) {
case 0: // handle messages for this window
if (uMsg == WM_CHAR) MyOnCharFnc();
if (uMsg == WM_PAINT) MyOnPaintFnc();
break;
case 1: // handle messages for the first child window
if (uMsg == WM_LBUTTONDOWN) MyOnButtonDownFnc();
default: defWindowProc();
}
}

Constructing the Message Map
We do not wish to manually build up such a big switch statement. The macros
BEGIN_MSG_MAP and END_MSG_MAP set up an empty message map.
#define BEGIN_MSG_MAP(theClass) \
public: \
BOOL ProcessWindowMessage(HWND hWnd, UINT uMsg, WPARAM
wParam,LPARAM lParam,LRESULT& lResult,DWORD dwMsgMapID = 0){ \
BOOL bHandled = TRUE; \
//Following line avoids possible “not used” warnings
hWnd; uMsg; wParam; lParam; lResult; bHandled;
switch(dwMsgMapID) { \
case 0:
#define END_MSG_MAP() \
break; \
default: \
ATLTRACE2(atlTraceWindowing, 0,
_T("Invalid message map ID (%i)\n"), dwMsgMapID);\
ATLASSERT(FALSE); \
break; \
} \
return FALSE; \
}

49

WTL Developer’s Guide

Note that the last line of BEGIN_MSG_MAP is “case 0:” which indicates we are
switching on the message map id. The macro ALT_MSG_MAP sets up an alternative
message map id, after which we could place handlers for other windows.
#define ALT_MSG_MAP(msgMapID) \
break; \
case msgMapID:

Message Handlers
Next we need to populate our message maps with handlers, which can be done using a
number of macros. The simplest of these is MESSAGE_HANDLER, which takes two
parameters, a message and a function pointer, and if the message parameter is equal to
the messageID parameter to ProcessWindowMessage, then the function pointer is
called.
#define MESSAGE_HANDLER(msg, func) \
if(uMsg == msg) { \
bHandled = TRUE; \
lResult = func(uMsg, wParam, lParam, bHandled); \
if(bHandled) \
return TRUE; \
}

If we wish to call a handler if any one of a contiguous range of messages was received,
we could use MESSAGE_RANGE_HANDLER:
#define MESSAGE_RANGE_HANDLER(msgFirst, msgLast, func) \
if(uMsg >= msgFirst && uMsg <= msgLast) { \
bHandled = TRUE; \
lResult = func(uMsg, wParam, lParam, bHandled); \
if(bHandled) \
return TRUE; \
}

For each handler specified in the message map, you must create a handler function, with
the following prototype:
LRESULT OnMessagetype(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
return 0;
}

The return value from this function becomes the return value from the window
procedure. The final Boolean parameter, bHandled, can be set to FALSE to state that
the message has not been handled, and that the search for message handlers in the
message map should continue. Before your handler is called, the initial value of
bHandled is set to TRUE, which is what most handlers want, so if you are happy with
this there is no need to set it within your handler. Note that normally ATL does not
unpack the parameters to the window procedure. The wParam and lParam values
contain important information, which is particular to each message type, and you will
have to examine the Win32 Platform SDK documentation to discover what they
represent. ATL does not un-wrap them and present them to your function nicely
converted to specific data types.

50

WTL Developer’s Guide

The New Windows Message and Event Handlers Dialog
Though Developer Studio does not offer any Wizard support to create CWindowImplderived classes, it does provide one dialog box to help you populate their message
maps.
You will first have to manually create a class deriving it from CWindowImpl. Then in
ClassView (not the MFC ClassWizard!) select the class and right click to bring up the
context menu. Select the Add Windows Message Handler menu item, and the New
Windows Message and Event Handlers dialog will be displayed. This dialog enables
you to add handlers for different window messages and events.
The list of messages depends on the type of window that is selected. For example, if a
pushbutton is selected, handlers may be added to detect single and double button clicks.
The most common notifications are supported. For others, you will have to manually
add them to the message maps in code.

Rendering with ATL
ATL itself does not provide any templates to help with GDI programming. This is one
of the added benefits of using WTL. If developing windowing code using ATL only,
you will have to use the Win32 GDI functions such as LineTo etc.
Sample: ATLWithCWindowImpl
The ATLWithCWindowImpl project examines how to implement a C++ class that is
derived from CWindowImpl and has a message map containing a number of message
handlers.

51

WTL Developer’s Guide

The project enables to user to draw lines in a window, and including rubber banding as
the line is being constructed.
It was started by running the ATL AppWizard and creating an EXE. Then a new file
was created and the code for our class CMyATLWindow added.
class CMyATLWindow : public CWindowImpl< CMyATLWindow, CWindow,
CWinTraits < WS_CAPTION | WS_POPUPWINDOW | WS_VISIBLE, 0> >{
private:
CSimpleArray m_pointList;
bool m_bAddingLine;
POINT m_StartPoint;
POINT m_CurrentPoint;
public:
CMyATLWindow (){
RECT r={0, 0, 510, 510};
Create(0, r, TEXT("Draw Lines"));
bAddingLine = false;
}
~CMyATLWindow (){
if (m_hWnd)
DestroyWindow();
}
void OnFinalMessage(HWND hWnd){
::PostQuitMessage(0);
}

The m_pointList data member maintains an array of points of the lines in the
picture. This will be used when the picture needs to be redrawn. The m_bAddLine
Boolean data member is used while detecting mouse moves, and states if we are in the
middle of adding a line (and rubber-banding effect is needed). The m_StartPoint is
the beginning point in the line, and the m_CurrentPoint is the last point entered.
The constructor and destructor create and destroy the window, and the
OnFinalMessage function causes the entire single-threaded application to shut
down when this window is closed (for more substantial applications, you certainly will
not want to add this to every window!).
The message map detects four messages and calls suitable member functions for each.
BEGIN_MSG_MAP(CMyWindows)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
MESSAGE_HANDLER(WM_MOUSEMOVE, OnMouseMove)
MESSAGE_HANDLER(WM_LBUTTONUP, OnLButtonUp)
MESSAGE_HANDLER(WM_PAINT, OnPaint)
END_MSG_MAP()

The functions OnButtonDown, OnMouseMove and OnButtonUp keep track of the
line as it is being constructed and perform rubber banding.
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
m_StartPoint.x = LOWORD(lParam);
m_StartPoint.y = HIWORD(lParam);

52

WTL Developer’s Guide

m_CurrentPoint.x = LOWORD(lParam)+20;
m_CurrentPoint.y = HIWORD(lParam)+20;
HDC hdc = ::GetDC(m_hWnd);
SetROP2(hdc, R2_NOT);
MoveToEx(hdc, m_StartPoint.x, m_StartPoint.y, NULL);
LineTo(hdc, m_CurrentPoint.x, m_CurrentPoint.y);
m_bAddingLine = true;
::ReleaseDc(m_hWnd, hdc);
return 0;
}
LRESULT OnMouseMove(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
if (m_bAddingLine){
HDC hdc = ::GetDC(m_hWnd);
SetROP2(hdc, R2_NOT);
// First erase last line drawn
MoveToEx(hdc, m_StartPoint.x, m_StartPoint.y,NULL);
LineTo(hdc, m_CurrentPoint.x, m_CurrentPoint.y);
// Update curent point
m_CurrentPoint.x = LOWORD(lParam);
m_CurrentPoint.y = HIWORD(lParam);
// Draw new line
MoveToEx(hdc, m_StartPoint.x, m_StartPoint.y,NULL);
LineTo(hdc, m_CurrentPoint.x, m_CurrentPoint.y);
::ReleaseDc(m_hWnd, hdc);
}
return 0;
}
LRESULT OnLButtonUp(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
HDC hdc = ::GetDC(m_hWnd);
SetROP2(hdc, R2_NOT);
// First erase last line drawn
MoveToEx(hdc, m_StartPoint.x, m_StartPoint.y, NULL);
LineTo(hdc, m_CurrentPoint.x, m_CurrentPoint.y);
// Reset drawing mode to black
SetROP2(hdc, R2_BLACK);
m_bAddingLine = false;
// Add points to array
m_CurrentPoint.x = LOWORD(lParam);
m_CurrentPoint.y = HIWORD(lParam);
m_pointList.Add(m_StartPoint);
m_pointList.Add(m_CurrentPoint);
// Draw newly added line
MoveToEx(hdc, m_StartPoint.x, m_StartPoint.y, NULL);
LineTo(hdc, m_CurrentPoint.x, m_CurrentPoint.y);
::ReleaseDc(m_hWnd, hdc);
return 0;
}

53

WTL Developer’s Guide

The OnPaint member function is called whenever the picture in the window needs to
be refreshed. It simply runs through the list of points, and for each pair moves to the
first and calls LineTo to the second.
LRESULT OnPaint(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled)
{
PAINTSTRUCT ps;
HDC hdc = (wParam != NULL) ?
reinterpret_cast(wParam) :
::BeginPaint(m_hWnd, &ps);
int numLines = m_pointList.GetSize() / 2;
int counter=0;
while (counter< numLines)
{
POINT ptbegin = m_pointList[counter*2];
POINT ptend = m_pointList[counter*2+1];
MoveToEx(hdc, ptbegin.x, ptbegin.y, NULL);
LineTo(hdc, ptend.x, ptend.y);
counter++;
}
if (wParam == NULL)
::EndPaint(m_hWnd, &ps);
return 0;
}
};

To instantiate your new class add this line to your WinMain:
CMyATLWindow wnd;

Command and Notification Handlers
In general ATL does not “crack” messages for us (WTL does!), but ATL does offer
some special help for commands (WM_COMMAND) and notifications (WM_NOTIFY).
Commands are sent to parent windows when, for example, menu items in the window
menu are selected or child pushbuttons are pressed. Notifications are used by the
common controls to inform the application in details about what is happening with
these controls.
You write your command handlers as follows:
LRESULT CommandHandler(WORD wNotifyCode, WORD wID,
HWND hWndCtl, BOOL& bHandled);

Note that here ATL does break out the notification code and the command id.
ATL provides four macros to support commands. Each macro tests that the uMsg field
is set to WM_COMMAND and another condition, and if so it calls the final parameter,
which is your command handler. The “other condition” is what distinguishes the four
macros from each other.
#define COMMAND_HANDLER(id, code, func) \
if(uMsg == WM_COMMAND && id == LOWORD(wParam)
&& code == HIWORD(wParam)) { \
bHandled = TRUE; \
lResult = func(HIWORD(wParam), LOWORD(wParam),

54

WTL Developer’s Guide

}

(HWND)lParam, bHandled); \

if(bHandled) \
return TRUE; \

COMMAND_HANDLER maps a control id and notification to a command handler.
#define COMMAND_ID_HANDLER(id, func) \
if(uMsg == WM_COMMAND && id == LOWORD(wParam)) { \
bHandled = TRUE; \
lResult = func(HIWORD(wParam), LOWORD(wParam),
(HWND)lParam, bHandled); \
if(bHandled) \
return TRUE; \
}

COMMAND_ID_HANDLER maps the control id only.
#define COMMAND_CODE_HANDLER(code, func) \
if(uMsg == WM_COMMAND && code == HIWORD(wParam)) { \
bHandled = TRUE; \
lResult = func(HIWORD(wParam), LOWORD(wParam),
(HWND)lParam, bHandled); \
if(bHandled) \
return TRUE; \
}

COMMAND_CODE_HANDLER maps the notification code only.
#define COMMAND_RANGE_HANDLER(idFirst, idLast, func) \
if(uMsg == WM_COMMAND && LOWORD(wParam) >= idFirst
&& LOWORD(wParam) <= idLast) { \
bHandled = TRUE; \
lResult = func(HIWORD(wParam), LOWORD(wParam),
(HWND)lParam, bHandled); \
if(bHandled) \
return TRUE; \
}

COMMAND_RANGE_HANDLER maps a range of control ids.
Notifications are normally used by the common controls to tell their parent of
significant events. The prototype for notification handlers is as follows:
LRESULT NotifyHandler(int idCtrl, LPNMHDR pnmh, BOOL& bHandled);

Note that the third parameter is of type pointer to NMHDR. Depending on the control
involved, this can be cast to a larger control-specific header. For example, the
DateTimePicker notification header would be:
typedef struct tagNMDATETIMECHANGE {
NMHDR nmhdr;
DWORD dwFlags;
SYSTEMTIME st;
} NMDATETIMECHANGE;

As the first field it NMHDR if can safely be accessed through a pointer to NMHDR and if
such a pointer were cast to a pointer to NMDATETIMECHANGE, the other fields could
be accessed. Hence very rich information can be supplied to the application. The four
notification macros are similar to the command macros, in that the first checks for a
control id and a notification code, a control id only, a notification only and a range of
control ids.
#define NOTIFY_HANDLER(id, cd, func) \
if(uMsg == WM_NOTIFY && id == ((LPNMHDR)lParam)->idFrom && cd ==

55

WTL Developer’s Guide

((LPNMHDR)lParam)->code) { \
bHandled = TRUE; \
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
if(bHandled) return TRUE; \
}
#define NOTIFY_ID_HANDLER(id, func) \
if(uMsg == WM_NOTIFY && id == ((LPNMHDR)lParam)->idFrom) { \
bHandled = TRUE; \
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
if(bHandled) \
return TRUE; \
}
#define NOTIFY_CODE_HANDLER(cd, func) \
if(uMsg == WM_NOTIFY && cd == ((LPNMHDR)lParam)->code) { \
bHandled = TRUE; \
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
if(bHandled) \
return TRUE; \
}
#define NOTIFY_RANGE_HANDLER(idFirst, idLast, func) \
if(uMsg == WM_NOTIFY && ((LPNMHDR)lParam)->idFrom >= idFirst &&
((LPNMHDR)lParam)->idFrom <= idLast) { \
bHandled = TRUE; \
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
if(bHandled) \
return TRUE; \
}

Sample : ATLCommandsAndNotifications
The ATLCommandsAndNotifications project demonstrates the use of command and
notification handlers. It uses a menu to generate the commands and a tree view to
generate the notifications.
class CMyCommandAndNotifWindow : public CWindowImpl<
CMyCommandAndNotifWindow, CWindow,
CWinTraits < WS_CAPTION | WS_POPUPWINDOW | WS_VISIBLE, 0> >{
public:
CWindow m_ctlTreeview;
CMyCommandAndNotifWindow (){
RECT r={0, 0, 510, 510};
HMENU hMenu = LoadMenu(_Module.GetResourceInstance(),
MAKEINTRESOURCE(IDR_MY_MENU));
Create(0, r,
TEXT("ATL Command and Notification Demo"),0,0,(UINT)hMenu);
RECT rc={20,20,150,300};
m_ctlTreeview.Create(WC_TREEVIEW, m_hWnd, &rc,
TEXT("CWindow Demo"),
WS_CHILD | WS_VISIBLE | TVS_HASLINES | TVS_LINESATROOT);
RECT rc1={200,20,500,50};
TV_INSERTSTRUCT TreeCtrlItem;
TreeCtrlItem.hInsertAfter = TVI_LAST;
TreeCtrlItem.item.mask = TVIF_TEXT | TVIF_PARAM;
TreeCtrlItem.item.pszText = "First Entry";
TreeCtrlItem.item.lParam = 0;
TreeCtrlItem.hParent = TVI_ROOT;
m_ctlTreeview.SendMessage(TVM_INSERTITEM, 0,
(LPARAM)&TreeCtrlItem);

56

WTL Developer’s Guide

TreeCtrlItem.item.pszText = "Second Entry";
TreeCtrlItem.hParent = TVI_ROOT;
m_ctlTreeview.SendMessage(TVM_INSERTITEM, 0,
(LPARAM)&TreeCtrlItem);
}
BEGIN_MSG_MAP(CMyCommandAndNotifWindow)
COMMAND_ID_HANDLER(ID_BEEP, OnBeep)
COMMAND_ID_HANDLER(ID_RELOCATE, OnRelocate)
NOTIFY_CODE_HANDLER(TVN_SELCHANGED, OnSelChange)
END_MSG_MAP()

Simple implementations of the handlers are provided to verify that they are executed.
The OnBeep handler calls the Win32 Beep API, the OnRelocate handler calls
CWindow::MoveWindow to reposition the window and the OnSelChange handler
outputs some debug text to show that the notification has been received.
LRESULT OnBeep(WORD wNotifyCode, WORD wID, HWND hWndCtl,
BOOL& bHandled){
OutputDebugString("OnBeep called\n");
Beep(100,100);
return 0;
}
LRESULT OnRelocate(WORD wNotifyCode, WORD wID, HWND hWndCtl,
BOOL& bHandled){
OutputDebugString("OnRelocate called\n");
MoveWindow( 200, 200, 600, 600, TRUE );
return 0;
}

Message Map Chaining
One of the great features of the MFC architecture is the way command and message
routing occurs. The class representing the window in which a message is received need
not be the one that handles that message. It is possible to forward to message to another
class, which might be better suited. With MFC much of this is done automatically.
ATL offers similar functionality and it is known as message map chaining. A number of
macros are provided and these can be inserted manually as appropriate. ATL provides
five macros and one C++ class (CDynamicChain) that are used to allow
implementers to flexibly route messages to different classes.
CHAIN_MSG_MAP and CHAIN_MSG_MAP_ALT redirect messages from a derived
class to be processed in the message map of a parent class. CHAIN_MSG_MAP redirects
the message to the parent’s default message map, and CHAIN_MSG_MAP_ALT
() redirects messages to the alternate message map identified by the
number parameter. Be clear that CHAIN_MSG_MAP should appear in the derived class’
message map, and the parameter to the macro, theChainClass, must be a base
class.
#define CHAIN_MSG_MAP(theChainClass) { \
if(theChainClass::ProcessWindowMessage(hWnd, uMsg,\
wParam, lParam, lResult)) \
return TRUE; \
}

57

WTL Developer’s Guide

#define CHAIN_MSG_MAP_ALT(theChainClass, msgMapID) { \
if(theChainClass::ProcessWindowMessage(hWnd, uMsg,\
wParam, lParam, lResult, msgMapID)) \
return TRUE; \
}

CHAIN_MSG_MAP_MEMBER and CHAIN_MSG_MAP_ALT_MEMBER are used when a
class wishes to redirect incoming messages to another class that is a data member of it.
(This other class must itself implement ProcessWindowMessage).
CHAIN_MSG_MAP_MEMBER redirects to the default message map whereas
CHAIN_MSG_MAP_ALT_MEMBER() redirects to the specified alternate.
#define CHAIN_MSG_MAP_MEMBER(theChainMember) { \
if(theChainMember.ProcessWindowMessage(hWnd, uMsg,\
wParam, lParam, lResult)) \
return TRUE; \
}
#define CHAIN_MSG_MAP_ALT_MEMBER(theChainMember,msgMapID){\
if(theChainMember.ProcessWindowMessage(hWnd, uMsg,\
wParam, lParam, lResult, msgMapID)) \
return TRUE; \
}

With these four chaining macros there must be either an inheritance or a containment
relationship between the classes. What happens if we wish to direct messages between
classes that are not related, or if we wish to change the destination class on the fly? This
is supported through the use of dynamic chaining, which uses the CHAIN_MSG_MAP
macro and the CDynamicChain class.
#define CHAIN_MSG_MAP_DYNAMIC(dynaChainID) { \
if(CDynamicChain::CallChain(dynaChainID, hWnd, uMsg,\
wParam, lParam, lResult)) \
return TRUE; \
}

CDynamicChain is a simple class that manages an array of mapping from a chain id
(an integer identifier) to a pointer to a CMessageMap and a map id (which will be 0 if
the default map is to be used, or a greater number if an alternate map should be used).
Entries can be added to the array using CDynamicChain::SetChainEntry and
removed with CDynamicChain::RemoveChainEntry. The
CDynamicChain::CallChain finds the CMessageMap object associated with a
chain id, and calls its ProcessWindowMessage function.
The class you write which is to receive the redirected message must be derived from
CMessageMap and have its own message map. The class you write which receives the
message must be derived from CDynamicChain and must call
CDynamicChain::SetChainEntry passing in the recipient class and a chain id.
Then when CHAIN_MSG_MAP_DYNAMIC is used in the message map, the chain id is
used to lookup the CDynamicChain array and the ProcessWindowMessage
function called in the appropriate recipient.
Sample: WindowMessageChaining
The WindowMessageChaining project explores the use of ATL message map chaining.

58

WTL Developer’s Guide

Base class chaining is performed by creating two classes, CMyBaseWindow and
derived from it, CMyDerivedWindow. Both implement messages maps – usually
these would contain different messages but to clearly indicate which piece of code
handles the message, we have both of them handling WM_LBUTTONDOWN, and the
handler for this prints out a string to the output window (when you are running it in the
debugger).
The message map for the derived class contains two entries, one specifying a base
chaining to the base class, and the other specifies a normal message handler. When the
derived class receives a WM_LBUTTONDOWN message, whichever of these entries
comes first will get to process it, so if CHAIN_MSG_MAP(CMyBaseWindow) is first
the output window shows "CMyBaseWindow::OnLButtonDown called” whereas if the
MESSAGE_HANDLER (WM_LBUTTONDOWN, OnLButtonDown) entry came first
the output window would show “CMyDerivedWindow::OnLButtonDown called".
class CMyBaseWindow : public CWindowImpl< CMyBaseWindow ,
CWindow, CWinTraits >{
public:
BEGIN_MSG_MAP(CMyBaseWindow)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
OutputDebugString(TEXT(
"CMyBaseWindow::OnLButtonDown called\n"));
return 0;
}
};
class CMyDerivedWindow : public CmyBaseWindow {
public:
BEGIN_MSG_MAP(CMyDerivedWindow)
CHAIN_MSG_MAP(CMyBaseWindow)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
OutputDebugString(TEXT(
"CMyDerivedWindow::OnLButtonDown called\n"));
return 0;
}
};

Chaining to a contained class member is performed using the CMyContainedClass
and CMyContainerWindow classes. CMyContainedClass derives from
CMessageMap, and its map detects WM_LBUTTONDOWN and the handler prints out a
string for this. CMyContainerWindow is a normal window and as a data member
has an instance of CMyContainedClass. It its message map it calls
CHAIN_MSG_MAP_MEMBER. When CMyContainerWindow is instantiated and the
mouse button clicks in the window, the string “CMyContainedClass::OnLButtonDown
called” is output.
class CMyContainedClass : public CMessageMap {
public:

59

WTL Developer’s Guide

BEGIN_MSG_MAP(CMyContainedClass)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam,
BOOL& bHandled){
OutputDebugString(TEXT(
"CMyContainedClass::OnLButtonDown called\n"));
return 0;
}
};
class CMyContainerWindow : public CWindowImpl< CMyContainerWindow ,
CWindow,
CWinTraits < WS_CAPTION | WS_POPUPWINDOW | WS_VISIBLE, 0> >
{
public:
CMyContainedClass contained;
BEGIN_MSG_MAP(CMyContainerWindow)
CHAIN_MSG_MAP_MEMBER(contained)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam,
BOOL& bHandled){
OutputDebugString(TEXT(
"CMyContainerWindow::OnLButtonDown called\n"));
return 0;
}
};

Dynamic chaining is shown using the classes CMyDestinationClass (which is
derived from CMessageMap) and CMySourceWindow (which is derived from
CDynamicChain). These two classes are not related by inheritance or containment, as
with the other chaining techniques. To set up the connection between instances of the
classes, the following code is added (e.g. in WinMain)
CMySourceWindow wnd3;
CMyDestinationClass DynamicDestination;
wnd3.SetChainEntry(1, &DynamicDestination, 0);

Note that the first parameter to SetChainEntry is the chain id – and if multiple calls
are may to SetChainEntry with the same id then the latter call replaces the previous
setting. The third parameter is the message map id, 0 for the default or a higher number
for alternates. If you use a higher number and no alternate of that number exists, an
ASSERT fails within the ATL code.
The implementation of the CMyDestinationClass consists of a message map that
detects WM_LBUTTONDOWN and the handler that outputs a string when it is detected.
class CMyDestinationClass : public CMessageMap {
public:
BEGIN_MSG_MAP(CMyDestinationClass)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){

60

WTL Developer’s Guide

};

}

OutputDebugString(TEXT(
"CMyDestinationClass::OnLButtonDown called\n"));
return 0;

The implementation of CMySourceWindow contains a message map which dynamic
chains chainID 1. Its OnLButtonDown is never called.
class CMySourceWindow : public CWindowImpl< CMySourceWindow,
CWindow, CWinTraits >,
public CDynamicChain {
public:
BEGIN_MSG_MAP(CMySourceWindow)
CHAIN_MSG_MAP_DYNAMIC (1)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
OutputDebugString(TEXT(
"CMySourceWindow::OnLButtonDown called\n"));
return 0;
}
};

When this code is run and the mouse clicked in the window the output is
CMyDestinationClass::OnLButtonDown called, which shows that the chaining has
worked.

Subclassing and Superclassing
CWindowImpl supports superclassing and instance subclassing.
To superclass a window you use the DECLARE_WND_SUPERCLASS instead of
DECLARE_WND_CLASS to define the class information.
DECLARE_WND_SUPERCLASS takes two class names as parameters – the name of the
new class, and the name of an existing class. During class registration (which is
initiated in CWindowImpl::Create) the class information for the existing class is
accessed, and its window procedure recorded in the data member
CWindowImplBaseT::m_pfnSuperWindowProc.
Later, when messages are being processed, if a particular message has no entry in the
message map it is forwarded to the
CWindowImplBaseT::m_pfnSuperWindowProc for processing by the base
class.
Subclassing of an existing window is performed in CWindowImplBaseT::
SubclassWindow(HWND hWnd), where the HWND parameter represents the
window to be subclassed. A CWindowImpl expects to manage a single HWND, so
either SubclassWindow (which uses an existing window) or Create (which
creates a new one without subclassing) should be called, but it is an error to call both.
The implementation of SubclassWindow calls SetWindowLong(hWnd,
GWL_WNDPROC) to replace the window procedure in use for the existing window. The
old window procedure is returned, and this is stored in

61

WTL Developer’s Guide

CWindowImplBaseT::m_pfnSuperWindowProc , and is called if messages are
not processed by our message map.
ATL does not support global subclassing (a search of the ATL source tree for
SetClassLongPtr or SetClassLong finds no matches).
Sample: SubClassAndSuperClass
The SubclassAndSuperclass project experiments with superclassing and subclassing in
ATL. The CMySuperWindow class superclasses the BUTTON control. It detects
WM_LBUTTONDOWN messages and prints out a string when they arrive.
class CMySuperWindow : public CWindowImpl< CMySuperWindow ,
CWindow,CWinTraits >{
public:
DECLARE_WND_SUPERCLASS ("MySuperWindow", "BUTTON")
CMySuperWindow (){
RECT r={0, 0, 510, 510};
Create(0, r, TEXT("MY SUPERCLASSED WINDOW"));
}
BEGIN_MSG_MAP(CMySuperWindow)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
OutputDebugString(TEXT(
"CMySuperWindow::OnLButtonDown called\n"));
return 0;
}
};

To show subclassing we first have to have an existing window. For this we write
CMyChildWindow. Note that it detects WM_LBUTTONDOWN and WM_LBUTTONUP
messages. Also note that its constructor calls Create, to really create a new window,
whose parent is a HWND we pass in.
class CMyChildWindow : public CWindowImpl< CMyChildWindow , CWindow,
CWinTraits < WS_CHILD | WS_VISIBLE, 0> >{
public:
CMyChildWindow (HWND hParent){
RECT r={20, 20, 150, 150};
Create(hParent, r, TEXT("MY CHILD WINDOW"));
}
BEGIN_MSG_MAP(CMyChildWindow)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
MESSAGE_HANDLER(WM_LBUTTONUP, OnLButtonUp)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
OutputDebugString(TEXT(
"CMyChildWindow::OnLButtonDown called\n"));
return 0;
}

62

WTL Developer’s Guide

LRESULT OnLButtonUp(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
OutputDebugString(TEXT(
"CMyChildWindow::OnLButtonUp called\n"));
return 0;
}
};

To subclass CMyWindow we write another class called CMySubClassedWindow. In
its constructor it takes in a HWND and calls SubclassWindow to subclass the window
identified by the HWND parameter. The message map for CMySubClassedWindow
detects the WM_LBUTTONDOWN message.
class CMySubclassedWindow : public CWindowImpl< CMySubclassedWindow ,
CWindow,
CWinTraits < WS_CHILD | WS_VISIBLE, 0> >{
public:
CMySubclassedWindow (HWND hSubclassThisWindow){
RECT r={20, 20, 150, 150};
SubclassWindow(hSubclassThisWindow);
}
BEGIN_MSG_MAP(CMySubclassedWindow)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
END_MSG_MAP()
LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
OutputDebugString(TEXT(
"CMySubclassedWindow::OnLButtonDown called\n"));
return 0;
}
};

To use both these classes we need to add the following (e.g. in WinMain):
CMyChildWindow wnd1(hWnd);
CMySubclassedWindow wnd2(wnd1.m_hWnd);

We first instantiate CMyChildWindow and then we subclass it in
CMySubclassedWindow. Now when we run this program and press the mouse
button down and up in the window, the output is:
CMySubclassedWindow::OnLButtonDown called
CMyChildWindow::OnLButtonUp called

Which is what we expect.

CContainedWindow
When you wish to process messages concerning a child window in the message map of
a parent window, you can use CContainedWindow. You may either create the
window within CContainedWindow or subclass an existing window. When
initializing the contained window you must specify an object that implements
CMessageMap and which will act as the parent window. You must also specify an
alternative message map id, which will be used when passing child messages to the
parent.

63

WTL Developer’s Guide

When using CContainedWindow you do not derive a class from it, rather you create
a data member of type CContainedWindow in another class (usually the parent
class), and then you initialize the data member and call Create (which super-classes the
window) or SubclassWindow.
To initialize a CContainedWindow, you call Create. CContainedWindow has
a number of constructors and a number of Create functions with different signatures.
It is important that you use a combination of one of each that supplies a pointer to the
parent object which implements CMessageMap, the class name of the contained
window and the alternate message map id.
Sample: ContainedWindowSample
The ContainedWindowSample shows the use of CContainedWindow. It creates two
instances of CContainedWindow, one using superclassing and the other using
subclassing. Both redirect their messages to their parent’s message map, but to different
map ids.
class CMyWindow : public CWindowImpl< CMyWindow , CWindow,
CWinTraits < WS_CAPTION | WS_POPUPWINDOW | WS_VISIBLE,
WS_EX_WINDOWEDGE> > {
public:
HWND m_hToSubclassAsContainedWindow;
CMyWindow (HWND hToSubclassAsContainedWindow)
:Subclassed(this, 2){
RECT r={0, 0, 510, 510};
m_hToSubclassAsContainedWindow =
hToSubclassAsContainedWindow;
Create(0, r, TEXT("MY WINDOW"));
}
CContainedWindow SuperClassed;
CContainedWindow Subclassed;
BEGIN_MSG_MAP(CMyWindow)
MESSAGE_HANDLER(WM_CREATE, OnCreate)
ALT_MSG_MAP(1)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDownSuperClassed)
ALT_MSG_MAP(2)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDownSubclassed)
END_MSG_MAP()
LRESULT OnLButtonDownSuperClassed(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled)
{
OutputDebugString(TEXT(
"CMyWindow::OnLButtonDown called for superclassed\n"));
return 0;
}
LRESULT OnLButtonDownSubclassed(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled)
{
OutputDebugString(TEXT(
"CMyWindow::OnLButtonDown called for subclassed\n"));
return 0;
}
LRESULT OnCreate(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){

64

WTL Developer’s Guide

RECT r={50, 50, 210, 210};
OutputDebugString("OnCreate\n");
SuperClassed.Create("edit", this, 1, m_hWnd, &r);
Subclassed.SubclassWindow(
m_hToSubclassAsContainedWindow);
return 0;
}
};

Higher-level UI
It is possible to build more sophisticated functionality above the ATL windowing
foundation. Potential areas of interest could be splitter windows, command bars,
hyperlink functionality or advanced controls such as bitmap buttons.
ATL does provide you with all the facilities to build such UI constructs. We will now
briefly look at one sample, to show it is possible. But the question must be asked - why
bother with this extra effort? WTL provides this type of functionality in a well-thought
out library of templates, and hence instead of coding such UI yourself, it would be
much better to use WTL directly.
Sample: ATLSplitter

The ATLSplitter sample shows how a splitter window could be implemented. It
consists of two windows, a workspace window, and as a child, a splitter, which is used
to divide the workspace.
The project contains a splitter class that has methods to react to button down and up
events and mouse menus, similar to what we saw in the ATLWithCWindowImpl
sample.
class CMySplitter : public CWindowImpl< CMySplitter, CWindow,
CWinTraits < WS_CHILD | WS_VISIBLE> >
{
public:
enum Orientation {
ORIENTATION_VERT,
ORIENTATION_HORIZ
};
private:
RECT m_rcLocation;
// dimensions of splitter itself
RECT m_rcBoundingBox;// dimensions of workspace being split

65

WTL Developer’s Guide

int

m_CenterPoint;
// offset from 0 in parent's
// co-ordinates for center of splitter
Orientation m_Orientation;// whether splitter is vert or horiz
HWND m_hParentWnd;
// Handle to parent window
bool m_bBeingMoved;
// Are we in the middle of a move
// WM_MOUSEMOVE handling is only done
// when true
int
m_parentCoord;
// Location within parent of splitter
public:
BEGIN_MSG_MAP(CMySplitter)
MESSAGE_HANDLER(WM_PAINT, OnPaint)
MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown)
MESSAGE_HANDLER(WM_LBUTTONUP, OnLButtonUp)
MESSAGE_HANDLER(WM_MOUSEMOVE, OnMouseMove)
END_MSG_MAP()
. . .
};

The workspace class contains one instance of the splitter class and must initialize it to
the correct dimensions.
CMySplitter m_VerticalSplitter;
. . .
m_VerticalSplitter.Init(m_hWnd,CMySplitter::ORIENTATION_VERT,
200, rcBoundingBox);

When the workspace is resized, it must inform the splitter.
LRESULT OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam,
BOOL& bHandled){
int ClientAreaWidth = LOWORD(lParam);
int ClientAreaHeight = HIWORD(lParam);
if (m_VerticalSplitter.IsWindowVisible()) {
RECT rcBoundingBox;
GetClientRect(&rcBoundingBox);
m_VerticalSplitter.SetBoundingBox(rcBoundingBox);
m_VerticalSplitter.GetFirstPane(rcBoundingBox);
// here would need to update contents of workspace
}
return 0;
}

Dialog Boxes
In addition to providing classes that manage windows, ATL also provides a range of
classes for dialog boxes. CDialogImpl is used to manage dialog boxes and just like
with CWindowImpl, you must derived your class from it manually. You must also
manually create a dialog template using Developer Studio’s ResourceView. The ATL
Object Wizard lets you create a class derived from CAxDialogImpl and
automatically adds a default dialog template for it. The difference between
CDialogImpl and CAxDialogImpl is that the latter supports ActiveX control

66

WTL Developer’s Guide

containment, which is an excellent feature if you plan to add ActiveX controls to the
dialog, but is unnecessary overhead (though not much) if not needed.
Once you have a class derived from one of the ATL dialog templates, you may add
message handlers to it and expand the message map, and edit the dialog templates with
ResourceView.
CSimpleDialog can be used where you need to construct a dialog based on a dialog
template, initialize fields within OnInitDialog, then display the dialog, react to
messages, and then handle OK and Cancel selections. CSimpleDialog has its own
predefined message map, in which the WM_INITDIALOG causes OnInitDialog to
be called, and either IDOK or IDCANCEL causes ::EndDialog to be executed. If
you wish to handle other message or commands you can either add your own message
map in your derived class or you can use CDialogImpl/CAxDialogImpl.
An important point to note if adding your message map to your derived class is that by
default the message map in CSimpleDialog itself will then not be used – if you are
interested in detecting WM_INITDIALOG, IDOK or IDCANCEL, you must either add
appropriate entries to your message map or chain the one in CSimpleDialog.
CDialogImpl is available for more sophisticated needs. It does not have a default
message map.
Sample : SimpleModalAndModeless
This project shows how to use CSimpleDialog, and in both modal and modeless
forms how to use CDialogImpl.
The MySimpleDialog class is derived from CSimpleDialog and contains its own
message map. It has a checkbox that in OnInitDialog is initialized to true, and in
OnOk the current value is retrieved.
class MySimpleDialog : public CSimpleDialog{
public:
MySimpleDialog(): m_bButton(BST_CHECKED){}
BEGIN_MSG_MAP(CMainDialog)
COMMAND_ID_HANDLER(IDOK, OnOk)
COMMAND_ID_HANDLER(IDCANCEL, OnCloseCmd)
MESSAGE_HANDLER(WM_INITDIALOG, OnInitDialog)
END_MSG_MAP()
UINT m_bButton;
LRESULT OnInitDialog(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
CheckDlgButton(IDC_SIMPLE_CHECKBOX, m_bButton);
return 0;
}
LRESULT OnOk(WORD wNotifyCode, WORD wID, HWND hWndCtl,
BOOL& bHandled){
m_bButton = IsDlgButtonChecked(IDC_SIMPLE_CHECKBOX);
OnCloseCmd(wNotifyCode, wID, hWndCtl, bHandled);
return 0;
}

67

WTL Developer’s Guide

};

This dialog is displayed with the following calls:
MySimpleDialog dlg;
dlg.DoModal();

Note that the windows for the controls in the dialog do not exist immediately after the
dialog’s C++ class is constructed – we do get a chance to change them
programmatically in OnInitDialog, which is called after they are created but before
the dialog is rendered on screen.
The MyModalDialog is a modal dialog, whose main feature is a pushbutton, which
when pressed relocates itself!
class MyModalDialog : public CDialogImpl{
public:
MyModalDialog(){}
enum {IDD=IDD_MODAL_DIALOG};
BEGIN_MSG_MAP(MyModalDialog)
COMMAND_ID_HANDLER(IDOK, OnOk)
COMMAND_ID_HANDLER(IDC_CTL_RELOCATE_BUTTON, OnRelocate)
END_MSG_MAP()
LRESULT OnRelocate(WORD wNotifyCode, WORD wID,
HWND hWndCtl, BOOL& bHandled){
RECT r;
CWindow wnd(GetDlgItem(IDC_CTL_RELOCATE_BUTTON));
wnd.GetWindowRect(&r);
r.left +=1;r.top+=1;r.right +=1;r.bottom+=1;
wnd.MoveWindow(&r, TRUE);
return 0;
}
LRESULT OnOk(WORD wNotifyCode, WORD wID, HWND hWndCtl,
BOOL& bHandled)
{
EndDialog(IDOK);
return 0;
}
};

This dialog is displayed with the following calls:
MyModalDialog dlg;
dlg.DoModal();

The MyModelessDialog is a modeless dialog which is implemented as follows:
#define WM_MODELESSCHILD_DESTROYED WM_USER+1
class MyModelessDialog : public CDialogImpl{
public:
CWindow m_ParentWnd;
MyModelessDialog(){}
enum {IDD=IDD_MODELESS_DIALOG};
BEGIN_MSG_MAP(MyModelessDialog)
COMMAND_ID_HANDLER(IDOK, OnOk)
COMMAND_ID_HANDLER(IDCANCEL, OnCancel)
END_MSG_MAP()

68

WTL Developer’s Guide

LRESULT OnOk(WORD wNotifyCode, WORD wID, HWND hWndCtl,
BOOL& bHandled){
m_ParentWnd = GetParent();
DestroyWindow();
return 0;
}
LRESULT OnCancel(WORD wNotifyCode, WORD wID, HWND hWndCtl,
BOOL& bHandled){
m_ParentWnd = GetParent();
DestroyWindow();
return 0;
}
void OnFinalMessage(HWND hWnd){
m_ParentWnd.PostMessage(WM_MODELESSCHILD_DESTROYED,0,0);
}
};

One issue that requires some application-specific design is how to destroy modeless
dialogs – both the window and the C++ class which represents it.
The modeless dialog itself can detect when it should disappear – when the user clicks in
the OK or Cancel buttons, or the Close button in the window frame (the “X” in the top
right corner of the window). Some developers have the class which created the
modeless dialog (e.g. the main application window class / mainframe) maintain a data
member which stores an instance of a modeless dialog class. The lifetime of this
instance mirrors the lifetime of the entire application – its visibility is turned on or off
as appropriate. This technique works fine and is often highly suitable.
Other developers wish to maintain an instance of a modeless dialog class only so long
at its window is displayed. Once its window is dismissed they wish both the window
and the C++ class instance to be destroyed. The modeless dialog itself can detect when
to disappear – and its OnOK or OnCancel methods can call DestroyWindow. This
will destroy the on-screen window, but the instance of the C++ class is still in memory.
If delete is called on it in OnOK or OnCancel it will crash – as the instance is not
finished processing. Instead, the piece of code that created the modeless dialog class
instance should be told to destroy it, using a custom window message. A good place to
put this is inside OnFinalMessage. The only problem here is to get the parent
window’s handle. By the time OnFinalMessage is called the window of the
modeless dialog is destroyed, so it we called GetParent directly there it would fail.
Therefore we call it elsewhere, and use a member variable to store the value, and we
use this value in OnFinalMessage. The final task is to detect the custom message
arriving in the parent and react appropriately. The following code shows how the parent
creates and destroyed the modeless dialog class.
BEGIN_MSG_MAP(CMainDialog)
COMMAND_ID_HANDLER(ID_SHOW_MODELESS, OnShowModeless)
MESSAGE_HANDLER(WM_MODELESSCHILD_DESTROYED, OnDestroyModeless)
END_MSG_MAP()
LRESULT OnShowModeless(WORD wNotifyCode, WORD wID,
HWND hWndCtl, BOOL& bHandled){
if (!m_dlgModeless){
m_dlgModeless=new (MyModelessDialog);

69

WTL Developer’s Guide

m_dlgModeless->Create(m_hWnd);
m_dlgModeless->ShowWindow(SW_SHOW);
}
return 0;
}
LRESULT OnDestroyModeless(UINT uMsg, WPARAM wParam,
LPARAM lParam, BOOL& bHandled){
if (m_dlgModeless){
OutputDebugString(TEXT("Destroying modeless class\n"));
delete m_dlgModeless;
m_dlgModeless=NULL;
}
return 0;
}

Windowing for ActiveX Controls
ATL provides good support for ActiveX control development and client containment.
Here we will examine a number of windowing issues related to ActiveX controls from
an ATL perspective.
Windowed or Windowless Controls
When creating ActiveX controls within ATL Object Wizard, in the Miscellaneous tab
there is an option called “Windowed Only”. By default, this checkbox is unselected.
In the past every ActiveX control had its own window, which became a child window
of the container (e.g. dialog box) in which the ActiveX control was placed. When one
or two controls were used, this was not a problem. When fifty controls were used, it
was noticeably slower. The idea behind windowless controls is that the control does not
have its own window; rather it lives within the window of its container, which improves
performance. Some containers mandate that the control have its own window, and some
will allow the control not to. Hence even a control that is capable of being windowless
might not always be created without a window. If the “Windowed Only” checkbox is
selected in the ObjectWizard, then the control will definitely be created with a window.
Some controls might have specific requirements that mandate this. In the generated
code, it results in this line being added to the control class’ constructor:
m_bWindowOnly = TRUE;

In the ATL templates, this variable is used inside a member function called
CComControlBase::InPlaceActivate, where it results in
CComControl::CreateControlWindow being called, which calls
CWindow::Create. The main class in your ATL ActiveX project will derive from
ATL’s CComControl, which in turn derives from ATL’s CComControlBase.

70

WTL Developer’s Guide

Basing an ActiveX Control on a Standard or Common Control
In addition to being based on a generic window (which needs to be rendered in
OnDraw), an ATL ActiveX control may also be based or an existing Standard or
common control. The Miscellaneous tab also has an Add control based on: combo-box.
This contains a list of some standard and common controls and one of them may be
chosen or another name entered. Note that if the selection in this combo-box is not
empty then the Windowed Only option is disabled. The ATL ActiveX control will
superclass the standard or common control.
Assuming we superclass the listview, the effect this has on the generated code is that a
CContainedWindow variable is added and this is initialized in the constructor (note
the last parameter is the message map id of 1):
CContainedWindow m_ctlSysListView32;
CLISTVIEWDEMO() :m_ctlSysListView32(
_T("SysListView32"),this,1){
m_bWindowOnly = TRUE;
}

The message map has this entry removed:
DEFAULT_REFLECTION_HANDLER()

And these entries added:
ALT_MSG_MAP(1)
// Replace this with message map entries for
// superclassed SysListView32
END_MSG_MAP()

Messages that were destined for the listview are now sent to the parent, with a message
map id of 1. This alternative message map could be populated with message handlers
that we wish to control.
ActiveX Control Message Reflection
With parent-child relationships among windows, the child window based on a control
often sends notification messages to the parent, which the parent will detect in its
window procedure. The normal mechanism of communication from an ActiveX control
to its container is via ActiveX events. The ActiveX control fires these events and the

71

WTL Developer’s Guide

container traps them. As an ActiveX control could be used with a variety of containers,
it would be much better to handle messages related to the ActiveX control within the
control itself, and not in its parent, and also when the control has something significant
to tell the parent, to so this via ActiveX events.
This concept is supported through the use of message reflection. The parent will reflect
back to the child any window messages it receives regarding the child. The child will
process these in its own window procedure, and if needed fire ActiveX events.
The ActiveX control container provides message reflection. It is an optional feature – a
control can at runtime determine if it is being contained in a container that supports it
by examining the ambient property MessageReflect – if it is true then the feature is
supported. If it is not supported, then there are three options. Either the control is not
very functional, or the parent has to do a lot of control-specific work, or the control
could create an extra window of its own to sit between the parent and the main control
window (the control window’s immediate parent would be the intermediate window,
which is also owned by the control and messages sent to it from children would be
processed with the control itself.
To enable the control to determine that the messages have been reflected back from the
parent as opposed to being sent directly to the control, the value OCM_BASE is added to
the message id.
With ATL control containment, there is support for message reflection, using two
macros and two member function of CWindowImplRoot. The two macros may be
used in messages maps, and in their implementation simply call the equivalent member
function. REFLECT_NOTIFICATIONS is used in the parent window and it calls
ReflectNotifications in CWindowImplRoot, which adds OCM_BASE to
the message value and calls SendMessage to send it to the child window.
The DEFAULT_REFLECTION_HANDLER macro is used in the control and it calls
DefaultReflectionHandler in CWindowImplRoot, which removes the
added OCM_BASE from the message value and calls DefWindowProc.
Rendering within ActiveX Controls
If your ActiveX control is superclassing a Windows standard or common control,
normally you need to do nothing to redraw it. The control itself will refresh as needed.
You may influence what it draws by setting values on the Windows control, such as the
text to display. For other ActiveX controls, your code will be called by the controlhosting infrastructure when the control needs to be redrawn.
If the control is windowed, CComBase detects the WM_PAINT message and calls
CComControlBase::OnDrawAdvanced, which finally calls your OnDraw
method, which you will need to supply. If the control is windowless, the container will
call IViewObject::Draw, which is implemented in ATL’s
CComControlBase::IViewObject_Draw; it then calls
CComControlBase::OnDrawAdvanced, which calls your OnDraw. It is passed a
structure of type ATL_DRAWINFO. The ATL ObjectWizard automatically generates
this default implementation of OnDraw:

72

WTL Developer’s Guide

HRESULT OnDraw(ATL_DRAWINFO& di){
RECT& rc = *(RECT*)di.prcBounds;
Rectangle(di.hdcDraw, rc.left, rc.top,
rc.right, rc.bottom);
SetTextAlign(di.hdcDraw, TA_CENTER|TA_BASELINE);
LPCTSTR pszText = _T("ATL 3.0 : FULL_CONTROL");
TextOut(di.hdcDraw,
(rc.left + rc.right)/2, (rc.top + rc.bottom)/2,
pszText, lstrlen(pszText));
return S_OK;
}

The two important fields in the ATLDRAWINFO structure are prcBounds, which is a
rectangle stating where to draw, and hdcDraw that is a device context, which can be
used to actually carry out the drawing.
Composite Controls
A composite control is an ActiveX control which can manage a dialog template that
contains standard and common controls, and other ActiveX controls. To the container, a
composite control behaves as a single ActiveX control. It is windowed. The ATL
template CComCompositeControl is the foundation for it.
CComCompositeControl : public CComControl >

Note that it uses CAxDialogImpl, which is also the foundation for normal dialog
boxes that can contain ActiveX controls. Composite controls can be used when you
need to supply somewhat more functionality than a single superclassed
standard/common control or a simple drawing control, yet less than a complete dialog
box. Think of them as pre-built blocks of functionality ready to slot in when needed.
HTML ActiveX Controls
ATL supports the creation of a type of ActiveX control known as an HTML control.
This control hosts the WebBrowser control from Internet Explorer. Naturally the client
application could host the web browser control directly, so it only makes sense creating
an HTML control if there is some “added-value” which is needed by the client, and you
do not wish to implement it directly within the client.
The HTML control generated by the ATL ObjectWizard by default renders an HTML
page that is stored as a resource (loaded from an external file during the build).

The control’s OnCreate method has this call:
HRESULT hr = wnd.CreateControl(IDH_MYHTMLCONTROL);

73

WTL Developer’s Guide

which is implemented inside ATL as:
HRESULT CAxWindows::CreateControl(DWORD dwResID,
IStream* pStream=NULL, IUnknown** ppUnkContainer=NULL){
TCHAR szModule[_MAX_PATH];
GetModuleFileName(_Module.GetModuleInstance(),
szModule, _MAX_PATH);
CComBSTR bstrURL(OLESTR("res://"));
bstrURL.Append(szModule);
bstrURL.Append(OLESTR("/"));
TCHAR szResID[11];
wsprintf(szResID, _T("%0d"), dwResID);
bstrURL.Append(szResID);
ATLASSERT(::IsWindow(m_hWnd));
return AtlAxCreateControl(bstrURL, m_hWnd, pStream,
ppUnkContainer);
}

The WebBrowser control understands a Windows specific protocol, called res:// in
addition to http:// and the other standard protocols. With res://, the data
following the protocol description contains the pathname to a DLL and a resource id of
an HTML page, which is located inside the resources of that DLL. If our control were
installed under C:\HtmlControlDemo.dll and the resource id of the HTML page were
102, then the full resource string would be:
res://C:\HtmlControlDemo.dll/102

ActiveX Control Containment
ATL provides a family of templates that support ActiveX control containment. The
naming convention is to use CAx as a prefix - e.g. CAxDialogImpl. To give an
indication of the complexity involved, the implementation of most of the ATL
functionality for control containment is in the atlhost.h file that is over 2,500 lines long.
Luckily most of the work is done for us, so as component and client developers we only
have to concern ourselves with the “added-value” of our solution. Note that ATL’s
control containment functionality works through the official ActiveX control COM
interfaces, and they totally disregard with which development environment the control
itself was created – VB, MFC, ATL or maybe even in low-level C. ATL control
containment works equally well with them all. No special tricks are performed if the
ActiveX control itself was also developed with ATL.
The AtlAxWin Window and CAxWindow / CAxHostWindow Templates
To contain ActiveX controls, an application must support a wide variety of COM
interfaces and a hosting window, which is used as the parent of the ActiveX control
window if it is windowed, and as the actual window for rendering if the ActiveX
control is windowless. In implementations of control containment with some other
development environments, the hosting window would be a top-level window or more
likely a dialog-box. With ATL, the hosting window is created specifically for each
hosted control, and this hosting window itself is usually a child of a dialog-box. It is an
intermediary between the ActiveX control and the dialog. This hosting window is of the

74

WTL Developer’s Guide

window class AtlAxWin. ATL provides the CAxHostWindow and CAxWindow
templates for client-side management of AtlAxWin windows.
If you place an ActiveX control inside a dialog, ATL control containment code will
automatically place AtlAxWin windows between the dialog and the ActiveX control.
This allows the ATL C++ templates, CAxHostWindow and CAxWindow, to manage
the AtlAxWin to handle most of the complex interaction with the ActiveX control,
leaving nothing or little for the client application developer to do.
For example, if you have a dialog box with three ActiveX controls, there will be an
ATL dialog box C++ class and its window, and three instances of C++ classes based on
CAxHostWindow and another three windows. In addition, if the ActiveX controls are
windowed-only, then there will be a further three windows, giving a total of seven
windows for this sample dialog. ATL control containment does support windowless
controls, so if the ActiveX controls being used also support windowless, only four
windows will be used for the dialog. If we assume the dialog has OK and Cancel
standard pushbuttons and three ActiveX controls, one of which is Windowed-Only, the
window hierarchy as displayed in SPY++ is:

Creating the Control
When CAxWindow is instantiated it creates the “AtlAxWin” window and then it
needs to internally use CoCreateInstance to instantiate a COM component, the
ActiveX control, and make it live within the “AtlAxWin” window. This cannot be
done until the “AtlAxWin” window actually exists. The easiest time to perform these
tasks (and what ATL actually does) is during the processing of WM_CREATE for
AtlAxWin.
The next question is which CLSID to use? The client code creating CAxWindow will
know and it somehow needs to pass this information into the handler for WM_CREATE.
Your first idea might be to use the LPCREATESTRUCT that is passed in to
CreateWindow. The client code can be written to instantiate CAxWindow directly,
but more often it instantiates a CAxDialogImpl, which is based on a dialog template
in the resources, which contains embedded ActiveX controls. The resource file will also
contain properties for each ActiveX control, and these also need to be passed to the
creation code. ATL uses the LPCREATESTRUCT field for a pointer to a stream for this
property information, so it is not available for the CLSID. The alternative approach,
which ATL takes, is to set the window name to a CLSID or information from which the
CLSID can be deduced. The “AtlAxWin” is never displayed with a title bar, so endusers will never see it. Once extracted, it is set to NULL. Hence the previous SPY++
screen dump showed empty titles for the AtlAxWin windows.

75

WTL Developer’s Guide

To see this is action, set up a break point in your debugger on ATL’s
AtlAxWindowProc in the atlhost.h header, run the application. When it hits the
breakpoint, examine the window hierarchy in SPY++.
static LRESULT CALLBACK AtlAxWindowProc(HWND hWnd,
UINT uMsg, WPARAM wParam, LPARAM lParam){
switch(uMsg){
case WM_CREATE:
. . .
// Property stream is lpCreate->lpCreateParams
// This value is 0 when creating AtlAxWin directly, and
// is set to poitner to an IStream based on global memory
// initialised from resources for dialog-based controls
CREATESTRUCT* lpCreate = (CREATESTRUCT*)lParam;
// Get string defining CLSID to use from window title
int nLen = ::GetWindowTextLength(hWnd);
LPTSTR lpstrName = (LPTSTR)_alloca((nLen + 1)
* sizeof(TCHAR));
::GetWindowText(hWnd, lpstrName, nLen + 1);
::SetWindowText(hWnd, _T(""));

As each ActiveX control is created, the debugger stops at the GetWindowText call.
In SPY++, we see a window of type AtlAxWin with a name of the CLSID. After this is
extracted, the call to SetWindowText NULLs the title. This happens for each of the
three controls. The output from SPY++ when the third and final control is created is:

We should emphasize again, that the AtlAxWin is an extra window to the dialog itself
and (if windowed only) the ActiveX control. If the control is windowed only, then
inside CComControlBase::InPlaceActivate which calls
CComControlBase::CreateControlWindow which actually creates the
window as a child of AtlAxWin. Once this is done, the output from SPY++ is as in the
previous screen dump.
Which control to create?
We have seen how the window title is used to pass in a string to the WM_CREATE
handling code. This string is usually a CLSID of the ActiveX control to create. This is
the case when the ActiveX control is created based on a dialog template from the
resource file. However, when used from client application code, the string can be set to
other values, such as a PROGID or a URL or even the pathname to a MS-WORD
document, and an appropriate ActiveX control for these will be created. ATL uses the
following ordered set of rules to decided which CLSID to use in
CoCreateInstance:

76

WTL Developer’s Guide

•

Firstly, if window name is NULL, or first character of window name is 0, do not
create any control and return

•

Secondly, if the initial seven characters of the window name are
“MSHTML:” (case-insensitive), then use CLSID_HTMLDocument (set a flag
to note it can handle HTML) and return

•

Thirdly, if the string is shorter than 255 characters, then do the following. If the
string begins with “{“ use CLSIDFromString to determine if it is a CLSID,
otherwise use CLSIDFromProgId to determine if it is a ProgID – if either of
these succeeded, then use CoCreateInstance with the CLSID of whichever
technique worked, and return

•

Finally, if none of the above worked, then call CoCreateInstance with
CLSID_WebBrowser (set a flag to note it can handle HTML)

This is implemented in the static function CreateNormalizedObject in atlhost.h.
Note that if a non-NULL string is passed in which does not contain “MSHTML:”, a
string-ified CLSID or a ProgID, then CLSID_WebBrowser is used – it is the default.
Hence we could pass in a pathname to a file from MS-WORD or any other application
which CLSID_WebBrowser can access through its Active Document support, or any
files that CLSID_WebBrowser can directly understand, such as files with the suffixes
*.htm or *.txt .
If the string was a CLSID or a ProgID, once CoCreateInstance was called ATL is
finished with it. Otherwise there is a bit more to do, as CLSID_HTMLDocument or
CLSID_WebBrowser have been instantiated, but now they need to be told what to
display. The AtlControlEx method does this.
CLSID_HTMLDocument is Internet Explorer’s rendering engine for HTML. ATL
uses it if the window title started with the seven characters “MSHTML:”. The rest of
the window title can be raw HTML. This needs to be passed to the
CLSID_HTMLDocument instance to render. The relevant snippet of code from
CreateControlEx is as follows:
CComPtr spHTMLDoc2;
hr = spUnk->QueryInterface(IID_IHTMLDocument2,
(void**)&spHTMLDoc2);
if (SUCCEEDED(hr)){
CComPtr spHTMLBody;
hr = spHTMLDoc2->get_body(&spHTMLBody);
if (SUCCEEDED(hr))
hr = spHTMLBody->put_innerHTML(
CComBSTR(lpszTricsData + 7));
}

We see it passes the window title from the eighth character onwards to the
put_innerHMTL method of the BODY. A valid message could be:
MSHTML:

Hello World

77 WTL Developer’s Guide CLSID_WebBrowser is Internet Explorer’s full engine. It incorporates the rendering engine for HTML, Active Document support, scripting and dynamic HTML. If CLSID_WebBrowser was instantiated, then the full window title is passed in to the instance via its IWebBrowser2::Navigate2 method, and it can decide what to do with it. The relevant snippet of code from CreateControlEx is as follows: CComPtr spBrowser; spUnk->QueryInterface(IID_IWebBrowser2, (void**)&spBrowser); if (spBrowser){ CComVariant ve; CComVariant vurl(lpszTricsData); spBrowser->put_Visible(VARIANT_TRUE); spBrowser->Navigate2(&vurl, &ve, &ve, &ve, &ve); } If the window title is a URL, the web browser will access it over HTTP and render the contents. If the title is another protocol that Internet Explorer understands (e.g. mailto:info@example.com), then it will be used. If the title is a filename (e.g. a full pathname to a Word 2000 document), then the ActiveX Document functionality comes into play. Enthusiastic Windows shell programmers will note that Navigate2 is used. In addition to accepting an URL/pathname, this method also accepts a PIDL to a location in the Windows Shell. So if we set the window title that that of a PIDL, can we browse the shell? Well, no. As is, the code here initializes the VARIANT parameter to Navigate2 as a string, and when you look at the IWebBrowser2::Navigate2 in the MSDN, you will see that strings are processed as if they are URLs. To browse the shell, we have to pass in a VARIANT of type VT_ARRAY|VT_UI1 containing a SAFEARRAY with the PIDL. The ATL code does not do this and it is a bad idea to modify it. However, there is nothing stopping us calling IWebBrowser2::Navigate2 sometime later. The later chapter that discusses the views in WTL applications has samples of using all the possible window title formats and a PIDL. CAxDialogImpl - ActiveX Controls within Dialog Boxes Dialog boxes that do not contain ActiveX controls are managed by ATL’s CDialogImpl template. Classes deriving from this must be manually created – the ATL ObjectWizard does not provide any help here. ATL’s CAxDialogImpl manages dialog boxes that do support ActiveX controls. The ATL ObjectWizard, in its miscellaneous section, does support creating these. As there is some extra processing inside CAxDialogImpl, if you do not plan to host ActiveX controls, you would be (slightly) better off using CDialogImpl. If you decide later that you do need ActiveX controls, you can simply switch CAxDialogImpl for CDialogImpl in the header file. The ObjectWizard is only available if DevStudio thinks it is editing an ATL project. This is of course the case if you have created the project with the ATL AppWizard. This is not the case it you have created the project with the WTL AppWizard (and not selected the COM Server option). There is a way to fix this – which is discussed in the later chapter on dialogs and controls. 78 WTL Developer’s Guide With the ResourceView, you can edit the resource template used as the basis for a dialog. The control CLSID, its id and its location in the dialog are stored in the DIALOG section of the RC file. Additional properties for each ActiveX control are stored in the DLGINIT section. When CAxDialogImpl is constructing a dialog, it needs to process both sections in order to create the specified dialog. //////////////////////////////////////////////////////////////// // Dialog IDD_MYACTIVEXDIALOG DIALOG DISCARDABLE 0, 0, 186, 95 STYLE DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU CAPTION "Dialog" FONT 8, "MS Sans Serif" BEGIN DEFPUSHBUTTON "OK",1,129,7,50,14 PUSHBUTTON "Cancel",2,129,24,50,14 CONTROL "",IDC_MEDIAPLAYER1, "{22D6F312-B0F6-11D0-94AB-0080C74C7E95}", WS_TABSTOP,10,0, 118,75 END ////////////////////////////////////////////////////////////////// Dialog Info IDD_MYACTIVEXDIALOG DLGINIT BEGIN IDC_MEDIAPLAYER1, 0x376, 340, 0 0x0000, 0x0000, 0x0001, 0x0000, 0x1383, 0x0000, 0x0c67, 0x0000, 0x0003, 0xffff, 0xffff, 0x000b, 0x0000, 0x000b, 0xffff, 0x000b, . . . 0x0000, 0x000b, 0x0000, 0x0003, 0xfda8, 0xffff, 0x000b, 0x0000, 0 END //////////////////////////////////////////////////////////////// During dialog construction, ATL creates an “AtlAxWin” for each control, and puts the DLGINIT data for the control in a stream based on global memory. As discussed earlier, this is passed into the handler for WM_CREATE in its LPCREATESTRUCT structure, and from there the contents of the stream are parsed by the control (e.g. if the control was created using ATL, its property map is traversed). The final issue to consider is once constructed and initialized, how does the containing application programmatically communicate with the control at run-time. MFC has its COleDispatchDriver class. VB internal run-time dispatching functionality, which the application developer can influence using the VB References dialog box. What happens with ATL? Like the other solutions, it is better that ATL is told at compile time about the ActiveX controls it will be using. This can be done using the #import construct. We have to #import the DLL containing the ActiveX control. Question: To use #import, we have to load in the CRT, something we often wish to avoid – is there any better way to access the type information for ActiveX controls, which does not consume hundreds of lines of code, yet gives us access to the dual interfaces of the ActiveX control? We can use CWindow::GetDlgControl to extract an interface pointer to the control and then we can use it like any other COM interface. 79 WTL Developer’s Guide Internally, what happens is that the CAxHostWindow/CAxWindow instance representing the ActiveX control maintains a data-member of the IUnknown interface to the control, which it received from CoCreateInstance (the riid passed into it is IID_IUnknown). It has a message handler for an ATL custom message type WM_ATLGETCONTROL, and it responds to it by add-ref’ing the IUnknown pointer it has and returning it as the handler result. MESSAGE_HANDLER(WM_ATLGETCONTROL, OnGetControl) LRESULT OnGetControl(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/){ IUnknown* pUnk = m_spUnknown; if (pUnk) pUnk->AddRef(); return (LRESULT)pUnk; } When CWindow::GetDlgControl is called, it converts the dialog item id into a window handler with the GetDlgItem call (this window will be of type “AtlAxWin”), and then it calls AtlAxControl helper function, which causes a WM_ATLGETCONTROL message to be sent to the AtlAxWin window. HRESULT GetDlgControl(int nID, REFIID iid, void** ppUnk){ HRESULT hr = E_FAIL; HWND hWndCtrl = GetDlgItem(nID); if (hWndCtrl != NULL){ *ppUnk = NULL; CComPtr spUnk; hr = AtlAxGetControl(hWndCtrl, &spUnk); if (SUCCEEDED(hr)) hr = spUnk->QueryInterface(iid, ppUnk); } return hr; } ATLINLINE ATLAPI AtlAxGetControl(HWND h, IUnknown** pp){ *pp = (IUnknown*)SendMessage(h, WM_ATLGETCONTROL, 0, 0); return (*pp) ? S_OK : E_FAIL; } Sample : ATLControlDemo The ATLControlDemo sample shows how to create a control in one project and host it in a different project. 80 WTL Developer’s Guide The one workspace is used for both. The fact that both projects are created with ATL does not have an impact on how they communicate – they will still use the rules of ActiveX controls. Use the ATL AppWizard to create the workspace and first project named ATLControlDemo. The project should be of type DLL. Use the ATL Object Wizard to insert a Full Control. Name it “MyAtlControl”. In ClassView, select IMyAtlControl, and right-click to bring up the context menu, and select Add Method. Use the dialog to add a method called Beep. This adds the method to the IDL file and adds a default implementation to the CMyAtlControl class. Change it to call the Beep API. STDMETHODIMP CMyAtlControl::Beep(){ ::Beep(200,200); return S_OK; } That is the ActiveX control finished. Build the project, which automatically registers it on the development machine. It also adds type information to the DLL. 81 WTL Developer’s Guide Now in FileView, select the Workspace root, right click and select Add New Project to Workspace. Select the ATL AppWizard and create a new EXE-based project called ATLControlContainer. Use the ObjectWizard to insert a new dialog box. In ResourceView, select the dialog template, right-click and select Add ActiveX Control. Select the MyAtlControl from the list. If it does not appear in the list it is probably because you have not actually compiled the AtlControlDemo project (which would have automatically registered the control). In Properties, assign the id IDC_MYATLCONTROL to the control. In the implementation file for the dialog box, add this import statement: #import "..\debug\ATLControlDemo.dll" no_namespace Change the OnInitDialog method to call the exposed Beep method in the ActiveX control: LRESULT OnInitDialog(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled){ CComPtr m_pIMyAtlControl; GetDlgControl(IDC_MYATLCONTROL, __uuidof(IMyAtlControl), (void **) &m_pIMyAtlControl); m_pIMyAtlControl->Beep(); return 1; // Let the system set the focus } In the application’s main cpp file, ATLControlContainer.cpp, include the header file for the dialog class and instantiate the dialog and call its DoModal. That’s it - compile and run. You should hear a beep should as the dialog is being displayed. 82 WTL Developer’s Guide Chapter 4 WTL Quick Tour Objectives The objectives of this chapter are to: • Examine the templates and classes in WTL • List the features which are not part of WTL • Describe issues such as the WTL namespace, CRT and error handling • List the WTL macros • See how debugging can be improved with AUTOEXP.DAT • Review the use of CRT and WTL • Compare MFC and WTL • Cover WTL’s CString class The WTL Distribution There are three parts to the WTL distribution – the fifteen C++ header files in the include directory, the WTL AppWizard and the sample projects. The core functionality is in the include directory. The AppWizard generates projects whose source code makes calls into the templates/classes defined in the header files. The sample projects should how to use them. You might spend a few hours examining what the WTL AppWizard produces and the sample projects, but you will spend many weeks learning all about the templates/classes defined in the header files. WTL provides a large number of templates and classes dedicated to user interface development. They may be used in tandem with, rather than competing with, other template libraries, such as ATL, STL and VC++ OLE DB Data templates. All these template libraries together may be considered as a competitor to MFC and VB. In most parts, WTL is a very thin layer on the low level Win32 APIs. In a small number of areas (e.g. scroll-bars, splitters, print preview) there is substantial new functionality. Apart from the WTL AppWizard, there are no specific WTL wizards integrated with DevStudio. One can use the ATL Windows Message Handler Wizard to add handlers, but these do not use WTL’s message crackers. 83 WTL Developer’s Guide Templates and Classes WTL is a collection of C++ templates and classes. There are some isolated cases of a small group of classes using inheritance (e.g. CDC, CPaintDC, CWindowDC and CPrintDC), but by and large WTL’s templates and classes are independent of each other and they share no common root, hence there is no such thing as a hierarchy diagram for WTL. There simply is no large hierarchy. In this section we will provide an overview of the functional areas covered by WTL and list the templates and classes available. Each functional area is neatly contained within its own WTL header file, which we also list. Every WTL application must include atlapp.h, which internally includes atluser.h and atlgdi.h. The other WTL header files only need to be included if the functionality they contain will be used in the application. Application Services (atlapp.h) These classes provide the foundation for the module (EXE or DLL) and control the message loop. They also support message filtering and idle handling. class class class class class CAppModule : public CComModule CServerAppModule : public CAppModule CMessageLoop CMessageFilter CIdleHandler Standard Controls and Common Controls (atlctrls.h) WTL provides a wrapper template for each standard control and common control in the Windows OS. The template is typically used with ATL’s CWindow, so an appropriate typedef is also provided for each control. In code, you will normally use the typedef’ed value, and not the template directly. Each control is defined as follows: template class CStaticT; typedef CStaticT CStatic; Here is the full list of controls: CButtonT CListBoxT CComboBoxT CEditT CScrollBarT CToolTipCtrlT CHeaderCtrlT CListViewCtrlT CTreeViewCtrlT CToolBarCtrlT CStatusBarCtrlT CTabCtrlT CButton CListBox CComboBox CEdit CScrollBar CToolTipCtrl CHeaderCtrl CListViewCtrl CTreeViewCtrl CToolBarCtrl CStatusBarCtrl CTabCtrl CTrackBarCtrlT CUpDownCtrlT CProgressBarCtrlT CHotKeyCtrlT CAnimateCtrlT CRichEditCtrlT CReBarCtrlT CMonthCalendarCtrlT CDateTimePickerCtrlT CIPAddressCtrlT CPagerCtrlT 84 CTrackBarCtrl CUpDownCtrl CProgressBarCtrl CHotKeyCtrl CAnimateCtrl CRichEditCtrl CReBarCtrl CMonthCalendarCtrl CDateTimePickerCtrl CIPAddressCtrl CPagerCtrl WTL Developer’s Guide These controls can be created using the Create method, but more often a dialog template defined in the ResourceView will contain the layout of the dialog and the controls which are to appear on it, and in source code you can instantiate one of these control templates and call its Attach method to hook up with the control in the dialog box. So to add a string to a list box in a dialog you might use the following: CListBox m_ListBox; . . . m_ListBox.Attach(GetDlgItem(IDC_LIST_DEMO)); m_ListBox.AddString(TEXT("One")); Note that ATL already supports ActiveX control containment (as discussed in detail in the previous chapter on ATL Windowing) and WTL adds no additional support. There are some additional classes/templates for controls that provide extra functionality. WTL’s CComboBoxEx is a wrapper for Win32’s ComboBoxEx control, which adds support for image lists to the standard combo box. In WTL, CComboBoxExT is derived from CComboBoxT, and adds extra functionality as needed. template class CComboBoxExT : public CComboBoxT< TBase > typedef CComboBoxExT CComboBoxEx; Flat scroll-bars are supported in WTL. In functionality they are the same as normal scrollbars, but in appearance, as the name suggests, they are flat, without the raised beveled edge. template class CFlatScrollBarImpl template class CFlatScrollBarT : public TBase, public CFlatScrollBarImpl > typedef CFlatScrollBarT CFlatScrollBar; The treeview control deals with a HTREEITEM, which is a handle to a complicated structure containing details of how each node should appear in the tree. To simplify dealing with HTREEITEM, WTL has a class called CTreeItem, which provides methods to get and set text, image, data and state information, and to walk the tree. The CTreeViewCtrlExT template is based on the CTreeViewCtrlT template. template class CTreeViewCtrlExT : public CTreeViewCtrlT< TBase > typedef CTreeViewCtrlExT CTreeViewCtrlEx; class CTreeItem; // wrapper for HTREEITEM CTreeViewCtrlExT furnishes methods that deal with CTreeItem in addition to the methods that work with raw HTREEITEM handles. For example, CTreeViewCtrlT provides this method to insert an item: HTREEITEM InsertItem(LPTV_INSERTSTRUCT lpInsertStruct); whereas CTreeViewCtrlExT has this method: CTreeItem InsertItem(LPTV_INSERTSTRUCT lpInsertStruct); Edit boxes and rich-edit boxes must deal with a number of common messages, such as ID_EDIT_COPY, ID_EDIT_CUT, ID_EDIT_PASTE and ID_EDIT_UNDO. These two helper classes are provided to automate the processing of such messages. 85 WTL Developer’s Guide template class CEditCommands; template class CRichEditCommands : public CEditCommands< T > Helper classes are also provided to manage image lists and toolinfo. class CImageList; class CToolInfo : public TOOLINFO A drag-listbox is similar to a normal listbox, but also offers functionality to drag items within the listbox. CDragListNotifyImpl supports message processing during the dragging. template class CDragListBoxT : public CListBoxT< TBase > typedef CDragListBoxT CDragListBox; template class CDragListNotifyImpl; Some common controls can be configured to use custom draw to render their items. They send a NM_CUSTOMDRAW message. This WTL template helps with the processing of such messages: template class CCustomDraw Command Bars (atlctrlw.h) In the past applications presented commands to end users in standard menubars. Then along came toolbars. Later still came the rebar control, first introduced with Internet Explorer. Some advanced applications, such as Microsoft Word 2000, provided the concept of command bars, which are a uniform hosting of menus and toolbar icons. WTL also supports functionality called command bars, but these are different to those that appear in Word 2000. In WTL, a command-bar is a more powerful version of a menubar. The toolbar is still used. The command bar is hosted in one band and the toolbar in a different band within a single rebar control in the WTL generated application. The bands are movable within that rebar. Command bars (and toolbars) in WTL are not dockable. A WTL command bar displays menus, and within each menu displays menu-items and the equivalent toolbar icon next to the menu item, thus increasing end-user knowledge of what each toolbar icon means. WTL provides classes to help with command bars: class CCommandBarCtrlBase : public CToolBarCtrl template class CCommandBarCtrlImpl : public CWindowImpl< T, TBase, TWinTraits> class CCommandBarCtrl : public CCommandBarCtrlImpl When generating an application with the WTL AppWizard, and assuming you accept the default settings, then your project will automatically use command-bars and you have no further action to take. Advanced Controls (atlctrlx.h) WTL supports a number of more advanced controls. 86 WTL Developer’s Guide CBitmapButton provides pushbuttons that display an image from an imagelist rather than text. template class CBitmapButtonImpl : public CWindowImpl< T, TBase, TWinTraits> class CBitmapButton : public CBitmapButtonImpl CCheckListView provides a list of checkboxes. template class CCheckListViewCtrlImplTraits; typedef CCheckListViewCtrlImplTraits CCheckListViewCtrlTraits; template class CCheckListViewCtrlImpl : public CWindowImpl; class CCheckListViewCtrl : public CCheckListViewCtrlImpl; CHyperlink enables the embedded of web-links within the application. When pressed, a web-browser is started displaying a specific URL. template class CHyperLinkImpl : public CWindowImpl< T, TBase, TWinTraits > class CHyperLink : public CHyperLinkImpl CWaitCursor sets the cursor. The actual cursor used depends on an input parameter – which defaults to IDC_WAIT. class CWaitCursor CMultiPaneStatusBarCtrl provides a status bar with sub-divisions, each of which can contain different text. template class CMultiPaneStatusBarCtrlImpl : public CWindowImpl< T, TBase > class CMultiPaneStatusBarCtrl : public CMultiPaneStatusBarCtrlImpl Dynamic Data Exchange (atlddx.h) DDX is used to transfer data in both directions between C++ data members and dialog box controls. In WTL, DDX involves a single template, CWinDataExchange, and many macros, such as DDX_TEXT. template class CWinDataExchange To use DDX, your dialog class should derived from both CDialogImpl and CWinDataExchange, and it should have a message map, such as: BEGIN_DDX_MAP(CMyDialogDlg) DDX_INT(IDC_AMOUNT_INT, m_nAmount) 87 WTL Developer’s Guide END_DDX_MAP() Currently this map must be created manually, though it is likely that in a future version of Visual C++ Wizard support will be added. System Dialogs And Property Sheets (atldlgs.h) WTL offers a full range of templates which expose the system dialogs. template class CFileDialogImpl : public CDialogImplBase class CFileDialog : public CFileDialogImpl template class CFolderDialogImpl class CFolderDialog : public CFolderDialogImpl class CCommonDialogImplBase : public CWindowImplBase template class CFontDialogImpl : public CCommonDialogImplBase class CFontDialog : public CFontDialogImpl class CRichEditFontDialogImpl : public CFontDialogImpl< T > class CRichEditFontDialog : public CRichEditFontDialogImpl template class CColorDialogImpl : public CCommonDialogImplBase class CColorDialog : public CColorDialogImpl template class CPrintDialogImpl : public CCommonDialogImplBase class CPrintDialog : public CPrintDialogImpl template class CPrintDialogExImpl : public CWindow, public CMessageMap, public IPrintDialogCallback, public IObjectWithSiteImpl< T > class CPrintDialogEx : public CPrintDialogExImpl template class CPageSetupDialogImpl : public CCommonDialogImplBase class CPageSetupDialog : public CPageSetupDialogImpl template class CFindReplaceDialogImpl : public CCommonDialogImplBase class CFindReplaceDialog : public CFindReplaceDialogImpl WTL also offers a range of templates to help with property sheet construction. class CPropertySheetWindow : public CWindow template class CPropertySheetImpl : public CWindowImplBaseT< TBase > class CPropertySheet : public CPropertySheetImpl class CPropertyPageWindow : public CWindow template class CPropertyPageImpl : public CDialogImplBaseT< TBase > template class CPropertyPage : public CPropertyPageImpl > Frames (atlframe.h) The following classes support frames. CFrameWndClassInfo is the window class for frame windows. 88 WTL Developer’s Guide class CFrameWndClassInfo CFrameWindowImpl is used as the basis of application-level C++ classes that manage the frames. template class CFrameWindowImplBase : public CWindowImplBaseT< TBase, TWinTraits > template class CFrameWindowImpl : public CFrameWindowImplBase< TBase, TWinTraits > To support MDI, WTL provides these templates: class CMDIWindow : public CWindow template class CMDIFrameWindowImpl : public CFrameWindowImplBase template class CMDIChildWindowImpl : public CFrameWindowImplBase The template COwnerDraw assists with the implementation of owner draw functionality. It is normally added to a class via multiple inheritance. The difference between COwnerDraw and CCustomDraw that we mentioned when introducing the controls, is that CCustomDraw processes NM_CUSTOMDRAW notifications delivered in a WM_NOTIFY message. This is supported by the following common controls: header, list view, rebar, toolbar, tooltip, trackbar and tree view controls. COwnerDraw processes the WM_DRAWITEM, WM_MEASUREITEM, WM_COMPAREITEM and WM_DELETEITEM messages, which are typically sent to owner-drawn button, combo box, list box, list view control, or menu items. template class COwnerDraw CUpdateUI is provided to update UI elements dynamically. Menu-items and toolbar buttons can be enabled or disabled using this. class CUpdateUIBase; template class CUpdateUI : public CUpdateUIBase GDI (atlgdi.h) WTL provides templates to manage all the GDI objects. The Boolean template parameter for each of these, t_bManaged, specifies whether the GDI object should be deleted in the destructor of the template instance. For each template, WTL has two typedefs, one of which is declared with t_bManaged set to true, and the other with t_bManaged set to false. These templates bring together all the Win32 C APIs which deal with each GDI. These expose typesafe C++ methods which call the underlying C API. template class CPenT typedef CPenT CPenHandle; typedef CPenT CPen; template class CBrushT 89 WTL Developer’s Guide typedef CBrushT typedef CBrushT template typedef CFontT typedef CFontT template typedef CBitmapT typedef CBitmapT template typedef CPaletteT typedef CPaletteT template typedef CRgnT typedef CRgnT template typedef CDCT typedef CDCT CBrushHandle; CBrush; class CFontT CFontHandle; CFont; class CBitmapT CBitmapHandle; CBitmap; class CPaletteT CPaletteHandle; CPalette; class CRgnT CRgnHandle; CRgn; class CDCT CDCHandle; CDC; A number of helper classes are derived from CDC to provide additional functionality. CPaintDC is used to handle WM_PAINT messages; it calls Win32’s BeginPaint in its constructor and EndPaint in its destructor. CClientDC and CWindowDC get the client area and full window DCs of a HWND parameter respectively. CEnhMetaFileDC is used to render into a enhanced metafile. It calls Win32’s CreateEnhMetaFile from its constructor and CloseEnhMetaFile from its destructor. class class class class CPaintDC : public CDC CClientDC : public CDC CWindowDC : public CDC CEnhMetaFileDC : public CDC Extra functionality is provided for enhanced metafiles. CEnhMetaFileT is a wrapper for an enhanced metafile handle. CEnhMetaFileInfo provides methods that call Win32’s GetEnhMetaFileBits, GetEnhMetaFileDescription, GetEnhMetaFileHeader and GetEnhMetaFilePixelFormat. template class CEnhMetaFileT typedef CEnhMetaFileT CEnhMetaFileHandle; typedef CEnhMetaFileT CEnhMetaFile; class CEnhMetaFileInfo Miscellaneous (atlmisc.h) Helper classes are provided to assist managing common data types. WTL’s CString is a perfect emulation of MFC’s equivalent. CSize, CPoint and CRect derived from Win32 data types, and offer method to initialize the values and carry out operations (e.g. summation) on them. class class class class CSize : public tagSIZE CPoint : public tagPOINT CRect : public tagRECT CString; These classes support recent file list and the file searching (using Win32’s FindFirstFile / FindNextFile APIs). 90 WTL Developer’s Guide class CRecentDocumentList; class CFindFile; Printing (atlprint.h) WTL provides good support for printing. PrinterInfo data is handled via: template class CPrinterInfo These are provided to manage a handle to a printer: template class CPrinterT typedef CPrinterT CPrinterHandle; typedef CPrinterT CPrinter; These are provided to manage printing’s DEVMODE information: template class CDevModeT; typedef CDevModeT CDevModeHandle; typedef CDevModeT CDevMode; When rendering for printing, use this DC type: class CPrinterDC : public CDC To gain information about print jobs, use: class IPrintJobInfo class CPrintJobInfo : public IPrintJobInfo class CPrintJob; Print preview window support is provided by: class CPrintPreview; template class CPrintPreviewWindowImpl :public CWindowImpl, public CPrintPreview class CPrintPreviewWindow : public CPrintPreviewWindowImpl Scrolling (atlscrl.h) Window scrolling is supported through this set of templates: template class CScrollImpl template class CScrollWindowImpl :public CWindowImpl,public CScrollImpl< T > template class CMapScrollImpl: public CScrollImpl< T > template class CMapScrollWindowImpl : public CWindowImpl< T, TBase, TWinTraits >, public CMapScrollImpl< T > Flat scrollbars are also supported. template class CFSBWindowT :public TBase, public CFlatScrollBarImpl > typedef CFSBWindowT CFSBWindow; 91 WTL Developer’s Guide Splitters (atlsplit.h) Horizontal and vertical splitters are supported using these templates: template class CSplitterImpl; template class CSplitterWindowImpl : public CWindowImpl< T, TBase, TWinTraits >, public CSplitterImpl, t_bVertical> template class CSplitterWindowT : public CSplitterWindowImpl< CSplitterWindowT, t_bVertical> typedef CSplitterWindowT CSplitterWindow; typedef CSplitterWindowT CHorSplitterWindow; Menus (atluser.h) The CMenuItemInfo class is a wrapper for Win32’s MENUITEMINFO structure and menus are managed through CMenuT, which provides typesafe access to all operations that are possible on menus. class CMenuItemInfo : public MENUITEMINFO template class CMenuT typedef CMenuT CMenuHandle; typedef CMenuT CMenu; What is NOT in WTL After having considered what are the major pieces of functionality within WTL, now we will examine what is NOT supported. WTL is solely aimed at developers of desktop user interfaces. It does this exceptionally well. It does not stray into other areas. Competing toolkits, such as VC++’s MFC or VB do provide incredible amounts of functionality covering vast areas. Their supporters might claim these toolkits offer a consistent mutually compatible, well-integrated range of functionality, but WTL supporters will counter with the assertion that MFC and VB are monolithic, slow-changing and simply too big (downloads of apps to be measured in megabytes, compared to WTL downloads which are in kilobytes). The philosophy of template programming – fast, static binding, supporting multiple data types, is still very much alive in WTL and is shared with other template libraries, such as ATL, STL and the data templates. WTL can be used in conjunction with these, or on its own. In this section we examine features not available with WTL. WTL does not compete with ATL, STL, Data Templates or COM libraries Functionality that is provided by other template libraries is not duplicated in WTL. This keeps WTL small and focused, yet developers can access other template libraries when needed. 92 WTL Developer’s Guide No Document Support WTL provides a frame and a view, but no document. Developers coming from MFC wonder why. WTL is concentrating on the user-interface, and the document itself is never seen, so it is considered outside the scope of WTL. MFC provided binary files that usually were based on structured storage. The latest in document format is basing everything on XML. For Internet content, formats could include XHTML for hypertext, PNG for images, and SVG for vector diagrams and SMIL-Boston for multimedia. PNG is a binary file; all the others are based on XML. Applications that need custom formats should define their own XML schemas and write the appropriate code to load and save XML files based on this schema. Chapter 16 of “Inside ATL”, Sheperd/King, MS-Press, contains an interesting (but short) description of how to manage documents. No Active Document Support WTL does not support Active Documents (formerly called OLE Documents) directly. WTL/ATL provides no support for Active Document Servers. ATL does provide some indirect support for Active Document Containment, through the use of Internet Explorer. This could be used in a WTL application, but it certainly does not provide the same level of functionality as MFC’s extensive Active Document support. Microsoft has indicated that it currently has no plans to added support to WTL for Active Documents. However, it should be noted that the popularity of Active Documents is waning. It adds considerably to the complexity of the application. The vast majority of end-users never actually benefit from the functionality as they ignore it. As we move our data storage formats from binary structured storage files to XML data, this trend will accelerate. Chapter 15 of “Creating Lightweight Components with ATL”, Bates, SAMS, contains a well-designed project which shows Active Document Server and Containment. One-side note: If planning to become a container, you will have to offer structured storage. The default (provided) implementation sits above binary files. If your data file format is based on XML with your custom tags, then somehow you have to get the data coming from the server in structured storage streams to be converted into data that end up as tags inside your XML file. The way this could be done is to implement a COM component that exposes ILockBytes. It might have a method called WriteAt that pumps the data to the XML file: HRESULT __stdcall LockBytesObj::WriteAt( ULARGE_INTEGER ulOffset, const void *pv, ULONG cb,ULONG *pcbWritten){ *pcbWritten = write-to-xml-file(pv, 1, cb, fp); return (S_OK); } 93 WTL Developer’s Guide When loading a document, use the Win32 API StgCreateDocfileOnILockBytes, and base it one your LockBytes component, rather than the normal binary data file created with StgCreateDocFile. The storage you get back can be passed to other components, which might call: hres = pIStorage->CreateStream(szStreamName, STGM_CREATE | STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &pIStream); hres = pIStream->Write(myDataWrite, strlen(myDataWrite)+1, &ulLenWritten); No ISAPI Support WTL does not provide ISAPI functionality. ISAPI is a well-defined API that sits between web servers and extension DLLs that are dynamically loaded to service specialist browser requests. These extension DLLs are written by third-party developers and provide features that are not part of the web server. Their advantage over CGI scripts is that they are only loaded once compared to CGI that requires a new process to be launched for each request. The ATL Server functionality in Visual C++ version 7 will provide powerful features in the area. Microsoft’s MSDN web site has details of this. No WinInet Support WTL provides no special WinInet support. Again, this is non-UI, and there is no reason you could not use the WinInet APIs directly. No Wrappers for Threading or Synchronization WTL provides no specific wrappers for Win32’s thread creation functions or synchronization functions. WTL works extremely well with Win32 threads and synchronization kernel objects and simply needs no additional support. The code generated by the WTL AppWizard Multithreaded SDI option contains an excellent implementation of a multithreaded SDI application – the code related to the threads is actually in the application code, not inside WTL templates. Rest assured, WTL and threads get on very well together! No Database Support WTL does not offer any database support. The VC++ OLE DB Templates offer both consumer and provider templates for data access. These can of course be used from within a WTL application. MFC offers consumer classes to access existing database tables. It offers no provider functionality. MFC also has the likes of CRecordView and DDX_Field functions for UI functionality when accessing ODBC databases. WTL does not offer corresponding functionality. The lack of CRecordView is not a problem – indeed, it could be considered a benefit. Many people consider the MFC approach of forcing database record UI into the totally un-related MFC Document/View architecture as less than stellar design. They are 94 WTL Developer’s Guide simply different - so let them be different. Very many MFC database applications did not use CRecordView. The lack of DDX_Field is more troublesome. The idea behind MFC’s DDX_Field is that you will have a C++ class for the UI dialog, and a C++ class for the database table. Usually there is a (one-to-one?) mapping from what appears on the dialog to the fields in the database table. It is inefficient to store the data values both in the dialog’s C++ class and in the database table’s C++ class. MFC’s DDX_Field eliminates the need for the dialog C++ data-members, and the data is copied directly from the dialog box on screen to the member variables of the C++ database class. Hence it is much more efficient. WTL does not support this idea of DDX_Field. Applications based of WTL, which do not use too many database tables, could simply copy the data from the C++ class representing the dialog to the C++ class representing the database table. Applications that make more substantial use of databases might consider implementing their own version of DDX_Field. Development Issues To get the most out of WTL there are a number of development issues that need to be well understood. The Finished Product The size of applications built with WTL is very small. When you compile the AppWizard generated default SDI app the resulting EXE is only 64K in size. The TinyWTL sample app has extraneous features stripped out (so it does absolutely nothing), but is only 24K in size. A substantial commercial-strength application could be provided in less than 250K. Note that when such a file is compressed, it generally loses two-thirds of its size and hence could be rapidly downloaded over the Internet. After building the application, you then have to consider how to get it installed on the client’s machine. Assuming you are not using the CRT or other libraries such as DirectX, then when you compile a WTL project the resulting EXE or DLL is the only file you will need to deliver to end-users. Hence the size of your installation is quite small. Not only that, it is the only file that can cause problems. The less you deliver, the fewer things that can go wrong. Many applications vendors operate on tight margins and even one support call from an end-user can wipe out the profit on the sale to that customer. WTL applications will have no external dependencies, thus producing savings in this area. To install the application, you could create an installation script (e.g. based on Windows Installer). For simple app you could get away with just delivering the EXE or DLL. Sometimes you may wish to add entries to the registry. In this case, you can have the end-user execute a command line such as regsvr32 or /RegServer An alternative would be to make the EXE or DLL auto-sense that it has no registry entries the first time it is run, and to add them at that point. 95 WTL Developer’s Guide CRT The C-Runtime library is a set of C functions built above the Win32 API. It is commonly known as the CRT. The source code for the CRT is provided as part of the Visual C++ distribution (in the src directory). When you examine it you will see that the functions in the CRT are implemented in ANSI C, some call Win32 APIs directly and some call helper functions that in turn call Win32 APIs. The CRT is mostly a collection of functions from the Standard Library in ANSI C (printf, strtok etc.), but it is also some Win32-specific functions (the beginthreadex CRT function does some CRT initialization and then calls the Win32 function CreateThread). If you program exclusively to the ANSI C Standard library then you code will be portable. There are plenty of multi-million lines of code projects out there that can be equally well compiled on Windows and UNIX and with changes to less than 2% of code it will work just fine. The CRT does offer some extra functionality over the Win32 API in some areas, such as supporting floating-point values in strings. Other benefits of CRT include support for C++ exception handling, STL and the calling of constructors/destructors for global variables. If you are programming with ATL and WTL, your code will definitely not be portable to UNIX/LINUX platforms, and the question arises, should I link with the CRT? By and large the CRT duplicates functionality already in the Win32 API, so often the answer is no. It is possible to survive without C++ exception handling and the calling of constructors/destructors for global variables. By linking with the CRT, you are making your binary larger. You are also introducing a dependency on an external library (MSVCRT.dll), and multiple versions of this can exist. It is just one extra thing that can cause problems. The WTL AppWizard produces a project with settings so that the CRT is provided with the debug build (this is needed for the ATLASSERT calls), and the CRT is not provided with the release build, as the latter defines the _ATL_MIN_CRT macro in the preprocessor. If you are not paying attention here, sometime over the next few months you will develop a project that uses a CRT call, and it will not link in release mode. You will scratch your head, and hopefully come back here and say, aha! To avoid such rigmarole, note now that CRT IS NOT PRESENT IN RELEASE BUILDS BY DEFAULT. If you do decide to use the CRT, simply remove the _ATL_MIN_CRT macro in the preprocessor. If using the CRT, then an interesting point arises with multithreading. There are multiple versions of the CRT – some which are statically bound, and some that are dynamically loaded, and some are for single-thread apps, and some for multithreaded apps. If you are using the CRT functions from more than one thread, you definitely must use the multithreaded version. It is also imperative that the EXE and all DLLs in the project use the same setting for the CRT. It is to be found in the Project Settings dialog in DevStudio, Code Generation category. Error Handling In the past, the recommendation was to examine the return value from every function and carry out remedial action if needed. This often led to code bloating. For every ten 96 WTL Developer’s Guide lines of real code you could have forty lines of error checking and correction code and it was often difficult to debug and to follow the logical path through the code. Then along came C++ exception handling. The functional code was placed within trycatch blocks and when an exception was thrown, either directly within the try-catch block or in a function called from the block, and then the exception handler was activated. This reduced the amount of code in the project and made it clear. However, exceptions are time consuming to handle, do not cross language boundaries and more importantly, require the use of the CRT. If you are writing code which will run on multiple operating systems it makes sense to use the CRT and its features, including exception handling. However, as discussed earlier, if you are solely targeting Windows, and wish to produce small binaries without external dependencies (e.g. on a particular version of the CRT library - MSVCRT.dll), then you will wish to avoid linking with the CRT, which also means avoiding C++ exceptions. WTL is designed so that the CRT is not needed in release builds – so the obvious question we have to look at is how does WTL handle error situations. There are two main categories of errors: • Operational Errors – e.g. hard-disk is full while saving file • Programming Errors – e.g. divide by zero somewhere Operational errors will always occur and a well-written application should be able to detect them and respond appropriately – such as informing the user of the problem, suggesting how to fix it (e.g. make space on the disk), and the application should continue running. Programming errors should never occur in a well-written application. If they do occur it is a sign of a defect in the application. Something is seriously wrong. How can an application sensibly handle divide by zero – it is a design defect as user interface code should have only allowed the user enter a number not equal to zero. Such errors should never exist in an application in use by end-users. They will occur in debug builds, when developers are testing their applications. That is expected, but such problems should be eliminated by the time the product is ready to ship. For operational errors, WTL detects the problems and reports them back to application code, for example as a Boolean return code. This is a sample excerpt from WTL’s CFindFile class, which calls the Win32 API FindFirstFile. BOOL FindFile(LPCTSTR pstrName = NULL){ Close(); if(pstrName == NULL) pstrName = _T("*.*"); lstrcpy(m_fd.cFileName, pstrName); m_hFind = ::FindFirstFile(pstrName, &m_fd); if(m_hFind == INVALID_HANDLE_VALUE) return FALSE; . . . CFindFile::FindFile returns a BOOL, which states whether the operation succeeded or not. There can often be problems with pathnames (e.g. network is temporarily down) and applications need to respond appropriately. How this response is 97 WTL Developer’s Guide made is left to the application code – WTL just says the error occurred. Note that the last error is often that of the Win32 API that was called, so application code often can get more information from calling GetLastError (WTL provides no wrapper for it). For programming errors, WTL’s approach is quite different. The design goal could be stated as: “In debug builds, when we detect programming errors we will invoke an assertion, and we will test the application so well the problems won’t occur in release builds”. The great benefit of this approach is that the release build is very fast, as it is not clogged up with millions of error-checking lines of code. ATLASSERT statements are liberally scattered through out the WTL source code. ATLASSERTs get eliminated during release builds. During debug builds, then evolve into ASSERTE calls, which in turn evaluate the Boolean expression passed in as a parameter, and if FALSE then it calls _CrtDbgReport, which displays a message box with the string-ified version of the expression parameter and breaks in the debugger at the appropriate line. As an example, the WTL’s CListBox has this method to insert a string into a listbox: int InsertString(int nIndex, LPCTSTR lpszItem){ ATLASSERT(::IsWindow(m_hWnd)); return (int)::SendMessage(m_hWnd, LB_INSERTSTRING, nIndex, (LPARAM)lpszItem); } The window handle data member must be initialized before this call. If not, during the call it invokes an assertion. It is expected that application developers have a comprehensive test plan to cover all the code paths, and coding errors such as calling CListBox::InsertString without initializing the CListBox will be detected (and fixed) using the debug build, and when finished, shipped as a very fast release build to end-users. That’s the theory. What happens if we … eh … skimped on testing! What happens if an error occurs on a customer site and we are using release builds. WTL carries out no programmer error checking in release builds. The above CListBox::InsertString method gets compiled into a straight call to Win32’s SendMessage. In general, WTL methods that are wrappers just call the underlying Win32 function and WTL methods that perform more elaborate operations work as best they can. If errors occur, then either just the last error value is set, the method returns an error code and the process continues, or the process is terminated. If it is vital for your application that all error values are checked, you can certainly make the piece of application code which call the likes of CListBox ::InsertString itself call ATLASSERT before and / or after the method calls or add other error handling code which might even be invoked even during release builds, and handle the error some other way. 98 WTL Developer’s Guide The WTL Namespace The new ISO C++ standard introduced the concept of namespaces. Template and class names all have to be unique with a namespace. In the past there was a single global namespace within a project. This sometimes caused problems as libraries from different vendors often has name clashes. What ISO C++ added was the idea of creating extra namespaces, in addition to the global namespace. If we have a C++ namespace for WTL and a namespace for ATL and yet another namespace for a third-party library, then all three can happily be used together safe in the knowledge that even if classes from the different libraries use the same names there will be no problems. Every WTL header file starts with: namespace WTL { and ends with: }; //namespace WTL Unlike a class definition, which can only be declared once, a namespace can be partially declared in multiple header files, and all they entries become part of its namespace. When distinguishing between an MFC class and a WTL class of the same name, we can use the full WTL class name, which includes its namespace name. So we can talk about WTL::CRect. The namespace is used just for naming purposes – it has no other programmatic significance. Within your source code, you will have to decide whether you wish to use the longform, WTL::CRect or the short-form, CRect. If you are only using WTL, then it is likely adding WTL:: every time you use a template/class name will be tedious, so you will wish to avoid it. You can do this by adding the “using namespace WTL;” line to your source code. However, if using multiple libraries, and there is a possibility of name clashes, then you will have to use the full name. These lines are at the bottom of the WTL header file, atlapp.h: #ifndef _WTL_NO_AUTOMATIC_NAMESPACE using namespace WTL; #endif //!_WTL_NO_AUTOMATIC_NAMESPACE In the project generated by the WTL AppWizard, _WTL_NO_AUTOMATIC_NAMESPACE is not defined, so the using entry is in force, and the generated code does not use WTL:: prefix. If you want to take this out, just define or set it in the preprocessor definitions. Template/Class/Method Naming The naming conventions of WTL deliberately follow those of MFC. This makes sense, as both are based above the Win32 API, and most WTL developers will be coming from a MFC background. 99 WTL Developer’s Guide WTL has class names such as CEdit, CStatic and CDC. Handlers for Windows messages have names such as OnInitDialog and OnOK. Data exchange is carried out using DDX, and a method called DoDataExchange. Where there is a difference between MFC and Win32 naming, WTL follows Win32, which is sensible. Therefore for the Win32 UpDown control, WTL has a template called CUpDownCtrlT whereas MFC has a class called CSpinButtonCtrl. Versions of Windows One appealing aspect of WTL is its full support of all new UI features in Windows 2000 and Windows Me. WTL can also be used with all older versions of Windows, and the level of functionality it exposes can be controlled via a #define of WINVER. Complete list of Macros used in WTL The implementation of the WTL functionality is controlled from a variety of #defines. They are listed in the following table, along with the default and what impact they have. Note that some of them are preceded and succeeded by one or two underscores – and you better make sure you get it right! Some of the macros are Win32-based, some are ATL-based and some are new to WTL. Win32 Macros Which Impact WTL Macro Default Impact _WIN64 Not defined If defined, the 64-bit friendly version of _SettingChangeDlgProc is used; if not defined, the 64/32-bit friendly functions GetWindowLongPtr etc. are mapped to the 32bit only GetWindowLong etc. _WIN32_WINNT Not defined If defined as 0x400, assume that the target platform is Windows NT 4 or later; if defined as 05x00, assume target is Windows 2000 or later WINVER Set to 0x0400 in stdafx.h If defined as 0x400, assume the target platform is one of Windows 95 or Windows NT 4 or later; if defined as 0x0500, assume the target platform is Windows 98, Windows 2000 or later _WIN32_IE Set to 0x0400 in stdafx.h Defines minimum version of Internet Explorer which is installed _UNICODE Not defined If defined, TCHARs expect to wchar_t etc. and use W versions of OS APIs, if not defined, TCHAR expands to char etc., and use A version of OS APIs 100 WTL Developer’s Guide ATL Macros Which Impact WTL Macro _ATL_MIN_CRT Default Depends Impact States whether the C-Runtime Library should be included or not. In the WTL AppWizard generated project settings, it is defined for release builds, and it is not defined for debug builds. When _ATL_MIN_CRT is defined, _ATL_USE_DDX_FLOAT may not be defined WTL Macros That You Can Define Macro Default Impact _WTL_NO_ AUTOMATIC_ NAMESPACE Not defined If not defined, then the statement using namespace WTL; is added, other wise each WTL construct needs to be preceded with WTL:: _WTL_NO_CSTRING Not defined If defined, states that you do not wish to use CStrings anywhere in the app _CMDBAR_EXTRA_ TRACE Not defined If defined, then extra debug output is supplied for command bars _ATL_USE_DDX_ FLOAT Not defined If defined, then no floating point DDX support is available. If true then it is available – and it will use the CRT (so this cannot be defined at the same time _ATL_MIN_CRT is defined) _ATL_NO_REBAR_ SUPPORT Not defined If true, do not add support for the rebar control _ATL_NO_MSIMG Not defined If not defined, then link with "msimg32.lib" and add these methods to WTL::CDC: AlphaBlend, TransparentBlt and GradientFill _ATL_NO_OPENGL Not defined If not defined, then link with “opengl32.lib” and add 15 OpenGL wrapper methods to WTL:CDC _WTL_NO_WTYPES Not defined If defined, then provide the WTL classes CSize, CPoint & CRect _ATL_USE_NEW_ PRINTER_INFO Not defined If defined, add support for printer templates based on PRINTER_INFO_8 and PRINTER_INFO_9 _RICHEDIT_VER Set to 0x0100 in stdafx.h Specifies the level of rich edit support 101 WTL Developer’s Guide _RICHEDIT_ Not defined If defined, add declarations of CRichEditFontDialogImpl and CRichEditFontDialog, which provide font selection for the Rich Edit control OEMRESOURCE Not defined If defined, add implementations of AtlLoadSysBitmap and AtlLoadSysBitmapImage which call Win32’s LoadBitmap and LoadImage with a NULL first parameter, which means they use the predefined Win32 bitmaps _ATL_NO_COM Not defined If not defined, then for many WTL classes, where there is a member function that takes a string parameter, add an overridden member that takes a BSTR. Also, add the member functions AllocSysString and SetSysString to WTL’s CString _OLEAUTO_H_ Not defined This macro is only evaluated if _ATL_NO_COM is not defined If _OLEAUTO_H_ is defined, then add a member function GetTextBSTR to WTL’s CListBoxT, and overridden member functions GetText to CTreeItem and GetTextFace to CDC, both of which retrieve a BSTR WTL Macros That You May Use But Should Not Define Macro Default Impact _WTL_VER 0x0300 Version of WTL. It is defined in the WTL header file AtlApp.h, but is not used anywhere. Future versions might need it. __ATLSTR_H__ Depends States that WTL’s CStrings may be used. By default, this is defined if both atlmisc.h is included and _WTL_NO_CSTRING is not defined Debugging with WTL The development environment for Visual C++ includes many excellent productivity features. (It is unfortunate that most developers have not taken the time to explore them). One of every developer’s favorites is while executing an application under the debugger, if you float the cursor above a variable name or look at it in the data window, your see the variable’s value. Visual C++ automatically knows how to handle simple variables such as integers and character pointers. But how should it display more complex data structures and classes? 102 WTL Developer’s Guide When you install Visual C++, a text file called AUTOEXP.DAT is placed under \Common\MSDev98\Bin\. This file contains instructions, using simple text-based formatting rules, telling the Visual C++ environment how to render specific data structures. The formatting rules are explained at the top of the file. By default, AUTOEXP.DAT contains entries for common Win32 structures, such as POINT and RECT (but not SIZE), and many common MFC classes. There are no entries for WTL. But that is not a problem, as we can add our own. Open up the file in Notepad and pop the following entries at the bottom of the file. WTL::CString = WTL::CSize =cx= cy= WTL::CPoint =x= y= WTL::CRect =top= bottom= left= right= WTL::CDCT<*> =hDC= WTL::CClientDC=hDC= hWnd = WTL::CWindowDC=hDC= hWnd = WTL::CPaintDC =hDC= hWnd = WTL::CServerAppModule =m_hEventShutdown= WTL::CPenT<*> =m_hPen= WTL::CBrushT<*> = m_hBrush= WTL::CFontT<*> = m_hFont= WTL::CBitmapT<*> = m_hBitmap= WTL::CPaletteT<*> = m_hPalette= WTL::CRgnT<*> = m_hRgn= ATL::CWindowImpl<*>=<,t> hWnd= ATL::CWindow=<,t> hWnd= Note that the namespace specification for the last two entries is ATL:: and not WTL::. Structures and classes that do not have entries in AUTOEXP.DAT are rendered using “…”, which is not very informative. The following two screen dumps are from a WTL application running under DevStudio’s Debugger, and the cursor is floating over a WTL variable of type CString. This is how a WTL CString is displayed without (left picture) and with (right picture) the AUTOEXP.DAT entries shown above: The AUTOEXP.DAT entries are of the format: type=[text] Type indicates the data-type (including the namespace name). When the debugger detects a variable of this type, it renders a string based on the rest of the entry. The [text] field is optionally, but if present is rendered as is. The field states which data member of the class or structure should be displayed, and the format field states how it should be displayed. So the entry . . .: WTL::CString = 103 WTL Developer’s Guide . . . says please display a WTL::CString as the m_pchData field, and display it as a NULL terminated string, and the entry. . : WTL::CBrushT<*> = m_hBrush= . . . says please display any class based on the WTL::CBrushT template as the text string “m_hBrush=” and the actual data value of the m_hBrush data member (in the default integer format). If an entry does not exist for a class, and if that class is derived from another class or a structure, then an attempt is made to find an entry for the base class or structure. Because of this, even without the entries for WTL::CPoint and WTL::CRect, you will see tooltips for them, because they are derived from the Win32 data types, POINT and RECT, and entries for these structures do exist in the default AUTOEXP.DAT. As there is no entry for Win32’s SIZE structure, then a tooltip is not displayed for WTL::CSize. To ensure the right match, it is important that the namespace name is correct and you are clear about whether you are matching a class or a template. For a template you must use <*>, and for a class, you may not use it. Note that CWindow is a class, not a template, and hence is entered without the <*>. All the standard and common control wrappers are based on CWindow. In addition to seeing the actual value of the variable, this feature is also helpful to know it if it NULL (e.g. non-initialized), or some value that is impossible (e.g. illegal memory address), thus indicating memory problems. Detailed Comparison between MFC and WTL Most developers starting with WTL will have already developed MFC applications. They will have an excellent knowledge of MFC and have used it in quite substantial projects. They will wish in some cases to leverage that knowledge to quickly develop WTL applications and sometimes might even need to port from MFC to WTL. To assist these, here we examine how the functionality of each of the MFC classes can be provided in a WTL application. We also look at what is in WTL that is not supported inside MFC. We follow the categories in the MFC hierarchy charts. Application Architecture • CCmdTarget – command routing is done in ATL message maps • CWinThread – use Win32 Multithreading APIs • CWinApp – CAppModule or CServerAppModule • COleControlModule – ATL ActiveX control infrastructure 104 WTL Developer’s Guide • CDocTemplate, CSingleDocTemplate, CMultiDocTemplate – WTL has no document support – use XML code • COleObjectFactory – Use ATL’s wide range of class factories • COleTemplateServer / COleMessageFilter – WTL has no Active Document support • COleDataSource, COleDropSource, COleDropTarget – WTL has no universal data transfer support – could implement directly • CConnectionPoint – Use ATL’s connection point templates and proxy generator • CDocument / CHtmlEditDoc / CRichEditDoc – WTL has no document support – you will need to manually implement these features • COleDocument / COleLinkingDoc / COleServerDoc, CDocItem/ COleClientItem/ COleDocObjectItem / COleServerItem / CDocObjectServerItem / CDocObjectServer – WTL does not support Active Documents • CRichEditCntrItem – WTL provides a template for the rich edit control, but has no support from Active Document server/containment for it, as this MFC class provides • COleControlContainer / COleControlSite – Use ATL’s ActiveX control containment functionality (CAxWindow etc.) Window Support • CWnd – Use ATL’s CWindow, CWindowImpl and CContainedWindow Frame Window • CFrameWnd – Use WTL’s CFrameWindowImpl • CDockFrameWnd – WTL provides no docking support • CMDIChildWnd – Use WTL’s CMDIChildWindowImpl • CMDIFrameWnd – Use WTL’s CMDIFrameWindowImpl • CMDIDockSiteFrame – WTL provides no docking support • COleIPFrameWnd – WTL provides no in-place activation support • CSplitterWnd – Use WTL’s CSplitterImpl, CSplitterWindowImpl and CSplitterWindowT Question - CTabMDIChildWnd and CTabFrameWnd are listed in the MFC hierarchy chart, but no doc provided – Any ideas what they are? VC++ v6.1 105 WTL Developer’s Guide Property Sheets • CPropertySheet – Use WTL’s CPropertySheetWindow, CPropertySheetImpl and CPropertySheet Dialog Boxes • CDialog – Use WTL’s CDialogImpl • CCommonDialog – WTL’s CCommonDialogImplBase • CColorDialog – WTL’s CColorDialogImpl and CColorDialog • CFileDialog – WTL’s CFileDialogImpl and CFileDialog • CFindReplaceDialog – WTL’s CFindReplaceDialog • CFontDialog – CFontDialogImpl and CFontDialog • COleDialog / COleBusyDialog / COleChangeIconDialog / COleChangeSourceDialog / COleConvertDialog / COleInsertDialog / COleLinksDialog / COleUpdateDialog / COlePasteSpecialDialog / COlePropertiesDialog – WTL offers no Active Document support at all • CPageSetupDialog – WTL’s CPageSetupDialogImpl and CPageSetupDialog • CPrintDialog – WTL’s CPrintDialogImpl and CPrintDialog • COlePropertyPage – ATL’s ActiveX control functionality provides similar support • CPropertyPage – WTL’s CPropertyPageWindow, CPropertyPageImpl and CPropertyPage Question - CDHtmlSinkHandler / CDHtmlEventSink / CDHtmlEventHandler / CDHtmlHostHandler/CDHtmlDialog /CMultiPageDHtmlDialog are listed in the MFC hierarchy chart but no doc provided – Any ideas what they are? VC++ v6.1 Views • CView – WTL offers no special class for the view. The WTL AppWizard generated application code does contain a view class derived from CWindowImpl –this simplifies the design immensely • CScrollView – Use WTL’s CScrollImpl, CScrollWindowImpl, CMapScrollImpl, CMapScrollWIndowImpl and CFSBWindowT • COleDBRecordView/CRecordView – WTL’s offers no corresponding functionality – which is a good, as trying to force database editing into MFC’s Document/View infrastructure was not a good approach – use WTL’s CDialogImpl< MYCLASS > [form view] when needed 106 WTL Developer’s Guide • CCtrlView – In MFC, this class derives from CView and is the parent of other views based on controls, such CEditView and CTreeView. WTL does not provide such an intermediary class WTL offers no specific templates for views based controls, as it has no need for them. It does offer a complete range – they are to be implemented by deriving from the CWindowImpl template based on the control in question (This is also supported in step 2 in the WTL AppWizard). • CEditView –CWindowImpl< MYCLASS, CEdit> • CListView - CWindowImpl< MYCLASS, CListViewCtrl> • CRichEditView CWindowImpl< MYCLASS, CRichEditCtrl > • CTreeView - CWindowImpl< MYCLASS, CTreeViewCtrl> • CFormView - CDialogImpl< MYCLASS > • CHtmlEditView – This class is listed in the MFC hierarchy chart, along with CHtmlEditBase / CHtmlEditCtrl, but no class documentation on the MSDN and no implementation in the VC++ installation is available – we assume it will be provided in a future version – and will be based on MSHTML Editing Platform. For more details. See: http://msdn.microsoft.com/workshop/browser/mshtmleditplaf.asp WTL provides no functionality to edit HTML, but this could be manually implemented • CHtmlView – CWindowImpl< MYCLASS, CAxWindow>; The ATL CAxWindow uses the Web Browser control internally Controls Both MFC and WTL offer a complete range of wrappers for Window controls. WTL’s controls are templates (names ends in T) and a default implementation based on CWindow is provided (without the T). Note that WTL always uses the correct OS name (e.g. UpDown control, TrackBar control), whereas MFC sometimes invents its own (e.g. SpinButtonCtrl or SliderCtrl) • CAnimateCtrl - CAnimateCtrlT CAnimateCtrl; • CButton - CButtonT CButton • CBitmapButton - CBitmapButton : public CBitmapButtonImpl • CComboBox - CComboBoxT CComboBox • CComboBoxEx - CComboBoxExT CComboBoxEx 107 WTL Developer’s Guide • CDataTimeCtrl CDateTimePickerCtrlTCDateTimePickerCtrl • CEdit - CEditT CEdit [Note: WTL also offers CEditCommands, to assist with the handling of common editing commands such as clear/copy/page/undo ] • CHeaderCtrl - CHeaderCtrlT CHeaderCtrl • CHotKeyCtrl - CHotKeyCtrlT CHotKeyCtrl • CIPAddressCtrl CIPAddressCtrlT CIPAddressCtrl • CListBox - CListBoxT CListBox • CCheckListBox – CCheckListViewCtrlImpl/CCheckListViewCtrl • CDragListBox - CDragListBoxT CDragListBox; [Note: WTL also offers CDragListNotifyImpl to help with notifications] • CListCtrl - CListViewCtrlT CListViewCtrl • CMonthCalCtrl CMonthCalendarCtrlT CMonthCalendarCtrl • CProgressCtrl - CProgressBarCtrlT CProgressBarCtrl • CReBarCtrl - CReBarCtrlT CReBarCtrl [Note that WTL offers extensive commandbar support, based on the rebar] • CRichEditCtrl - CRichEditCtrlT CRichEditCtrl [Note: WTL also offers CRichEditCommands to help with editing command processing • CScrollBar - CScrollBarT CScrollBar • CSliderCtrl CTrackBarCtrlT CTrackBarCtrl • CSpinButtonCtrl CUpDownCtrlT CUpDownCtrl • CStatic - CStaticT CStatic • CStatusBarCtrl - CStatusBarCtrlT CStatusBarCtrl [Note that WTL offers substantial extra functionality in CMultiPaneStatusBarCtrlImpl / CMultiPaneStatusBarCtrl] • CTabCtrl - CTabCtrlT CTabCtrl 108 WTL Developer’s Guide • CToolBarCtrl CToolBarCtrlT CToolBarCtrl • CToolTipCtrl CToolTipCtrlT CToolTipCtrl [Note that WTL also offers CToolInfo, to help with the initialization of Win32’s TOOLINFO structure] • CTreeCtrl CTreeViewCtrlT CTreeViewCtrl and CTreeViewCtrlExT:public CTreeViewCtrlT< TBase > and CTreeItem • CHtmlEditBase / CHtmlEditCtrl – See note regarding these classes in the earlier discussion of MFC’s CHtmlEditView • COleControl – This is MFC’s base class for ActiveX controls – ATL offers equivalent (really, far superior) functionality Exceptions • CException, CArchiveException, CDaoException, CDBException, CFileException, CInternetException, CMemoryException, CNotSupportException, COleException, CResourceException, CUserException – WTL has no exception classes. File Services • CFile, CMemFile, CSharedFile, COleStreamFile, CMonikerFile, CAsyncMonikerFile, CDataPathProperty, CCachedDataPathProperty, CSocketFIle, CInternetFIle, CGopherFIle, CHttpFile – WTL offers no handling file support • CRecentFileList – Use WTL’s CRecentDocumentList Graphical Drawing • CDC – WTL’s CDC • CClientDC – WTL’s CClientDC • CMetaFileDC – WTL’s CEnhMetaFileDC [Note that WTL also offers a number of additional classes for enhanced metafiles] • CPaintDC – WTL’s CPaintDC • CWindowDC – WTL’s CWindowDC Control Support • CDockState – WTL offers no support for docking • CImageList – WTL’s CImageList 109 WTL Developer’s Guide Graphical Drawing Objects • CGdiObject – WTL has no common base for its GDI templates • CBitmap – WTL’s CBitmap • CBrush – WTL’s CBrush • CFont – WTL’s CFont • CPalette – WTL’s CPalette • CPen – WTL’s CPen • CRgn – WTL’s CRgn Menu • CMenu – WTL’s CMenu [WTL also offers CMenuItemInfo, a helper to access Win32’s MENUINFO structure] Command Line • CCommandLineInfo – WTL’s CServerAppModule has the FindOneOf and ParseCommandLine helper functions – [In the WTL AppWizard, step 1, if Act as a COM Server is selected, the generated code uses CServerAppModule, otherwise it uses CAppModule] MFC’s Arrays/Lists/Maps/Typed Template Collections WTL has no container classes. You may either use the excellent STL (many of whose features are dependent on the CRT) or ATL offers a limited range of collection classes that are not dependent on the CRT. MFC’s ODBC/DAO/Synchronization/Sockets/Internet Services WTL offers no equivalent support for MFC classes in these areas. In WTL applications, MFC’s ODBC/DAO functionality can be replaced with the VC++ OLE DB Consumer templates. Synchronization can be provided by using the relevant Win32 APIs directly. Sockets functionality can be provided by direct calls to WinSock2. Internet services can be provided by direct calls to the WinInet APIs. Internet Server API (ISAPI) MFC offers CHttpArgList, CHtmlStream, CHttpFilter, CHttpFilterControl, CHttpServer, CHttpServerContext, CInternetSession, CInternetConnection, CFtpConnection, CGopherConnection, CHttpConnection, CFtpFindFile, CGopherFindFile and CGopherLocator. ATL will cover this area comprehensively in Visual C++ v7, in a technology to be known as ATL Server. For more details, see http://msdn.microsoft.com/vstudio/nextgen/default.asp 110 WTL Developer’s Guide Run-time Object Model Support • CArchive – This is used to serialize data, which might ultimately end up in a document or sent over a socket – WTL offers no similar functionality • CDumpContext – ATL/WTL does not fully cover this, but some of the debugging facilities within ATL partially help – for all the features, you would have to implement a dump function for each class, and walk some collection of class instances and call the function • CRuntimeClass – Use ISO C++ RTTI if needed – but in general it is considered bad practice to use either of these, as they is costly in terms of performance/size and can cause problems in future (if client code has some big switch doing class specific processing, what happens if later a new class is added – then client has to be change – (class specific code should be inside the class!) Simple Value Types • CPoint – WTL’s CPoint • CRect – WTL’s CRect • CSize – WTL’s CSize • CString – WTL’s CString • CTime – This is MFC’s wrapper for absolute time, stored as time_t - WTL offers no similar class • CTimeSpan - This is MFC’s wrapper for the interval between times, stored as time_t - WTL offers no similar class Structures • CCreateContext – MFC uses this when creating frame windows and views for a document. WTL provides no document support Question - CHttpArg is listed in the MFC Hierarchy Chart but no documentation and no implementation is provided – Any ideas what it is? VC++ v6.1 • CMemoryState – This is used for debugging memory leaks – WTL/ATL provides no similar functionality, apart from ATL’s functionality which traces interface reference counting • COleSafeArray – WTL offers no SafeArray support – one can use the Win32 SafeArray APIs directly [Should this not be part of Automation Types section in the MFC hierarchy?] • CPrintInfo – WTL’s CPrintJob 111 WTL Developer’s Guide Support Classes • CCmdUI – WTL’s CUpdateUI • COleCmdUI – WTL provides no Active Document Support • CDaoFieldExchange – WTL/ATL provides no DAO support – DAO is a legacy API, and for new projects you will be much better off to using VC++ OLE DB Data Consumer templates, which can be used inside WTL applications • CDataExchange – WTL’s CWinDataExchange • CDbVariant – This is a MFC wrapper for VARIANTs that is not based on OLE – useful for the MFC ODBC data access – For WTL, use the OLE DB data consumer templates • CFieldExchange - WTL/ATL provides no ODBC support – ODBC is a legacy API, and for new projects you will be much better off using VC++ OLE DB Data Consumer Templates, which can be used inside WTL applications Question - CImage is listed in the MFC Hierarchy Chart but no documentation and no implementation is provided – Any ideas what it is? VC++ v6.1 • COccManager – This is a MFC helper class which provides ActiveX control Containment – Use ATL’s CAxWindow • COleDataObject – Use ATL’s IDataObjectImpl • COleDispatchDriver – Use ATL’s CComDispatchDriver • CPropExchange – Use ATL’s Active Control properties functionality • CRectTracker – WTL offers no similar class • CWaitCursor – WTL’s CWaitCursor Question – CWinDHtmlDDX & CDHtmlDDX are listed in the MFC Hierarchy Chart but no documentation and no implementation is provided – Any ideas what they are? VC++ v6.1 OLE Type Wrappers • CFontHolder – This is a wrapper for Platform SDK IFontDisp interface returned as the Font stock property for ActiveX controls. WTL/ATL provides no similar support – you can simply use IFontDisp (and IFont) directly • CPictureHolder - This is a wrapper for Platform SDK IPictureDisp interface returned as the MouseIcon and Picture stock properties for ActiveX controls. WTL/ATL provides no similar support – you can simply use IPictureDisp (and IPicture) directly 112 WTL Developer’s Guide Automation Types • COleCurrency / COleDateTime / COleDateTimeSpan – These are wrappers for various fields in Automation’s VARIANT structure. WTL offers no similar support – you can use ATL’s CComVariant or VC++’s native _variant_ • COleVariant – CComVariant or VC++ native _variant_ What is in WTL but not MFC Unlike WTL, MFC does not provide wrappers for the folder dialog or the rich edit font dialog (WTL provides CFolderDialogImpl / CFolderDialog and CRichEditFontDialog / CRichEditFontDialogImpl). MFC does not support equivalents of WTL’s CEditCommands or CRichEditCommands. MFC does support menubars and toolbars, but does not provide the extra functionality of WTL’s command bars. MFC does not provide a wrapper for HTREEITEM, whereas WTL provides CTreeItem. WTL provides excellent enhanced metafile support, beyond what MFC offers. WTL offers these additional control classes, which have no direct equivalent in MFC: • CPageCtrl • CFlatScrollBarImpl • CCustomDraw • CHyperLink WTL’s CString To begin to get a feel for WTL’s functionality, we will here examine one of its classes – CString. The WTL CString class manages a character buffer. It provides virtually every conceivable operation you would like to carry out on a string. It offers all the functionality of MFC’s CString class – it is such a perfect emulation of MFC’s CString that their class documentation is interchangeable! Every method in MFC’s CString is in WTL’s CString. A string in WTL’s CString is stored in a contiguous array of characters. WTL supports 16-bit (UNICODE) and 8-bit characters (ASCII) and a mixture (MBCS). CString offers a number of constructors that takes as the input parameter strings in different formats. CString(); 113 WTL Developer’s Guide CString(const CString& stringSrc); CString(TCHAR ch, int nRepeat = 1); CString(LPCSTR lpsz); CString(LPCWSTR lpsz); CString(LPCTSTR lpch, int nLength); CString(const unsigned char* psz); The internal buffer will automatically grow and contract as needed. Member functions are provided to get the length of the string, to determine it is NULL and to empty it. int GetLength() const; BOOL IsEmpty() const; void Empty(); CStrings are reference-counted, so that when passing strings around the reference count is incremented and the entire string is not copied. Only if a string is about to be changed will a separate copy of the shared string data be made. If a CString is going to be used from multiple threads, and that use includes writing from at least one thread, then care must be taken with synchronization. Member functions are provides to access the CString as if it were a character array. The LPCTSTR() operator means a CString may be passed in anywhere a LPCTSTR is expected. TCHAR GetAt(int nIndex) const; TCHAR operator[](int nIndex) const; void SetAt(int nIndex, TCHAR ch); operator LPCTSTR() const; A complete range of assignment & concatenation operators is supported. Member functions are provided for string comparisons, extractions, conversions and trimming. int Compare(LPCTSTR lpsz) const; int CompareNoCase(LPCTSTR lpsz) const; int Collate(LPCTSTR lpsz) const; CString Mid(int nFirst, int nCount) const; CString Mid(int nFirst) const; CString Left(int nCount) const; CString Right(int nCount) const; CString SpanIncluding(LPCTSTR lpszCharSet) const; CString SpanExcluding(LPCTSTR lpszCharSet) const; void MakeUpper(); void MakeLower(); void MakeReverse(); void TrimRight(); void TrimLeft(); void AnsiToOem(); void OemToAnsi(); Member functions are provided for string replacement, insertions & appending. int Replace(TCHAR chOld, TCHAR chNew); int Insert(int nIndex, TCHAR ch); int Insert(int nIndex, LPCTSTR pstr); 114 WTL Developer’s Guide int Delete(int nIndex, int nCount = 1); int Find(TCHAR ch) const; int ReverseFind(TCHAR ch) const; int FindOneOf(LPCTSTR lpszCharSet) const; int Find(LPCTSTR lpszSub) const; const CString& Append(int n) void __cdecl Format(LPCTSTR lpszFormat, ...); void __cdecl Format(UINT nFormatID, ...); BOOL __cdecl FormatMessage(LPCTSTR lpszFormat, ...); BOOL __cdecl FormatMessage(UINT nFormatID, ...); WTL’s CString supports loading strings from a resource file: // Windows support BOOL LoadString(UINT nID); It also supports Automation BSTRs. BSTR AllocSysString() const; BSTR SetSysString(BSTR* pbstr) const; These methods are provided to allow access to the data for direct manipulation LPTSTR GetBuffer(int nMinBufLength); void ReleaseBuffer(int nNewLength = -1); LPTSTR GetBufferSetLength(int nNewLength); void FreeExtra(); After calling GetBuffer, and changing the data, it is important to later call ReleaseBuffer, so that CString again takes responsibility for the buffer. To lock and unlock the buffer for reference counting uses these methods: LPTSTR LockBuffer(); void UnlockBuffer(); From just examining this one WTL class, it should become clear that when WTL provides support for a feature, it is very comprehensive. 115 WTL Developer’s Guide Chapter 5 The WTL AppWizard Objectives The objectives of this chapter are to: • Use the WTL AppWizard to create all support application types • Review the generated build settings • Examine the application features available in Step 1 of the AppWizard – COM Server and ActiveX hosting • Examine the UI features available in Step 2 • Conduct a code walkthrough of all the generated source code Just Say “HelloWorld” In the time-honored Kernighan & Ritchie tradition of computerdom, we had better start our exploration of WTL with the simplest HelloWorld application we can create. We could either display the text in a dialog box or in an SDI or MDI application window. We could display it as ordinary text with a view, or a listbox, richeditbox, treeview, listview or each a HTML page. It is such a difficult design decision; let’s try them all! Naturally we do not wish to type (that’s too much effort) so we will use the WTL AppWizard and hopefully if it is any good it won’t have too much effort. Of course we wish to examine how to use the AppWizard, but it is very simple to use and only contains two steps – it is almost trivial, but the real magic is what it produces. Our goal here is to comprehensively understand the generated application code. After all, we will use the AppWizard for two minutes at the beginning or a project, but have to spend six months or more building our application above the foundation of the code it generates. Modal Dialog-Based Application To use the WTL AppWizard to create a dialog-based application, start DevStudio, select File/New, select the ATL/WTL AppWizard, and the following first Wizard step will appear. As we wish to create a dialog-based app, select Dialog-Based and select Modal (we will examine modeless dialogs shortly). As we have selected Dialog-Based, all step 2 options are disabled (as they relate to standard application windows), so we can select finish at the bottom of step 1. Then we use the ResourceView to edit the dialog IDD_MAINDLG and add a static control with the text “HelloWorld”. Compile and run. 116 WTL Developer’s Guide The wizard-weary gray-haired Visual C++ developer will recall his/her early MFC days and suspiciously think, “WTL is easy, but what is really happening – with MFC there was a million lines of code generated and it was quite difficult to understand”. With WTL, things are actually different, and somewhat easier. The External Dependencies are based directly on the WTL header files. There is no WTL DLL to use – only the header files are needed. The resources are in a file called .rc and there is an icon .ico. Note that the icon used for the top-left corner defaults to “ATL”. There are three header files, stdafx.h, resources.h and maindlg.h. All of which are quite small. Stdafx.h contains these #defines and #includes: #define WINVER 0x0400 #define _WIN32_IE 0x0400 #define _RICHEDIT_VER 0x0100 #include #include extern CAppModule _Module; #include 117 WTL Developer’s Guide Resources.h contains defines for resource Ids, such as: #define IDD_ABOUTBOX #define IDR_MAINFRAME #define IDD_MAINDLG 100 128 129 Maindlg.h contains the implementation of the CMainDlg class. It is the core functionality of this application. It derives from the ATL Windowing class CDialogImpl. It contains an ATL Windowing message map, which maps buttons Ids to member functions. It also maps WM_INITDIALOG to OnInitDialog (all very familiar to MFC developers!). class CMainDlg : public CDialogImpl { public: enum { IDD = IDD_MAINDLG }; BEGIN_MSG_MAP(CMainDlg) MESSAGE_HANDLER(WM_INITDIALOG, OnInitDialog) COMMAND_ID_HANDLER(ID_APP_ABOUT, OnAppAbout) COMMAND_ID_HANDLER(IDOK, OnOK) COMMAND_ID_HANDLER(IDCANCEL, OnCancel) END_MSG_MAP() LRESULT OnInitDialog(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/){ // center the dialog and set icons . . . return TRUE; } 118 WTL Developer’s Guide LRESULT OnAppAbout(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ CSimpleDialog dlg; dlg.DoModal(); return 0; } LRESULT OnOK(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/) { // TODO: Add validation code EndDialog(wID); return 0; } LRESULT OnCancel(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ EndDialog(wID); return 0; } }; Two source files are generated, stdafx.cpp and .cpp. Stdafx.cpp contains just two lines, #includes of stdafx.h and the ATL header file, atlimpl.h. The main source file, .cpp, #includes various WTL header files and maindlg.h, then create a variable called _Module based on the WTL data type CAppModule, and provides the following WinMain: int WINAPI _tWinMain(HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPTSTR lpstrCmdLine, int nCmdShow){ HRESULT hRes = ::CoInitialize(NULL); ATLASSERT(SUCCEEDED(hRes)); // initialize the common controls . . . hRes = _Module.Init(NULL, hInstance); ATLASSERT(SUCCEEDED(hRes)); CMainDlg dlgMain; int nRet = dlgMain.DoModal(); _Module.Term(); ::CoUninitialize(); return nRet; } Note that there are calls to CoInitialize and CoUninitialize, even though we have not selected the COM Server application feature. That is it. It is easy to really understand your first WTL application within ten minutes. Default Project Settings The WTL AppWizard produces project files with two pre-configured build settings – one for release and one for debug. This contrasts with the ATL AppWizard, which produces builds for Debug, Unicode Debug, Release MinSize, ReleaseMinDependency, Unicode Release MinSize and UnicodeMinDependency. At first one might consider ATL to be more helpful – but as we will see very often during our examination of WTL, the minimalist approach that WTL offers is better. 119 WTL Developer’s Guide The vast majority of developers just need two builds – one for debugging and one for release. Decisions concerning UNICODE support or the tradeoff of minimum build versus minimum size is made once during development, and it is very rare that developers need to have multiple builds for these alternatives. They make their choice and that is that. Having all these extra build settings hanging around can cause problems and confusion. Project settings changes can incorrectly be made to the wrong build, and they seemingly have no effect. Groups of developers working on the same project have to all know which one to use. In most projects involving ATL, all but two of the build settings become “dormant”, are never updated but are not deleted. It just adds to potential problems. Smart developers get rid of these as soon as possible. Unicode is supported if _UNICODE is defined. The minimum sized binary is achieved if _ATL_DLL and _ATL_MIN_CRT are defined. The minimum dependency is achieved if _ATL_STATIC_REGISTRY and _ATL_MIN_CRT are defined. When developing with WTL, you should decide which of these settings you desire and simply add they to the release and debug builds. If you really need additional builds (thought it is unlikely) they simply create them. The default preprocessor definition for a WTL AppWizard generated project is for the DEBUG build: WIN32,_DEBUG,_WINDOWS,_MBCS,STRICT And for the release build: WIN32,NDEBUG,_WINDOWS,_MBCS,STRICT,_ATL_MIN_CRT Precompiled headers are supported through stdafx.h and stdafx.cpp. Normally WTL does not use the C Run-Time Library. This is enforced when _ATL_MIN_CRT is added, which is the default setting for the release build. If you do decide to use CRT, then remove _ATL_MIN_CRT. If using the CRT, one point you must be careful with is which version you are using. For efficiency reasons, there are multiple versions of the CRT. Sometimes applications will only have one thread and with the CRT to be as fast as possible. Other times applications will have multiple threads and wish to call CRT functions concurrently, so a special version of the CRT exists with synchronization as appropriate. This is slightly slower so is only used with multi-threaded applications. How does the compiler know which one your application should use? It does not. Instead, it lets you tell it, via a build setting. In DevStudio, Project menu, select “Settings …” . The Project Settings dialog will be displayed. Select the C/C++ property page and the Code Generation category. The Use run-time library list shows the available options regarding which CRT to use. It is almost always best if an EXE and all its DLLs use the same setting for this. It is often disastrous if an EXE with multiple threads that uses the CRT only links with the singlethreaded version of the CRT. 120 WTL Developer’s Guide The WTL AppWizard generate projects have this setting set to Multithreaded, which makes sense. By default WTL projects do not use the CRT, but if they do they are thread-safe. Exception handling is enabled for debug build and disabled for release builds. C++’s RTTI is not enabled for either Debug or Release builds. Most developers do not need RTTI. In WTL projects that are not COM servers, there are neither custom steps, pre-link steps nor post-build steps. In WTL projects that are COM servers, there is this custom step that registers the EXE if its timestamp is more recent than the dummy trg timestamp file: "$(TargetPath)" /RegServer echo regsvr32 exec. time > "$(OutDir)\regsvr32.trg" echo Server registration done! SDI Application Next, we will try to create a HelloWorld application that is presented as a typical main window in an application. We need to create a new project, which we call HelloWorldWin. In the WTL AppWizard, select the SDI Application project type. Leave both application features unchecked. Then select next. In the second step of the AppWizard use the default selections. Open the view.h file, and in the OnPaint method replace the TODO comment with this one line of code: dc.TextOut(0, 0, TEXT("HelloWorld")); When you build and run the application the following window will appear: 121 WTL Developer’s Guide Let’s look at what is happening behind the scenes. There are many more entries in the resources. Additional entries are provided for the menubar, toolbar, and versioning. There is a string table with almost fifty strings, which are used in various places in the application. The files stdafx.h, stdafx.cpp and resources.h are the same are for the dialog-based sample. The about box code is in a separate class, CAboutDlg in a separate file, aboutdlg.h. The three files that are different in the sample are .cpp, view.h and mainfrm.h. .cpp contains two functions – WinMain which is similar to the Dlg implementation, with the change that the dlg.DoModal code is replaced by a called to the second function in the file, Run. int Run(LPTSTR /*lpstrCmdLine*/ = NULL, int nCmdShow = SW_SHOWDEFAULT){ CMessageLoop theLoop; _Module.AddMessageLoop(&theLoop); CMainFrame wndMain; if(wndMain.CreateEx() == NULL) { ATLTRACE(_T("Main window creation failed!\n")); return 0; } wndMain.ShowWindow(nCmdShow); 122 WTL Developer’s Guide int nRet = theLoop.Run(); _Module.RemoveMessageLoop(); return nRet; } Run sets up instances of a class within our application called CMainFrame (defined in our mainfrm.h) and the WTL class CMessageLoop. _Module maintains a map of threadId / messageloop pairs, and calls to AddMessageLoop and RemoveMessageLoop edit it as needed. Later we will see the use of GetMessageLoop that retrieves the message loop for the current thread. CMessageLoop manages arrays of message filters and idle handlers and performs the traditional Win32 message dispatching. Idle handlers are functions that are called when there are no more messages on the queue. They are usually used for background processing. Message filters are functions that can be used to customize how messages get translated, before the main message loop gets at them. The WTL AppWizard has generated an implementation of a class called CMainFrame in mainfrm.h for us. CMainFrame manage the main application window and within it must render the menubar, toolbar, statusbar and a separate client area windows called the view. In effect, it performs the same role as the MFC class of the same name. class CMainFrame : public CFrameWindowImpl, public CUpdateUI, public CMessageFilter, public CIdleHandler{ public: DECLARE_FRAME_WND_CLASS(NULL, IDR_MAINFRAME) CHelloWorldWinView m_view; CCommandBarCtrl m_CmdBar; BEGIN_MSG_MAP(CMainFrame) MESSAGE_HANDLER(WM_CREATE, OnCreate) COMMAND_ID_HANDLER(ID_APP_EXIT, OnFileExit) COMMAND_ID_HANDLER(ID_FILE_NEW, OnFileNew) COMMAND_ID_HANDLER(ID_VIEW_TOOLBAR, OnViewToolBar) COMMAND_ID_HANDLER(ID_VIEW_STATUS_BAR, OnViewStatusBar) COMMAND_ID_HANDLER(ID_APP_ABOUT, OnAppAbout) CHAIN_MSG_MAP(CUpdateUI) CHAIN_MSG_MAP(CFrameWindowImpl) END_MSG_MAP() BEGIN_UPDATE_UI_MAP(CMainFrame) UPDATE_ELEMENT(ID_VIEW_TOOLBAR, UPDUI_MENUPOPUP) UPDATE_ELEMENT(ID_VIEW_STATUS_BAR, UPDUI_MENUPOPUP) END_UPDATE_UI_MAP() LRESULT OnCreate(. . .) { // create command bar window, set up toolbar and call // the view’s Create function . . . CMessageLoop* pLoop = _Module.GetMessageLoop(); 123 WTL Developer’s Guide pLoop->AddMessageFilter(this); pLoop->AddIdleHandler(this); return 0; } LRESULT OnFileExit(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ PostMessage(WM_CLOSE); return 0; } LRESULT OnFileNew(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/) { // TODO: add code to initialize document return 0; } // Implementation of the other message handlers . . . }; CMainFrame derives from the following two WTL templates and two WTL classes. • CFrameWindowImpl – Boilerplate code for manages frames • CUpdateUI – Works in conjunction with an UPDATE_UI_MAP to perform data exchange between UI elements (e.g. toolbar & statusbar) and code • CMessageFilter – custom message filtering • CIdleHandler – custom idle handling CMainFrame has a macro called DECLARE_FRAME_WND_CLASS that identifies the resource Id to use to when accessing UI resources such as toolbar, accelerator, and menu descriptions. CMainFrame instantiates a class called CView, which is responsible for the client area. The CMainFrame::OnCreate function instantiates UI elements such as the toolbar, the statusbar and the menubar, and calls Create on the View class. It also gets the message loop for the current thread through a call to GetMessageLoop and appends a message filter and idle handler. There is a message map that maps menubar command Ids (e.g. ID_APP_EXIT & ID_FILE_NEW) to handler functions (e.g. OnFileExit & OnFileNew). Note that unlike MFC, where default implementation handlers for common commands are part of the library, with WTL these handlers are directly in the application’s code. The WTL AppWizard generated implementations of the handlers provide a full implementation where possible (e.g. OnFileExit posts a WM_CLOSE, which results in app shutdown) and otherwise the handlers contain a TODO comment (e.g. OnFileNew needs to be written by the application developer). Unlike MFC, WTL does not provide 124 WTL Developer’s Guide its own serialization code and binary file formats. Instead modern applications are strongly encouraged to base their serialization on XML, and it is easy enough to add your own XML code behind OnFileNew etc to call the XML parser. When OnFileExit posts a WM_CLOSE, neither the WTL templates nor the WTL AppWizard-generated application code detects this message, so by default it is passed to DefWindowProc, which responds by calling the Win32 API DestroyWindow, which posting a WM_DESTROY message. The application code does not detect this either, but the WTL template CFrameWindowImplBase does – by mapping it to a call to its member function OnDestroy. It in turn calls the Win32 API PostQuitMessage, which pops a WM_QUIT message onto the message queue. LRESULT OnDestroy(UINT, WPARAM, LPARAM, BOOL& bHandled){ if((GetStyle() & (WS_CHILD | WS_POPUP)) == 0) ::PostQuitMessage(1); bHandled = FALSE; return 1; } The WTL CMessageLoop::Run method has a for loop taking messages off the message queue and processing them. It calls the Win32 API GetMessage, which returns 0 if the message received is WM_QUIT. If this is the case, there is a break out of the loop. bRet = ::GetMessage(&m_msg, NULL, 0, 0); . . . else if(!bRet){ ATLTRACE2(atlTraceUI, 0, _T("CMessageLoop::Run - exiting\n")); break; // WM_QUIT, exit message loop } If you wish to customize shutdown, you could consider detecting the WM_CLOSE message and when finished calling DestroyWindow manually. The view class provides a straightforward ATL window that occupies the space from underneath the toolbar to above the statusbar. It has a message map that by default handles one message, WM_PAINT, which gets mapped to OnPaint. The implementation of OnPaint by default contains a TODO comment, but we changed that to call TextOut. class CHelloWorldWinView : public CWindowImpl{ public: DECLARE_WND_CLASS(NULL) BOOL PreTranslateMessage(MSG* pMsg){ pMsg; return FALSE; } BEGIN_MSG_MAP(CHelloWorldWinView) MESSAGE_HANDLER(WM_PAINT, OnPaint) END_MSG_MAP() LRESULT OnPaint(UINT /*uMsg*/, WPARAM /*wParam*/, 125 WTL Developer’s Guide LPARAM /*lParam*/, BOOL& /*bHandled*/) CPaintDC dc(m_hWnd); dc.TextOut(0,0, TEXT("HelloWorld")); return 0; { } }; Adding a Menu-item And a Toolbar Icon The resources generated by the WTL AppWizard include a menu and a toolbar, both identified by IDR_MAINFRAME. If we wish to add commands to menus or the toolbar, then we need to extend these resources in ResourceView, assign command ids to them, detect these command ids in the message map for the frame window and map them to handler functions. An example will make this clearer. Imagine we wished to add a menu item and toolbar icon which when pressed, would result in the Win32 Beep API being called. We start by adding a menu item to the resources, giving it an id of ID_FILE_BEEP. We then add a toolbar icon to the toolbar resource. We also give it also the id of ID_FILE_BEEP. To support tooltips, append “\nBeep” to the prompt. Both the menubar and the toolbar are going to be hosted by the mainframe class. Therefore we add this command handler to its message map: COMMAND_ID_HANDLER(ID_FILE_BEEP, OnFileBeep) And we add this implementation LRESULT OnFileBeep(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ Beep(200,200); return 0; } 126 WTL Developer’s Guide When we build and run the application we see the extra Beep menu item and Beep toolbar icon. When either is pressed, we should hear a beep from the PC’s speakers. When we hover the mouse over the beep toolbar icon we see its tooltip. Assuming you accept the defaults for step 2 in the WTL AppWizard, the generated application uses WTL’s command bars to display the contents of the menubar and toolbar within rebar controls. One neat trick of WTL command bars is that when rendering menu items that have the same command id as a toolbar icon, that icon is also rendered along with the menu-item name. This greatly increases awareness among endusers of what such icons mean – it is especially useful when very many icons are in use. MDI Application If we wish to edit multiple document types concurrently, the MDI is one approach. The top-level mainframe window contains a number of childframe windows, each of which can edit a document. Visual C++’s DevStudio itself is a classic example. To examine the MDI support within WTL, create a new project called HelloWorldMDI, and in the WTL AppWizard select the MDI Application option. As with the SDI Application sample, in the OnPaint method in the Cview class add this line: dc.TextOut(0,0, TEXT("HelloWorld")); Compile and run. There is another frame containing the menubar, toolbar, status bar, and in its client area there can be a multiple MDI child window. Each of these has a view and hence during each of their refreshes they output the string “HelloWorld”. 127 WTL Developer’s Guide The code produced is very similar to that of the SDI Application. All files are exactly the same, except for mainfrm.h. Also, there is an additional file called ChildFrm.h. The CMainFrame class is defined in mainfrm.h as: class CMainFrame : public CMDIFrameWindowImpl, public CUpdateUI, public CMessageFilter, public CidleHandler In the SDI sample it derives from CFrameWindowImpl, whereas in the MDI sample it derives from CMDIFrameWindowImpl, which WTL defines as: template class CMDIFrameWindowImpl; The message map handles three additional messages: COMMAND_ID_HANDLER(ID_WINDOW_CASCADE, OnWindowCascade) COMMAND_ID_HANDLER(ID_WINDOW_TILE_HORZ, OnWindowTile) COMMAND_ID_HANDLER(ID_WINDOW_ARRANGE, OnWindowArrangeIcons) Handler functions for the new messages merely pass the call onto default implementations inside CMDIFrameWindowImpl’s CMDIWindow. You can change these as needed. LRESULT OnWindowCascade(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/) { MDICascade(); return 0; } LRESULT OnWindowTile(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ MDITile(); return 0; } LRESULT OnWindowArrangeIcons(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ MDIIconArrange(); return 0; 128 WTL Developer’s Guide } OnFileNew is also changes to create a new ChildFrame. LRESULT OnFileNew(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/) { CChildFrame* pChild = new CChildFrame; pChild->CreateEx(m_hWndClient); // TODO: add code to initialize document return 0; } The ChildFrm.h file contains the definition of the new ChildFrame class. It is defined simply as: class CChildFrame : public CMDIChildWindowImpl { public: DECLARE_FRAME_WND_CLASS(NULL, IDR_MDICHILD) CHelloWorldMDIView m_view; virtual void OnFinalMessage(HWND /*hWnd*/){ delete this; } BEGIN_MSG_MAP(CChildFrame) MESSAGE_HANDLER(WM_CREATE, OnCreate) MESSAGE_HANDLER(WM_FORWARDMSG, OnForwardMsg) CHAIN_MSG_MAP(CMDIChildWindowImpl) END_MSG_MAP() LRESULT OnCreate(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& bHandled){ m_hWndClient = m_view.Create(m_hWnd, rcDefault, NULL, WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN, WS_EX_CLIENTEDGE); bHandled = FALSE; return 1; } LRESULT OnForwardMsg(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM lParam, BOOL& /*bHandled*/){ LPMSG pMsg = (LPMSG)lParam; if(CMDIChildWindowImpl::PreTranslateMessage(pMsg)) return TRUE; return m_view.PreTranslateMessage(pMsg); } }; Multiple Threads SDI The SDI Application is fine when we wish to edit a single document of a single document type at a time. This happens if the application only wishes to display a single 129 WTL Developer’s Guide SDI window, and within it edit a single document (Notepad is a good example). The MDI approach manages multiple child frames within a single top-level main frame. What the Multiple Threads SDI option in the WTL AppWizard provides is the capability of editing multiple documents in multiple top-level SDI windows (the Notepad-like WTL example MTPad shows this – Microsoft Word 2000 is also a good example). The end-user sees a separate top-level window for each open document, but only a single process is running. There should be a slight performance increase (as multiple threads in one process will work faster than multiple processes) and resources (memory/files/database connection, etc.) can be shared among all the threads. Now we wish to examine how this works. Create a new project called HelloWorldMT, and in the WTL AppWizard select the Multiple Threads SDI option. When we select this, the “Create as a COM Server” option is disabled. This is something we will investigate further later. Select the defaults for step 2 options. The code produced is quite similar to that from “SDI Application” option with 3 areas of difference. In both the resource file and the mainfrm.h file there are a small difference and there is a substantial difference in the .cpp file. The menubar in Multiple Threads SDI has an additional menu item, titled “New Window” which has a command id of ID_FILE_NEW_WINDOW. In mainfrm.h, the message map has a new entry: COMMAND_ID_HANDLER(ID_FILE_NEW_WINDOW, OnFileNewWindow) And an implementation of OnFileNewWindow is provided. LRESULT OnFileNewWindow(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ ::PostThreadMessage(_Module.m_dwMainThreadID, WM_USER, 0, 0L); return 0; } What this is doing is posting a private message, WM_USER, onto the message queue of the primary thread. (Therefore the developer should not try to use this particular message type for private purposes). We will need to see what happens to that message. The source file contain WinMain, .cpp, is quite different. The WinMain implementation has the call to the Run function removed, and it is replaced by: CHelloWorldMTThreadManager mgr; int nRet = mgr.Run(lpstrCmdLine, nCmdShow); The CThreadManager class is completely defined inside the .cpp file. The implementation of the Run function has been moved into this class and it includes multi-threading functionality. This class is used to manage a set of threads, each of which in turn manages one SDI window. This class defines one internal structure, and provides a static function that is the threadproc for the per-SDI thread, methods to add and remove threads and a Run method. 130 WTL Developer’s Guide class CHelloWorldMTThreadManager { public: // thread init param struct _RunData { LPTSTR lpstrCmdLine; int nCmdShow; }; The RunData structure stores the command line and the nCmdShow parameter. A new instance of this structure is created each time a new thread is needed, and it tells the thread which command line and which nCmdShow option to use. The threadproc parameter is a pointer, and we need to pass in a pointer and an int. Therefore we need this extra structure containing a pointer and an integer, and pass in a pointer to it. Our multithreading experts might say this information is read-only and it is only necessary that it exists once. After all, we are allocating either eight or twelve bytes (depending on whether we are using Win32 or Win64), which are both occupying space and time consuming, and it would better if we could avoid it. The way it is used here is that the first SDI window gets the command line and the CmdShow from WinMain and that subsequent threads are passed a NULL command line and SW_SHOWNORMAL. The wizard-generated code ignores the command line and uses nCmdShow parameter to determine the rendering of the mainframe. As your first thread will be receiving the command line and subsequent threads will receive NULL, you might consider adding code to determine if the command-line is non-NULL and if so to process it – how this is to be is obviously application-specific, but one typical command-line parameter could be a document filename to open. At this point our multithreading experts will argue (they are a persistent bunch) that surely we could have the structure for the first thread (which needs real data), and for subsequent thread pass in NULL as the threadproc parameter, and inside the threadproc if the parameter is NULL ignore the command line and use SW_SHOWNORMAL when rendering the mainframe. Yes, this will work, but it limits our future flexibility – what happens if we wish to do different things in different threads, then this defaulting will hamper us. In conclusion, the current implementation is fine, and don’t worry about a dozen bytes (1GB of RAM does not cost much!). If you are losing a night’s sleep over it, then use the NULL parameter idea. static DWORD WINAPI RunThread(LPVOID lpData){ CMessageLoop theLoop; _Module.AddMessageLoop(&theLoop); _RunData* pData = (_RunData*)lpData; CMainFrame wndFrame; if(wndFrame.CreateEx() == NULL){ ATLTRACE(_T("Frame window creation failed!\n")); return 0; } wndFrame.ShowWindow(pData->nCmdShow); 131 WTL Developer’s Guide ::SetForegroundWindow(wndFrame); delete pData; int nRet = theLoop.Run(); _Module.RemoveMessageLoop(); return nRet; // Win95 needs this } RunThread is the threadproc for each SDI thread. Note that it is static, which is a requirement for a C++ class member to be a threadproc. Apart from the code concerning _RunData, this is the same as we examine previously in the SDI Application sample. DWORD m_dwCount; HANDLE m_arrThreadHandles[MAXIMUM_WAIT_OBJECTS - 1]; CHelloWorldMTThreadManager() : m_dwCount(0){ } There are two data members, one to hold the count of and the other to hold the handles of the SDI threads. We need to keep track of these to determine when thread dies, and when the final one exits then we can shutdown the process. The count is initialized to 0. int Run(LPTSTR lpstrCmdLine, int nCmdShow){ MSG msg; // force message queue to be created ::PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE); AddThread(lpstrCmdLine, nCmdShow); int nRet = m_dwCount; DWORD dwRet; while(m_dwCount > 0){ dwRet = ::MsgWaitForMultipleObjects(m_dwCount, m_arrThreadHandles, FALSE, INFINITE, QS_ALLINPUT); if(dwRet == 0xFFFFFFFF) ::MessageBox(NULL, _T("ERROR: Wait for multiple objects failed!!!"), _T("HelloWorldMT"), MB_OK); else if(dwRet >= WAIT_OBJECT_0 && dwRet <= (WAIT_OBJECT_0 + m_dwCount - 1)) RemoveThread(dwRet - WAIT_OBJECT_0); else if(dwRet == (WAIT_OBJECT_0 + m_dwCount)){ ::GetMessage(&msg, NULL, 0, 0); if(msg.message == WM_USER) AddThread("", SW_SHOWNORMAL); else ::MessageBeep((UINT)-1); } else ::MessageBeep((UINT)-1); } return nRet; } }; The Run method is called from WinMain. It first calls the AddThread method to add the initial thread that processes the real command line and displays the first window. 132 WTL Developer’s Guide After this it initializes the nRet parameter to the value of m_dwCount, which has just been set to 1 inside AddThread. The last line in Run returns nRet as the return value of the method. In WinMain, the return value from Run is used as the return value for the entire process, which means it becomes the process’ exit code. Most of the time this is not important – if it is for your application you can change it as needed. The core of the Run method is a while loop, in which a blocking call is make to MsgWaitForMultipleObjects, which waits until a message arrives on the message queue for the upon which this API call is made or any kernel handle in the list becomes signaled. The kernel handles are the handles to the thread Ids, and they will become signaled when the thread exits. Threads get added when the user selects “New Window”, which results in a WM_USER message getting popped onto the primary thread’s message queue. Threads exit when the end-user closes the SDI window. In both cases it is the call to MsgWaitForMultipleObjects that detects these and responds accordingly by calling AddThread and RemoveThread as needed. When the number of SDI threads hits 0, then no windows are displayed and the application can shutdown. In the WaitForMultipleObjects API, MAXIMUM_WAIT_OBJECTS kernel objects can be waited upon simultaneously. MAXIMUM_WAIT_OBJECTS is set to 64 on Windows 2000 and the older Windows OS versions. With MsgWaitForMultipleObjects, the limit is MAXIMUM_WAIT_OBJECTS-1, or 63. What happened to the last one? The OS itself is adding an event handle to the list and thus taking up one slot. This event gets signaled when messages are detected on the message queue. The hard coded strings inside the call to MessageBox are noted – your internationalization engineers will need to change that to resource strings. Note there is a call to PeekMessage with the comment “force message queue to be created”. What is happening here is that the primary thread running this method has no window, and yet each SDI thread contains a menu item called “Add Window”, which when selected causes the SDI thread to pop a WM_USER message onto the message queue of the primary thread. As you will recall from our discussion of Win32 windowing, inside the Windows OS, the kernel maintains message queues for threads that might need them. Not every thread will be using windows (think of all that number-crunching code in the world which has no user interface), so it would be inefficient to automatically create these queues. Instead, what the kernel does is to create a message queue structure the first time calls are made to the windowing APIs on that thread. As the primary thread has no windows, up to this point the message queue does not exist. The call to PeekMessage here within the primary thread will result in the kernel creating it for this thread. The call to MsgWaitForMultipleObjects a few lines on will also cause the message queue for the primary thread to be created. So why is the call to PeekMessage needed? If we comment it out and build and run the application, we will notice it seems to run fine. Where problems can occur is when the secondary thread (created in the call to AddThread) starts sending messages to the 133 WTL Developer’s Guide primary thread before the primary thread has a message queue. This could occur if there is a context switch between the calls to AddThread and MsgWaitForMultipleObjects, and when the secondary thread starts running, the end-user selected the New Window menu item. It is unlikely to happen, but the PeekMessage call completely eliminates that possibility. To simulate the problem we could force a context switching by changing this code in the Run method . . .: ::PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE); AddThread(lpstrCmdLine, nCmdShow); . . . to these lines: // ::PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE); AddThread(lpstrCmdLine, nCmdShow); Sleep(20000); When we run this application, the primary thread runs, executes AddThread and sleeps. The secondary thread is launched from AddThread and causes the SDI window to be displayed – if the end-user then quickly selects “New Window” a few times, we see nothing immediately. When the sleep duration expires, the message queue is create, and all messages subsequent to that are processed. What happens to the messages posted onto the message queue from the SDI threads when the message queue did not exist? On Windows 2000, a solitary message is recorded – so if we press “Add Window” three times, then at the end of the sleep period only one new message is created. This is probably different on other versions of Windows. A word of warning - as with any multithreading coding, you have to be extremely careful with sequencing your calls to the OS. The code generated by WTL AppWizard as presented is correct. However, as it is now part of your application, you could quite easily, and probably will, start making changes to it. Again, this is fine – that is what you are paid to do if it is needed. This warning relates to the call to MsgWaitForMultipleObjects. This returns when new messages are received on the message queue for this thread. Messages that have been looked at by PeekMessage are not consider new. In the AppWizard generated code, the sequence of calls is PeekMessage, AddThread and MsgWaitForMultipleObjects. If you change their order to AddThread, PeekMessage and MsgWaitForMultipleObjects, and if there is a context switch between the first two of these, if could happen that the user clicks “Add New Window” in the SDI thread and a WM_USER message gets popped onto the primary thread’s message queue. Sometime later there is another context switch, and the primary thread resumes executing. It calls PeekMessage, which detects the WM_USER message on the queue and changes its state so it is not new. Then MsgWaitForMultipleObjects executes and it simply ignores the WM_USER message because it is not new. To clearly demonstrate the problem, change these two lines in the wizard-generated code: ::PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE); AddThread(lpstrCmdLine, nCmdShow); to these lines: AddThread(lpstrCmdLine, nCmdShow); Sleep(20000); // give ourselves plenty of time to “Add Window” 134 WTL Developer’s Guide ::PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE); When the application runs, select New Window from the File menu – you will note that the command is ignored by the application. In the original code, a new window would be displayed. To avoid the problem, you should either leave the code generated by the WTL AppWizard as is, remove the call to PeekMessage or change the call to MsgWaitForMultipleObjects to a call to MsgWaitForMultipleObjectsEx with the MWMO_INPUTAVAILABLE parameter. dwRet = ::MsgWaitForMultipleObjectsEx(m_dwCount, m_arrThreadHandles, INFINITE, QS_ALLINPUT, MWMO_INPUTAVAILABLE); It is unlikely that you will change the code in this way, but you have been warned! DWORD AddThread(LPTSTR lpstrCmdLine, int nCmdShow){ if(m_dwCount == (MAXIMUM_WAIT_OBJECTS - 1)){ ::MessageBox(NULL, _T("ERROR: Cannot create ANY MORE threads!!!"), _T("HelloWorldMT"), MB_OK); return 0; } _RunData* pData = new _RunData; pData->lpstrCmdLine = lpstrCmdLine; pData->nCmdShow = nCmdShow; DWORD dwThreadID; HANDLE hThread = ::CreateThread(NULL, 0, RunThread, pData, 0, &dwThreadID); if(hThread == NULL){ ::MessageBox(NULL, _T("ERROR: Cannot create thread!!!"), _T("HelloWorldMT"), MB_OK); return 0; } m_arrThreadHandles[m_dwCount] = hThread; m_dwCount++; return dwThreadID; } 135 WTL Developer’s Guide Multiple Threads SDI Architecture Windows on Screen Secondary Threads controlling SDI Window If user selects “New Window” from file menu, post a WM_USER message on primary thread’s message queue If user selects “Exit” from file menu, post a WM_CLOSE message on this thread’s message queue This message is not handled directly by WTL or application code, so is passed to DefWindowProc, which responds by posting a WM_DESTROY message - This message is handled inside WTL’s CFrameWindowImplBase::OnDestroy which posts a WM_QUIT message, which terminates the thread The primary thread has no window Each secondary thread controls one visible SDI window Primary Thread Calls WinMain, which calls CThreadMgr::Run CThreadMgr::Run calls AddThread – for first SDI Window Calls MsgWaitForMultipleObjects in a loop If WM_USER window message received, call AddThread to create a new SDI thread which in turn creates a new SDI window If a thread handle becomes signaled, call RemoveThread If no more thread handles are available, return, and then exit WinMain Array of Thread Handles Message Queue for Primary Thread Only ever contains WM_USER messages Messages are appended by seconday threads Messages are extracted in primary thread’s message loop, which calls AddThread Primary thread waits until handles to secondary threads become signaled (i.e. secondary thread has exited) To control this is needs to manage an array of handles to the secondary threads - handles are appended at the end of the array in a call to AddThread Handles are not actually removed from the array – rather they are overwritten. When the primary thread detects that a secondary thread has exited, it calls RemoveThread which copies the uppermost entry in the array to overwrite the value in the index occurred by the exited thread Primary thread itself exits when no secondary threads exist 136 WTL Developer’s Guide The AddThread method is responsible for creating a new SDI thread that in turn causes a new SDI window to be displayed. It just calls the Win32 CreateThread API and pops the thread handle at the end of the handle array and increments the m_dwCount variable that stores the number of entries. There is no need to switch the call to m_dwCount++ with a call to InterlockedIncrement(&m_dwCount) because this code will only every be used from the primary thread – it is never used from the SDI threads. void RemoveThread(DWORD dwIndex){ ::CloseHandle(m_arrThreadHandles[dwIndex]); if(dwIndex != (m_dwCount - 1)) m_arrThreadHandles[dwIndex] = m_arrThreadHandles[m_dwCount - 1]; m_dwCount--; } The RemoveThread method closes the handle of the thread that has just exited and removes its entry from the array of handles. Note that this array must be kept contiguous, so the technique used is to copy the top-most entry to the index of the thread to remove. Modeless Dialog-Based Application The last application type to discuss is a modeless dialog-based application. The output from the WTL AppWizard for this is a mixture of the modal dialog and the SDI Application approaches. In the Modeless Dialog code, CMainDlg is derived from CDialogImpl, just like the Modal Dialog sample, but also from CUpdateUI, CMessageFilter and CIdleHandler, as in the SDI sample. class CMainDlg:public CDialogImpl, CUpdateUI,public CMessageFilter, public CIdleHandler There is a empty update UI map. The closing of the dialog in the modal dialog sample was through a call to ATL’s CDialogImpl::EndDialog, which in turn called the Win32 API, EndDialog. In the modeless dialog sample, the closing of the dialog results in the following generated code being called: void CloseDialog(int nVal){ DestroyWindow(); ::PostQuitMessage(nVal); } There is also a call to CUpdateUIBase::UIAddChildWindowContainer added to OnInitDialog for updating of UI elements (CUpdateUIBase is the parent of CUpdateUI). In the modal dialog sample, the .cpp file contained the WinMain implementation, which handles the message loop in a call to CDialogImpl::DoModal, which in turn calls the Win32 API DialogBoxParam. 137 WTL Developer’s Guide In the modeless dialog sample, the .cpp file is exactly as with the SDI Application. It contains the WinMain and does the message loop through an instantiation of WTL’s CMessageLoop. Application Features - ActiveX Control Hosting Now we turn our attention to the WTL AppWizard “Application Features” in step 1, “ActiveX Control Hosting” and “Act as a COM Server”. When “ActiveX Control Hosting” is selected, two changes are made to the code. Stdafx.h has some extra headers included. Instead of this line at the bottom of the file: #include Now there are these lines #include #include #include #include Also, WinMain is changed to initialize COM and a call to AtlAxInit is added. AtlAxInit registers the "AtlAxWin" window, which is used for ActiveX control hosting. int WINAPI _tWinMain(HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPTSTR lpstrCmdLine, int nCmdShow){ HRESULT hRes = ::CoInitialize(NULL); // initialize common controls . . . hRes = _Module.Init(NULL, hInstance); ATLASSERT(SUCCEEDED(hRes)); AtlAxWinInit(); int nRet = Run(lpstrCmdLine, nCmdShow); _Module.Term(); ::CoUninitialize(); return nRet; } It is important to note that selecting “ActiveX Control Hosting” is only adding header file support, registering the ATLAxWin window class and initializing COM for the primary thread, and that is all it does. It does not itself actually add any ActiveX control. This can be done later as needed. “ActiveX Control Hosting” and Modal Dialog Applications A special case is when the WTL AppWizard is asked to generate a modal or modeless dialog application and ActiveX Control Hosting is selected. The generated dialog code derives from CAxDialogImpl instead of CDialogImpl. “ActiveX Control Hosting” and SDI/MDI apps with Form view type The second step within the WTL AppWizard allows you to select the basis for the view, which we will discuss shortly. Normally it is set to a (blank) generic window. It can 138 WTL Developer’s Guide also be based on listview, treeview, etc. One additional option is to base it on a dialog template (a form). When the view is based on a form, the generated view class derives from CDialogImpl, regardless of the setting for “ActiveX Control Hosting” in step 1. However, when “ActiveX Control Hosting” is set, then an additional checkbox in set 2, Host ActiveX Controls, is enabled, and when set the form dialog derives from CAxDialogImpl and hence can host ActiveX controls. Application Features - Act as a COM Server All the samples produced up to now have concentrated exclusively on windowing. The ATL AppWizard can produce code that supports COM component creation. What happens if we wish to support both windowing and COM components within the one project? There is where the “Act As a COM Server” feature comes in. When this feature is selected a number of changes are made to the project settings and the generated code. A custom build step is added to register the components in the project (if any). One point of interest here is that the rgs file for each component has the LocalServer entry set to “LocalServer32 = s '%MODULE%'” and we will need to return to this issue when examining how startup happens via Automation. In stdafx.h, there is an external definition of _Module instantiating CServerAppModule and not CAppModule as in the previous sample. Note that CServerAppModule itself derives from CAppModule and adds methods very similar to the ATL AppWizard generated code when you select the “Executable” option. It has familiar methods such as StartMonitor, MonitorShutdown, FindOneOf and new methods such as ParseCommandLine and Register/UnregisterAppId. Also in stdafx.h, the header file atlcom.h is included, thus providing the application code with access to ATL’s COM definitions. Stdafx.cpp includes these additional lines, to do with registry management code: 139 WTL Developer’s Guide #ifdef _ATL_STATIC_REGISTRY #include #include #endif An IDL file called .idl is generated and it contains a default type library definition (just as would be generated had we run the ATL AppWizard). The main changes are in .cpp. Essentially what it needs to do is merge the code from the WTL AppWizard and the ATL AppWizard auto-generated generated .cpp files. It starts by adding includes for the GUIDs and interfaces: #include "initguid.h" #include "HelloWorldWithCOMServer.h" #include "HelloWorldWithCOMServer_i.c" It instantiates a variable called _Module of type CServerAppModule. Note it MUST be called _Module as this name is used in various ATL macros. CServerAppModule _Module; It has an empty object map. Later, the ATL ObjectWizard may be used to populate this map, as in standard ATL programming. BEGIN_OBJECT_MAP(ObjectMap) END_OBJECT_MAP() The Message Loop The Run function is the similar to that of the SDI Application sample: It creates a CMessageLoop, instantiates CMainFrame and calls CMessageLoop::Run. The one change is the addition of a call to _Module.Lock(). This increments the lock count with the ATL COM code uses to decide whether the COM server can be shutdown. In essence, what is happening is that the code is pretending that the CMainFrame is actually a COM component, and while displayed, the process running this EXE should not shut down. We will have to track how shutdown actually happens later. int Run(LPTSTR /*lpstrCmdLine*/ = NULL, int nCmdShow = SW_SHOWDEFAULT){ CMessageLoop theLoop; _Module.AddMessageLoop(&theLoop); CMainFrame wndMain; if(wndMain.CreateEx() == NULL){ ATLTRACE(_T("Main window creation failed!\n")); return 0; } _Module.Lock(); wndMain.ShowWindow(nCmdShow); int nRet = theLoop.Run(); _Module.RemoveMessageLoop(); return nRet; } 140 WTL Developer’s Guide The WinMain code has called to CoInitialize/_Module.Init() and _Module.Term()/CoUninitialize at the beginning and end. It initializes the common controls. The command line is parsed for UnregServer and RegServer, and if detected the appropriate calls in _Module are made and bRun set to false. The command line is also checked for “Automation” and if there a bAutomation flag is set. // check for UnregServer and RegServer . . . if(lstrcmpi(lpszToken, _T("Automation")) == 0){ bAutomation = true; break; } The core of WinMain is as follows: if(bRun){ _Module.StartMonitor(); hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER, REGCLS_MULTIPLEUSE | REGCLS_SUSPENDED); ATLASSERT(SUCCEEDED(hRes)); hRes = ::CoResumeClassObjects(); ATLASSERT(SUCCEEDED(hRes)); if(bAutomation){ CMessageLoop theLoop; nRet = theLoop.Run(); } else { nRet = Run(lpstrCmdLine, nCmdShow); } _Module.RevokeClassObjects(); ::Sleep(_Module.m_dwPause); } The Automation Command-line Parameter Note that when the Automation flag is passed in on the command line then no application window is displayed. This is keeping with the convention of application starting through Automation not displaying any UI unless specifically asked to via an Automation API. However, there is a problem here. When the COM run-time is asked to launch a server through a client’s call to CoCreateInstance, the COM run-time retrieves the “LocalServer” value from the registry, appends the string “-Embedding” to create a command-line, and calls CreateProcess to launch a new process using this command line. As noted earlier, the WTL AppWizard (and the ATL AppWizard), generate the LocalServer string in the component’s rgs file to be just %MODULE%, which gets expanded to the installed path to the EXE. The COM libraries will append “-Embedding” to this, but no one is appending “/Automation” and therefore by default the WTL-generated application, even when started via CoCreateInstance in a client, never even sees “/Automation” on the command-line. This is not a problem with ATL projects, because they are expected to store COM components and are designed to be always started this way. It is a problem with WTL projects, because they sometimes will be started directly by users - in which case they will immediately 141 WTL Developer’s Guide display the UI, and sometimes via COM by client applications - in which case by default they do not wish to display the UI, and will need to provide a component with an Automation method for displaying the UI. This is in keeping with the convention of Automation, which allows a client to decide whether the server’s UI should be displayed. Note that the application developer needs to write a component to display the UI – the WTL AppWizard generated code does not provide this. Note that whether ‘-‘ or ‘/’ is used to start a new command-line parameter is irrelevant. Most applications (including all ATL and WTL ones) search for both. The “Embedding” command-line parameter is used to signify that the application should work as an ActiveX Document Server and is being started by an ActiveX Document Container. The “/Automation” command-line parameter is used to signify that the application has been launched by a client wishing programmatic access to the Automation components inside the applications. What effect those command-line parameters have on an application are of course entirely an internal matter for the application. ATL applications just ignore them, as they always assume there are run as components. WTL applications wish to display the user interface when run directly by the end-user, and not when programmatically started via Automation. The solution is easy. Firstly one could check for Embedding rather than Automation, by changing the line: else if(lstrcmpi(lpszToken, _T("Automation")) == 0) for the line: else if(lstrcmpi(lpszToken, _T("Embedding")) == 0) Alternatively, one could add “/Automation” to the LocalServer entry in the .rgs file. With a DLL, there is no “command-line”, and therefore for component servers housed inside DLLs, the InprocServer32 or InprocServer64 entry merely states the pathname to the DLL. With EXEs, there are command-lines, and when a process is started by CoCreateInstance it takes the FULL string in the LocalServer32 or LocalServer64 field, appends “–Embedding” to it, and calls the Win32/64 API CreateProcess and passes the command-line string as the second parameter. Specifically, LocalServer need not be just a path, it can be a path and command-line parameter, such as “/Automation”. So, a solution to the problem would be to change the .rgs file as follows: LocalServer32 = s '%MODULE% /Automation’ However, this introduces another problem. The FindOneOf function is the WTL AppWizard generated code only checks for one parameter, and returns a pointer to the beginning of that parameter (first character after the ‘/’ or’-‘) until the end of the entire command-line. (It does not copy any data). As “-Embedding” is automatically appended to the command-line, what FindOneOf returns is actually “Automation –Embedding” and therefore the line: else if(lstrcmpi(lpszToken, _T("Automation")) == 0) fails to find a match. One solution would be to change this to: else if(lstrcmpi(lpszToken, _T("Automation -Embedding")) == 0) and another (probably better) solution would be to change the FindOneOf function to change the next whitespace to a NULL character. 142 WTL Developer’s Guide One final point before we leave the issue of the command line is what happens if an application only wishes to expose components with custom interfaces. There are no conventions regarding how the command-line should be set for this scenario, as normally the programmatic exposure of desktop applications is done purely via Automation. Think of the likes of Word 2000, which can be programming from virtually every Windows programming environment, precisely because it exposes an Automation API. However, if you really do need to have custom interfaces, and want to avoid the use of the “/Automation” command-line option, then you are free to pick you own, or simply rely on “–Embedding”. Application Termination The next item to consider is how the application (whether started via Automation or not!) gets shut down. With normal ATL projects, the _Module instance maintains a variable called m_nLockCnt, which is incremented when components are instantiated and decremented when the instances terminates. When it hits zero, a win32 event is set, so that the main application code can shutdown. With WTL applications, we might have one or more instances of components running, or windows only running, or a mixture of both. When should we shut down? The answer is when all windows and all component instances are terminated. When the application was started by the end-user, it could be that while running a client application programmatically created instances of components within the process, and so even when the end-user shuts down the last window (and thinks the entire application has disappeared), we should still keep the process running until the client released the component instance(s). When the application was started first by a client application via Automation, it could happen that a UI of that component is visible to end-users, and they might use a command such as “New Window” to edit another document within the same process. Alternatively, it could be that they try to launch the application again, and through certain tweaking with some code, you have it using the existing process if available. Later, when the programmatic client releases all its components, the process should not shutdown until the end-user windows are also shut directly by the end-user. To manage all this WTL uses the m_nLockCnt count for both top-level windows AND component instances, and when the value hits zero this means the process can exit. _Module has methods Lock and Unlock to increment and decrement this value, and GetLockCount to returns its current value. As components are instances and terminate the normal ATL code with increment and decrement m_nLockCnt as needed. What about WTL’s windows? We saw in the implementation of the Run function in .cpp that there is call to Lock when we are displaying a window: _Module.Lock(); When a window is closed (e.g. by selecting Exit from the File menu), then the WM_DESTROY message is detected and mapped to a call to a new method OnDestroy in the frame window class in mainfrm.h. 143 WTL Developer’s Guide LRESULT OnDestroy(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/){ // if UI is the last thread, no need to wait if(_Module.GetLockCount() == 1){ _Module.m_dwTimeOut = 0L; _Module.m_dwPause = 0L; } _Module.Unlock(); return 0; } This decrements the count. Note also it sets pauses to zero- the pauses are there for component code so that a process with no component instances will run for a short time longer, just in case a request comes in from a client to create a new instance – it is much faster doing so in an existing process, as opposed to launching a completely new process. There is a slight bug in the WTL method, CServerAppModule::Term. It calls the Win32 function CloseHandle for m_hEventShutdown. However, this has already been closed in the MonitorShutdown function, and therefore the repeated call in CServerAppModule::Term results in an error - a "First chance exception Invalid Handle" exception is thrown. Void term(){ if (m_hEventShutdown != NULL) CloseHandle(m_hEventShutdown); CAppModule::Term(); } The solution is to either remove the CloseHandle call in CServerAppModule::Term or alternatively in MonitorShutdown, after calling CloseHandle, then set m_hEventShutdown to NULL. Sample Using COM Server To create a sample using the COM Server support in WTL, create a new project called HelloWorldSDI_WithCOMServer, and in the AppWizard in Step 1 select “SDI Application” and “Act As A COM Server”, and select the defaults in step 2. Now from DevStudio’s Insert menu run the ATL Object Wizard, and insert a component called CConversation – with a dual-interface called IConversation (select “dual” in the Attributes property page). In DevStudio’s ClassView, select IConversation, right click and select “Add A Method”, and in the displayed dialog add a method to IConversation called Chat. There is no need for parameters. Now in ClassView select CConversation, and directly beneath it, select IConversation and the Chat method – select it and the Conversation.cpp file should be displayed, showing a default implementation of CConversation::Beep (note: do not select the top-level IConversation, which refers to the IDL entry). Add this line to CConversation::Beep: ::Beep(200,200); Fix the problem regarding the command-line Automation parameter, as discussed earlier. Then build. 144 WTL Developer’s Guide Issues regarding registration of Universal Marshaler Our WTL COM Server exposes an Automation Interface. So we need to build a simple WTL dialog-based client that calls the IConversation Interface. Run the WTL AppWizard again and in the dialog box add a push button. #import the typelibrary which is embedded in the server’s EXE. #import "..\debug\HelloWorldSDI_WithComServer.exe" rename_namespace("HelloLib") using namespace HelloLib; An aside here: Some developer dislike hard-coding a pathname into a source file. It can also be an annoyance if you are building the release version and first you have to build the debug version. You can avoid this by placing the typelibrary in a file on the OS search path or extending DevStudio’s Options menu/Directories list. Alternatively, you could consider that its use as here is merely for test purposes, and that you might only be building the debug version of the test project. (Note that the type library that is produced is the same regardless whether it is the debug or release builds). Add a command handler to be called when the pushbutton is selected. LRESULT OnChat(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ IConversationPtr m_ConversationPtr; HRESULT hr = m_ConversationPtr.CreateInstance(__uuidof(Conversation)); if (FAILED(hr)) _com_issue_error(hr); if (m_ConversationPtr){ m_ConversationPtr->Chat(); } m_ConversationPtr.Release(); m_ConversationPtr = NULL; return 0; } When we run our client app we note that unfortunately when it runs it reports an error when trying to connect to the WTL-generated application. If we try creating the client in VB, we would start by displaying the VB References dialog, and trying to select our typelib from the list, which is generated from the registry’s typelibrary entries. However, the typelibrary for our project does not appear in this list. Even when we add it manually using the browse button, when we later code up a call to the interface, it fails. Private Sub Command1_Click() Dim SH As New Conversation SH.Chat End Sub What is going on here? What are the clients not working? The debugger will tell us the problem is when instantiating the server. Let’s work our way back from the client to the server. The client and the server are in separate processes, so we need marshaling between them. We are using Automation, so we want to use type-library marshaling. 145 WTL Developer’s Guide For this to happen, two important sets of registry entries are needed. Under the registry HKCR\Interfaces key, there should be an entry for the IID of the IConversation interface, and it should have a sub-key of ProxyStubClsid32. As we are using Automation and the OS provides use with a suitable proxy stub DLL known as the Universal Marshaler (with a CLSID of 00020424-0000-0000-C000-000000000046), its value should be used. Also, the typelib id should be registered, as this is needed for type-library marshaling. However, when we examine the registry neither of these sets of keys exist. The reason the VB References dialog did not show the typelibrary for HelloWorldSDI_WithCOMServer was because its typelibrary was not registered. Now the ATL experts will argue here that surely this cannot be. When we generate ATL out-of-proc servers based on Automation, these registry entries are correctly made. Yes, that is correct with ATL – and surely WTL is tied in closely with ATL? Hmmm… Let us first examine what happens with an ATL EXE when “/register” is passed in on the command-line. For executables, the ATL AppWizard generates code will call FindOneOf and then has this code when the RegServer command-line parameter is found: if (lstrcmpi(lpszToken, _T("RegServer"))==0) { _Module.UpdateRegistryFromResource( IDR_SimpleExeProject, TRUE); nRet = _Module.RegisterServer(TRUE); bRun = FALSE; break; } Note it calls the CComModule::RegisterServer method, with a TRUE parameter. The result of this is that the typelibrary and all interfaces are registered as needed, and clients can attach without further ado. Now, when we examine the same code in the WTL AppWizard generated code, there is a slight difference. The RegServer handling code calls: nRet = _Module.RegisterServer(); When we examine CComModule::RegisterServer we see it defaults the first parameter, bRegTypeLib, to FALSE. HRESULT RegisterServer(BOOL bRegTypeLib = FALSE, const CLSID* pCLSID = NULL) 146 WTL Developer’s Guide Registry Layout for Local Servers supporting Automation HKCR = Entries you must make, other entries already exist Hard-disk CLSID {CLSID of Conversation} LocalServer32={path to Conversation EXE} {CLSID of Universal Marshaler} InprocServer32={path to oleaut32.dll} Typelibrary typelib id of Conversation Typelibrary 1.0 0 Win32={pathname to typelibrary embedded inside Conversation’s EXE} Interfaces IID of IConversation interface ProxyStubClsid32 = CLSID of Universal Marshaler TypeLib = typelib id of Conversation Typelibrary 147 WTL Developer’s Guide So now we see the difference between the ATL- and WTL-AppWizard generated code. What does the bRegTypeLib flag control? When set to true it results in a call to ::RegisterTypeLib, which registers the interfaces and typelib entries necessary for Automation. When set to false, which is the case for WTL, these entries are not made. Note that regardless of the bRegTypeLib, the CLSID entries are made due to the entries in the .rgs file. Assuming you wish to make the Interfaces and Typelib entries, all you have to do is change the call to RegServer in your WTL-AppWizard generated code as follows: nRet = _Module.RegisterServer(TRUE); Step 2 of the WTL AppWizard Now we will move on to examine step 2 of the WTL AppWizard. Note that if Dialog Based is selected in step (with or without the Modal Dialog checkbox selected), all the options in step 2 are disabled. The following discussion assumes you have selected a project type other than dialog-based. The default settings for Step 2 are as follows: The available options can be subdivided into a number of sections. Firstly, we can select whether we wish to have a toolbar, rebar, command bar or status bar in place. Status Bar The status bar checkbox can be selected independently of the others. If selected, it causes a number of additions to the generated application. A call to CreateSimpleStatusBar (with no parameter) is inserted in the Mainframe’s Create method. Your Mainframe class is derived from WTL’s CFrameWindowImpl, which in turn is derived from CFrameWindowImplBase. 148 WTL Developer’s Guide CreateSimpleStatusBar is implemented in CFrameWindowImplBase. When called with no parameters it loads a string resource identified by ATL_IDS_IDLEMESSAGE (which is set to “Ready” in the generated resources) and calls an overloaded method, which in turn calls Win32’s CreateStatusWindow, which create a status window as a child of the mainframe window, and records the status bar’s HWND in a data member called m_hWndStatusBar. BOOL CreateSimpleStatusBar(UINT nTextID = ATL_IDS_IDLEMESSAGE, DWORD dwStyle = WS_CHILD | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS|SBARS_SIZEGRIP, UINT nID = ATL_IDW_STATUS_BAR){ TCHAR szText[128];// max text lentgth is 127 for status bars szText[0] = 0; ::LoadString(_Module.GetResourceInstance(),nTextID,szText,128); return CreateSimpleStatusBar(szText, dwStyle, nID); } BOOL CreateSimpleStatusBar(LPCTSTR lpstrText, DWORD dwStyle = WS_CHILD | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS | SBARS_SIZEGRIP, UINT nID = ATL_IDW_STATUS_BAR) { ATLASSERT(!::IsWindow(m_hWndStatusBar)); m_hWndStatusBar = ::CreateStatusWindow(dwStyle, lpstrText, m_hWnd, nID); return (m_hWndStatusBar != NULL); } A menu item titled Status Bar is added to the View menu with these properties: The mainframe message map is extended with this entry: COMMAND_ID_HANDLER(ID_VIEW_STATUS_BAR, OnViewStatusBar) The OnViewStatusBar handler function is added. LRESULT OnViewStatusBar(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ BOOL bNew = !::IsWindowVisible(m_hWndStatusBar); ::ShowWindow(m_hWndStatusBar, bNew ? SW_SHOWNOACTIVATE : SW_HIDE); UISetCheck(ID_VIEW_STATUS_BAR, bNew); UpdateLayout(); return 0; } This line is also added to the CMainframe::Create method. UISetCheck(ID_VIEW_STATUS_BAR, 1); The menu item has a check box beside it, and when selected, the status bar should be displayed. This line is initializing the checkbox to be set, as the status bar is initially displayed. 149 WTL Developer’s Guide Finally it add this entry to the UPDATE_UI_MAP for the mainframe: UPDATE_ELEMENT(ID_VIEW_STATUS_BAR, UPDUI_MENUPOPUP) The UPDATE_UI_MAP is used by WTL to keep track of what it needs to update dynamically. Here it updates what is displayed in the status bar, but we can also use it to update the rendering of toolbar icons and menu items. WTL also offers a sophisticated status bar class called CMultiPaneStatusBarCtrl. It supports a status bar that has sub-divisions. The WTL AppWizard does not provide extra support for its use, but it can easily be added manually. We will examine it is a later chapter. Command Bars, Rebars and Toolbars If the toolbar checkbox is un-selected, then the Rebar and Command Bar checkboxes are automatically disabled (hence to use a rebar or command bar you must have a toolbar). If the Rebar checkbox is un-selected, then the Command Bar checkbox is automatically disabled (hence to use a command bar you must have a rebar). Think of a command bar as a super menubar that displays menu item text and alongside icons that share the same command id. It lives within a band inside a rebar control. The command bar takes those icons from a toolbar (the toolbar itself is also displayed in another band in the rebar). A command bar is not a Win32 control – it is something specific to WTL and involves quite an amount of code. If you don’t have a rebar, then it does not make sense to have a command-bar. If you don’t have a toolbar, then there would be no icons for a command bar to display, and if you only have a single menubar then it does not make sense to have rebar or command bars. If the Command Bar checkbox is selected, then the following occurs. The CMainFrame class has this extra data member: CCommandBarCtrl m_CmdBar; These lines are added to CMainFrame::OnCreate: // create command bar window HWND hWndCmdBar = m_CmdBar.Create(m_hWnd, rcDefault, NULL, ATL_SIMPLE_CMDBAR_PANE_STYLE); // attach menu m_CmdBar.AttachMenu(GetMenu()); // load command bar images m_CmdBar.LoadImages(IDR_MAINFRAME); // remove old menu SetMenu(NULL); . . . CreateSimpleReBar(ATL_SIMPLE_REBAR_NOBORDER_STYLE); AddSimpleReBarBand(hWndCmdBar); When the mainframe window is created it automatically loads up a menubar with the IDR_MAINFRAME menubar resource. What is happening in this command bar calling code is that it calls Win32’s GetMenu API to get a HMENU of the loaded menubar. 150 WTL Developer’s Guide Within the call to the command bar’s AppendMenu, the contents of the existing menubar are copied to the command bar, and toolbar icons with the same command IDs are also added. When that is finished, we now have two copies of the menubar – one that was loaded by default and is the traditional menubar for the window, and the other which is in the commandbar. We only need one – hence the call to SetMenu(NULL), which eliminates the existing menubar. The call to AddSimpleReBarBand adds the command-bar as a band to the rebar control. (To prove that two menubars do exist for a short time, simply comment out the SetMenu(NULL) call – when you run the application you will see the old menubar and the new commandbar). Note that the call to CreateSimpleReBar when command bars are supported has the parameter ATL_SIMPLE_REBAR_NOBORDER_STYLE. If the command bar checkbox is not selected in the WTL AppWizard, then the call to CreateSimpleReBar is passed no parameter, and it defaults to ATL_SIMPLE_REBAR_STYLE. These two defines are: #define ATL_SIMPLE_REBAR_STYLE \ (WS_CHILD | WS_VISIBLE | WS_BORDER | WS_CLIPCHILDREN | WS_CLIPSIBLINGS | RBS_VARHEIGHT | RBS_BANDBORDERS | RBS_AUTOSIZE) #define ATL_SIMPLE_REBAR_NOBORDER_STYLE \ (WS_CHILD | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS | RBS_VARHEIGHT | RBS_BANDBORDERS | RBS_AUTOSIZE | CCS_NODIVIDER) The difference between them is one has WS_BORDER and the other has CCS_NODIVIDER. The following three screen dumps shows an app with rebar and command-bar options selected (left), with the rebar option selected and command-bar option not selected (center) and with neither the rebar nor the command bar options selected (right). When the command-bar is selected, then the rebar shows band dividers (those verticalline grip marks at the front of each band) and shows no border. It is noted that the command-bar is a couple of pixels taller than the menubar, and that when only the rebar is selected that a border is displayed (beneath the toolbar). When neither the rebar nor the command bar options are selected, then a normal toolbar is displayed. Note that its icons have a border around each of them. It is recommended for most applications to choose both the rebar and command bar options (this is the WTL AppWizard defaults settings). The extra command bar functionality is powerful and involves no extra work on the part of the application developer. End users will appreciate the extra functionality the command-bar provides over traditional menubars. However, command bars require the rebar control, and this is available in version 4.70 and later of Comctl32.dll, which is supported natively in Windows 2000/Windows 98 or later, but for Windows 95 and Windows NT 4 then Internet Explorer 3 or later needs to be installed. The vast majority (but not all) end- 151 WTL Developer’s Guide users with these older OSes also have IE3 installed. If your application must work on machines which do not IE3 installed, or if you wish to completely emulate the user interface of an old application, then you should un-select rebar and command bar support. The in-between choice, of selecting rebar but un-selecting command bar support, does not seem to be sensible in most situations – if you have rebars you might as well have command bars, which most end-users will appreciate – this choice would probably mean that the application developer intends to manually change some of the generated code for his/her specific purposes. Now will explore what additions are made to the generate source code when these options are selected. If you select toolbar support (by default it is selected), then the following additions are made. The OnIdle method has a call to UIUpdateToolBar added. The menubar resource has a toolbar menu item added beneath the View menu. An entry is added to the CMainFrame’s message map: COMMAND_ID_HANDLER(ID_VIEW_TOOLBAR, OnViewToolBar) And a command handler is supplied: LRESULT OnViewToolBar(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/) { BOOL bNew = !::IsWindowVisible(m_hWndToolBar); ::ShowWindow(m_hWndToolBar, bNew ? SW_SHOWNOACTIVATE : SW_HIDE); UISetCheck(ID_VIEW_TOOLBAR, bNew); UpdateLayout(); return 0; } This line is also added to the CMainframe::Create method. UISetCheck(ID_VIEW_TOOLBAR, 1); The menu item has a check box beside it, and when selected, the status bar should be displayed. This line is initializing the checkbox to be set, as the toolbar in initially displayed. Finally it add this entry to the UPDATE_UI_MAP for the mainframe: UPDATE_ELEMENT(ID_VIEW_TOOLBAR, UPDUI_MENUPOPUP) 152 WTL Developer’s Guide If toolbar support is selected, but rebar support is not, then the toolbar creation code inside Mainframe’s Create method is simply a call to: CreateSimpleToolBar (); If toolbar and rebar support is selected, but command bar support is not, then the code is: HWND hWndToolBar = CreateSimpleToolBarCtrl(m_hWnd, IDR_MAINFRAME, FALSE, ATL_SIMPLE_TOOLBAR_PANE_STYLE); CreateSimpleReBar(); AddSimpleReBarBand(hWndToolBar); If toolbar, rebar support and command-bar support are selected, then the code is: // create command bar window HWND hWndCmdBar = m_CmdBar.Create(m_hWnd, rcDefault, NULL, ATL_SIMPLE_CMDBAR_PANE_STYLE); // attach menu m_CmdBar.AttachMenu(GetMenu()); // load command bar images m_CmdBar.LoadImages(IDR_MAINFRAME); // remove old menu SetMenu(NULL); HWND hWndToolBar = CreateSimpleToolBarCtrl(m_hWnd, IDR_MAINFRAME, FALSE, ATL_SIMPLE_TOOLBAR_PANE_STYLE); CreateSimpleReBar(ATL_SIMPLE_REBAR_NOBORDER_STYLE); AddSimpleReBarBand(hWndCmdBar); AddSimpleReBarBand(hWndToolBar, NULL, TRUE); Use a View Window Step 2 of the WTL AppWizard contains a checkbox with the title Use a view window. By default this is selected. When selected, the generated code contains a view class, and the mainframe class has a data member which is an instance of the view class, and in CMainframe::Create a call is made to the view create member function. The view is created as a child window of the mainframe window, with the WS_EX_CLIENTEDGE. The view occurs the client area in the mainframe – covering all the space beneath the toolbar and above the statusbar. The framewindow has a m_hWndClient data member, and this records the HWND of the view. When the framewindow is resized, WTL detects the WM_SIZE and resizes the view window as appropriately. When the Use a view window checkbox is un-selected, the view type listbox in the AppWizard is disabled. Applications generated with or without a view look almost identical. The background of the view and the background of the frame’s client area are displayed in the default background color (usually white). The only distinction is that with the view, one can see the edge generated by the WS_EX_CLIENTEDGE attribute, which is absent when the view is not there. 153 WTL Developer’s Guide View Type The view type combo-box specifies how the view should be created. It may be set to one of the following: • Generic window (blank window – the default) • Form (Dialog based) • Listbox • Edit • Listview • Treeview • Rich-Edit • HTML Page Using the WTL AppWizard, you may not create a view based on a control not in this list and you may not create multiple views (but we will later examine how to do both of these manually). The mainframe class generated by the WTL AppWizard is not changed in any way when different view types are selected (with the exception of HTML Page). When generic window view type is selected, then the view code is generated as follows: class CWithEverythingView : public CWindowImpl{ public: DECLARE_WND_CLASS(NULL) BOOL PreTranslateMessage(MSG* pMsg){ pMsg; return FALSE; } BEGIN_MSG_MAP(CWithEverythingView) MESSAGE_HANDLER(WM_PAINT, OnPaint) END_MSG_MAP() LRESULT OnPaint(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/) { CPaintDC dc(m_hWnd); //TODO: Add your drawing code here return 0; } }; When a view based on a control is selected, then the code generated is: class CViewBasedOnListViewView : public CWindowImpl{ public: DECLARE_WND_SUPERCLASS(NULL, CListViewCtrl::GetWndClassName()) BOOL PreTranslateMessage(MSG* pMsg){ 154 WTL Developer’s Guide pMsg; return FALSE; } BEGIN_MSG_MAP(CViewBasedOnListViewView) END_MSG_MAP() }; No WM_PAINT handler is required, as the control will draw itself when needed. The view window derives from CWindowImpl based on an appropriate control class. The DECLARE_WND_SUPERCLASS macro is passed the control’s name. The listview, treeview, edit, listbox and rich-edit all work this way. If you wished to create a view based on a different control, you could simply make the appropriate change to one of the support control view types. The list that is supported contains the controls that are most frequently used as the basis of a view. Other controls might not be that suitable. For example, a pushbutton should not be the basis for a view. Note that the version of rich edit used depends on the _RICHEDIT_VER macro setting, which is defaulted to 0x0100 in stdafx.h. Also note that the WTL distribution contains a good examine of RichEdit called MTPad. When the view type is selected as Form (dialog based), then a dialog template is generated in the ResourceView with an id of IDD__FORM and the view class is generated as follows: class CViewBasedOnFormView : public CDialogImpl{ public: enum { IDD = IDD_VIEWBASEDONFORM_FORM }; BOOL PreTranslateMessage(MSG* pMsg){ return IsDialogMessage(pMsg); } BEGIN_MSG_MAP(CViewBasedOnFormView) END_MSG_MAP() }; Note that the view is based on CDialogImpl and not CAxDialogImpl. The difference between selecting an SDI/MDI app with its view type set to form, as here, and selecting a dialog box (in step 1 of the WTL AppWizard), is that the SDI/MDI app has a mainframe with command bars and statusbar. When the view type is set to HTML Page, then the view will be based on the web browser. The view class will be generated as: class CViewbasedOnHtmlPageView : public CWindowImpl{ public: DECLARE_WND_SUPERCLASS(NULL, CAxWindow::GetWndClassName()) BOOL PreTranslateMessage(MSG* pMsg) { if((pMsg->message < WM_KEYFIRST || pMsg->message > WM_KEYLAST) && (pMsg->message < WM_MOUSEFIRST || pMsg->message > WM_MOUSELAST)) return FALSE; // give HTML page a chance to translate this message 155 WTL Developer’s Guide return (BOOL)SendMessage(WM_FORWARDMSG, 0, (LPARAM)pMsg); } BEGIN_MSG_MAP(CViewbasedOnHtmlPageView) END_MSG_MAP() }; The view is based on CAxWindow. You will recall from our discussion of WTL Windowing, that CAxWindow will instantiate an ActiveX control to completely cover its window. Which ActiveX control depends on the window title – it could be a CLSID, a ProgID, in-line HTML (with the MSHTML prefix) or any string that CLSID_WebBrowser can understand, such as a URL, a pathname to a HTML or text file, or a pathname to a document from an Active Document Server (e.g. a Word 2000 document). The view window title is never displayed and normally when creating the view window, the WTL AppWizard uses a NULL window title. However, with Html Page, the title is used to indicate to the web browser what to render. The generated code is: //TODO: Replace with a URL of your choice m_hWndClient = m_view.Create(m_hWnd, rcDefault, _T("http://www.microsoft.com"), //  change as needed WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN, WS_EX_CLIENTEDGE); ActiveX Control The final part of step 2 of the WTL AppWizard is the Host ActiveX Controls checkbox. This is only enabled if all of the following conditions are true: • The project type in step 1 is not set to Dialog Based • The Enable ActiveX Control Hosting check-box in step 1 is selected • The view type in step 2 is set to Form (dialog-based) If you select Host ActiveX Controls then the generated view is based on CAxDialogImpl (otherwise it is based on CDialogImpl) and it is implemented as follows: class CViewBasedOnFormWithActiveXControlView : public CAxDialogImpl{ public: enum { IDD = IDD_VIEWBASEDONFORMWITHACTIVEXCONTROL_FORM }; BOOL PreTranslateMessage(MSG* pMsg) { if((pMsg->message < WM_KEYFIRST || pMsg->message > WM_KEYLAST) && (pMsg->message < WM_MOUSEFIRST || pMsg->message > WM_MOUSELAST)) return FALSE; HWND hWndCtl = ::GetFocus(); if(IsChild(hWndCtl)) { // find a direct child of the dialog from the // window that has focus while(::GetParent(hWndCtl) != m_hWnd) hWndCtl = ::GetParent(hWndCtl); 156 WTL Developer’s Guide // give control a chance to translate this message if(::SendMessage(hWndCtl, WM_FORWARDMSG, 0, (LPARAM)pMsg) != 0) return TRUE; } return IsDialogMessage(pMsg); } BEGIN_MSG_MAP(CViewBasedOnFormWithActiveXControlView) END_MSG_MAP() }; Threads with COM and Windowing Technically, there should be no problems using both the Win32 windowing APIs and COM together in a multithreaded application, as both can support calls from multiple threads provided suitable precautions are taken. How COM reacts in a multi-threaded environment is guided by whether CoInitialize or CoInitializeEx is called, and in the case of CoInitializeEx, what parameter is used. The two interesting parameters to CoInitializeEx are COINIT_APARTMENTTHREADED and COINIT_MULTITHREADED. The first indicates that for COM components instantiated on one thread, all method calls to an instance of a COM component will be on the same thread in which instantiation occurred. This effectively means that data members of that instance do not need synchronization protection, as they can only be used from one thread. Separate instances of the same component may be instantiated on separate threads, so global variables and static (class) members do need protection. COINIT_MULTITHREADED means that a component instance may be used simultaneously from different threads. Such components should provide synchronization protection for all writeable data members. COM should only be used in a thread if CoInitialize[Ex] has been called on that thread. It is perfectly acceptable to call CoInitialize[Ex] from a subset of the available threads (even only one) and use other threads for non-COM functionality. Calling CoInitialize is equivalent to calling CoInitializeEx with COINIT_APARTMENTTHREADED. CoInitializeEx is not available for the original (“classic”) Windows 95 distribution, but is available when the DCOM add-on is installed with Windows 95. CoInitializeEx is available for all later Windows OSes – 98, 98SE, Me, NT 4, 2000, CE and Whistler. CoInitialize is not available for Windows CE – you must use CoInitializeEx. Threads in which CoInitializeEx(COINIT_APARTMENTTHREADED) are called need a message loop, because COM uses a hidden window to ensure that incoming method calls are handled on the right thread at an appropriate time. Threads in which CoInitializeEx(COINIT_MULTITHREADED) are called do not need a message loop (for COM), because the incoming method calls are made on one of a queue of threads which the RPC run-time maintains. It is quite possible that the same or different clients make method calls to the same instance of a components, and these method call execute at the same time on different threads – the component needs to protect against synchronization problems. When dealing with WTL, we will have 157 WTL Developer’s Guide windows on screen, so we will also need a message loop for that, but still the RPC threads are used for the methods calls. The WTL AppWizard generated code is meant to run on all platforms, including the original Windows 95, and therefore the code contains a call to CoInitialize. There is also a comment containing a call to CoInitializeEx. // If you are running on NT 4.0 or higher you can use the // following call instead to make the EXE free threaded. This // means that calls come in on a random RPC thread. // HRESULT hRes = ::CoInitializeEx(NULL, // COINIT_MULTITHREADED); It is left as a design decision to the application developer to decide, which is a sensible strategy. With regarding to windowing, windows may be created on any thread. There is a message queue per thread and messages for a particular window are queued to the message queue of the thread in which the window was created. Calls to SendMessage may be make from a thread different to the thread upon which the window was created, but the thread that calls SendMessage will block until the message is actually processed by the thread that handles the message queue for the window. This can lead to deadlock situations in poorly designed applications. Calls to PostThreadMessage post the message on the message queue of the specified thread and returns immediately. Sample using Multiple Threads SDI with COM Server Now that we have examined the treading issues with windows and COM, and confirmed that one can have a multiple threads SDI application AND support COM, we will create a sample project. Even though the WTL AppWizard does not directly create a multi-threaded SDI app that supports COM Server, we can still use it to help us. Essentially what we have to do is use it to create two separate projects – one which is based on Multiple Threads SDI without the COM Server option, and the other based on the (single-threaded) SDI Application with the COM Server option, and merge the code to complete our project. We will start by creating a project called HelloWorldMT_WithCOMServer and in the WTL AppWizard select SDI Application with the COM Server option. We start with this as it contains most of the code we need. Next we create a dummy project, which we will discard, based on Multiple Threads SDI. It will help to have both of these projects open in separate instances of DevStudio, to facilitate copying and pasting. The previous discussion of Multiple Threads SDI contained a detailed description of the differences between the (single-threaded) SDI Application and the Multiple Threads SDI Application. We need to apply all these changes manually to the HelloWorldMT_WithCOMServer project source code and make appropriate changes. Firstly delete the generated Run function in HelloWorldMT_WithCOMServer. 158 WTL Developer’s Guide Copy the RunData structure and the RunThread and AddThread member functions from the CHelloWorldMTThreadManager class in the dummy Multiple Threads SDI Application to HelloWorldMT_WithCOMServer.cpp. Within WinMain, change the message loop code to the following: if(bRun) { _Module.StartMonitor(); hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER, REGCLS_MULTIPLEUSE | REGCLS_SUSPENDED); ATLASSERT(SUCCEEDED(hRes)); hRes = ::CoResumeClassObjects(); ATLASSERT(SUCCEEDED(hRes)); CMessageLoop theLoop; if (!bAutomation) AddThread(lpstrCmdLine, nCmdShow); nRet = theLoop.Run(); _Module.RevokeClassObjects(); ::Sleep(_Module.m_dwPause); } In mainfrm.h, add a message map entry: COMMAND_ID_HANDLER(ID_FILE_NEW_WINDOW, OnFileNewWindow) In the ResourceView, add a “New Window” entry to the file menu. Add the command handler for CMainFrame::OnFileNewWindow, which calls AddThread. LRESULT OnFileNewWindow(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ AddThread(0, SW_SHOWNORMAL); return 0; That’s it – now we have a multi-threaded SDI app which can work as a COM server. 159 WTL Developer’s Guide Chapter 6 Dialog Boxes and Controls Objectives The aims of this chapter are to: • Cover how WTL supports dialog boxes • Describe dynamic data exchange and validation functionality • Introduce the WTL wrappers for each type of window control Introduction Virtually all applications that expose a user interface contain one or more dialog boxes, each showing multiple controls. A much smaller proportion of applications need to display windows with a client area in which graphical primitives can be rendered. We will start our in-depth look at WTL by examining how it handles dialog boxes and controls. In later chapters we will cover client-area graphics. In WTL, the top-level window can be a dialog box itself, or can be a mainframe containing a child window known as a view, based on a dialog (resource template) or a single control. When the top-level window is not a dialog, it usually contains a command bar (menubar) and toolbar, whose entries when selected often result in dialog boxes being displayed. Dialogs use an assortment of standard controls, common controls and ActiveX controls. Data must be copied between these controls and data members of C++ classes representing the dialogs. The resource information containing the layout for the dialog may be created using a resource editor. The dialog must be displayed on screen and when appropriate removed. Dialogs may be modal or non-modal. Property sheets and wizards may be provided. There are numerous system dialogs that the OS provides to automate and standardize how common UI tasks should work. Message loops are needed to process messages. In our exploration of WTL dialogs and controls we will need to understand all these issues. What WTL Supports WTL provides comprehensive support for the complete range of dialog box functionality. It supports modal dialogs, modeless dialogs, and property sheets and provides easy-to-use wrappers for the system dialogs, such as OpenFile. WTL provides wrappers for each Win32 standard control, such as buttons, listboxes, edit boxes, and for each common control, such as listview, treeview, imagelist and 160 WTL Developer’s Guide monthcalendar. WTL supports more advanced functionality, which is not directly based on existing controls, including bitmap buttons, hyperlinks and a wait cursor. WTL supports the hosting of ActiveX controls through the existing ATL mechanism. In short, every aspect of dialog boxes development is very well catered for by WTL. One shortcoming is the integration between WTL and the development environment within Visual C++ v6. There is no WTL ClassWizard (the ClassWizard that is there is for MFC). What is currently provided is a New Windows Message and Event Handler wizard, which allows messages for CWindowImpl-derived classes to be mapped to function handlers. This was designed for ATL windowing and can also be used with WTL, as WTL is based on ATL windowing. It supports mapping all the WM_XXX messages for the dialog itself to member function, and for each control id, mapping a subset of notifications that they send to their parent (the dialog). The handler functions it creates are generic (they all have the same prototype). WTL offers message crackers, (e.g. when a lParam contains a POINT data, a WTL CPoint is used in the prototype). These are not supported by the current version of the wizard. It is likely with the next version of Visual Studio that the development environment will be extended with more Wizard support for WTL. The WTL Message crackers refer to the WM_XXX messages only. They do not provide any special functionality for the notification of a WM_COMMAND message. We will examine them in a later chapter. The concept of mapping UI controls to C++ classes (similar to the Member Variables tab of the MFC ClassWizard) is currently not supported in WTL. At the moment this has be done manually in source code. 161 WTL Developer’s Guide How WTL works with Dialog boxes and Controls To prepare to learn about dialog development with WTL, you first need a good understanding of traditional Win32 UI programming (from the “Petzoldian” era of Windows programming) and you need to know about ATL windowing. This might be a good point to review the earlier chapters on these topics. WTL uses both of these extensively, and most of WTL’s support is a set of lightweight wrappers and more functionality in some specific areas. What WTL adds is a set of wrappers for each Windows standard control and each Windows common control. WTL provides a member function for absolutely every message that can be sent to these controls. The naming of the member functions of these wrappers is an exact match for the messages that each control accepts (just one reason for the usefulness of a clear understanding of Win32 UI programming). In contrast, MFC sometimes renamed certain features or amalgamated them. For the dialog box template itself and message mapping, WTL uses ATL windowing. MFC provided elaborate message routing, which provided additional functionality but which often caused as many problems as it solved. In contrast, WTL/ATL windowing message routing is very simple and easy to understand. It is trivial to follow it in a debugger. Varying Degrees of Interaction with Controls Application developers need different levels of interaction with controls. It could be applications only need to determine when a specific message is received. For example, when a pushbutton is pressed its parent window (usually the dialog window), is sent a WM_COMMAND message with the BN_CLICKED notification. To detect these WTL provides the concept of message maps, which maps incoming messages to handlers within the dialog class. Alternatively, it could be that applications need to set and get data stored in the control. For example, when displaying a dialog, developers might wish to set the text to appear in an edit box, and when the dialog disappears to retrieve (the possibly updated) text. For this, WTL provides DDX functionality. Finally, it could be that applications wish to manage the control in detail. For this, WTL provides wrapper classes, such as CEdit, which provides the necessary fine-grained access to the control’s properties. WTL provides good support for all these levels of interaction with controls. It is a design decision developers will need to take as to which technique to use. Most applications will use all of them at some stage. Dialog Boxes in WTL Dialogs within WTL are based directly on ATL’s CDialogImpl or CAxDialogImpl. The “Ax” term means that the dialog provides the necessary support to host ActiveX controls. The following discussion of working with dialogs applies equally well to CDialogImpl and CAxDialogImpl-based dialogs. In the chapter on ATL Windowing we looked at the specifics of ActiveX controls hosting within CAxDialogImpl. 162 WTL Developer’s Guide WTL is a thin layer above ATL and Win32 SDK As with other parts of WTL, its dialog box support is a very thin layer above what is provided by ATL and the Win32 SDK. If you read through the WTL source code you will very quickly hit the Win32 SDK. WTL AppWizard-Generated Dialogs When you run the WTL AppWizard and as the Project Type select the Dialog option, the generated code contains one dialog that acts as the main window of the application. By default, the generated dialog is based on CDialogImpl. If you select the ActiveX Control Hosting option in the step 1, the dialog is based on CAxDialogImpl. Note that when you select Dialog as the Project Type in step 1 of the WTL AppWizard, then all options in Step 2 are disabled. If you select either a SDI or MDI application for Project Type, and in step 2 of the WTL AppWizard, select Form (Dialog-based) as the View Type, then by default the generated view is based on CDialogImpl. However, if the Host ActiveX Controls checkbox in step 2 of the WTL AppWizard is selected, then the view is based on CAxDialogImpl. Note that Host ActiveX Controls checkbox is normally disabled. It becomes enabled only when all three of these conditions are true: Project Type in step 1 is not set to Dialog, ActiveX Control Hosting is selected in step 1 and the View Type in step 2 is set to Form (Dialog-Based). Creating Additional Dialog Boxes Manually Apart from the WTL AppWizard generated dialog, you must create additional dialogs based on CDialogImpl manually. There is no wizard support. With Visual Studio 7 it is expected that wizard support will be provided. The current ATL Object Wizard does support the creation of CAxDialogImpl-based dialogs but not CDialogImpl. To add a CDialogImpl-based dialog, you need to create the dialog resource and a C+ + class to manage the dialog. To create the resource, display ResourceView in DevStudio’s workspace, select Dialog, and in the context menu select Insert Dialog. A dialog template will be added to the resource. [The term “template” used for resources refers to the layout of the controls – it is of course a totally distinct concept from a C++ template]. Use the ResourceView’s control palette to add controls onto the dialog and then set their various properties. Specify an IDD identifier for the dialog. To construct a C++ class, you will need to manually create a class with code similar to this: class CMyFirstManualDlg:public CDialogImpl{ public: enum { IDD = IDD_MY_FIRST_MANUAL_DIALOG }; BEGIN_MSG_MAP(CMyFirstManualDlg) MESSAGE_HANDLER(WM_INITDIALOG, OnInitDialog) COMMAND_ID_HANDLER(IDOK, OnOK) COMMAND_ID_HANDLER(IDCANCEL, OnCancel) END_MSG_MAP() LRESULT OnInitDialog(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/){ // center the dialog on the screen CenterWindow(); 163 WTL Developer’s Guide return TRUE; } LRESULT OnOK(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ // TODO: Add validation code EndDialog(wID); return 0; } LRESULT OnCancel(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ // TODO: Add validation code EndDialog(wID); return 0; } }; The enum IDD must be set to that you used in ResourceView. Also note that you must have a message map. This will implement a virtual function ProcessWindowMessage, which is needed to get everything to compile. These are tedious manual steps; after you have completed them for one dialog, the use of CTRL-C and CTRL-V mean it does not take too long, and hopefully VS7 will automate this. Instantiating and Displaying Dialogs Just as with MFC, one instantiates and displays a WTL dialog using this code: CMyFirstManualDialog dlg; int res = Dlg.DoModal(); To dismiss a dialog (usually after the user selecting OK or CANCEL), the dialog class itself will call EndDialog, and the parameter passed to this will become the return value from DoModal. Dialog Initialization When a dialog class is instantiated, the windows for the dialog and its control do not exist. A common mistake for new MFC developers and it applies equally to WTL is to have code such as: CMyFirstManualDialog dlg; Dlg.m_edit.SetWindowText(TEXT("My initial text")); // WRONG! int res = Dlg.DoModal(); The problem here is that the edit box does not exist in the hierarchy of windows until during the DoModal call and therefore there is no valid destination from the window message resulting from the call to SetWindowText. The proper way to handle this situation is to execute the code after the window has been created but before it has been displayed on screen. The OS send a WM_INITDIALOG message to the dialog when at exactly this point in time, and we can detect it with a message map entry: MESSAGE_HANDLER(WM_INITDIALOG, OnInitDialog) 164 WTL Developer’s Guide So the solution is as follows: CMyFirstManualDialog dlg; int res = Dlg.DoModal(); . . . class CMyFirstManualDialog : CDialogImpl{ BEGIN_MSG_MAP(CMyFirstManualDlg) MESSAGE_HANDLER(WM_INITDIALOG, OnInitDialog) END_MSG_MAP() LRESULT OnInitDialog(. . .){ m_edit.SetWindowText(TEXT(“My initial text”)); return TRUE; } . . . }; If the data to use when initializing the controls is known where we instantiate the dialog, and not inside the dialog class, then we could pass it in as a constructor parameter or add an extra method which gets called before DoModal. CMyFirstManualDialog dlg; Dlg.m_strName = TEXT(“My name”): int res = Dlg.DoModal(); . . . class CMyFirstManualDialog : CDialogImpl{ CString m_strName; . . . LRESULT OnInitDialog(. . .){ m_edit.SetWindowText(m_strName); return TRUE; } Dialog Termination Dialogs terminate when a call is made to CDialogImpl::EndDialog. It is up to you code to decide when to call this method. There is no default handling of WM_COMMAND messages such as IDCANCEL, which is sent to the dialog when the close button (‘X’ symbol) in the dialog’s top corner is pressed, or (by default) when the ESC key is pressed. This contrasts to MFC’s implementation, which in its default message map maps the IDCANCEL message to a call to MFC’s CDialog::OnCancel, which results in EndDialog being called. With WTL/ATL, if you do not handle the IDCANCEL message then nothing will happen – specifically the dialog box will not disappear, even when you pressed the Close button. It is the responsibility of your code to call EndDialog, and therefore you will usually add OK and Cancel pushbuttons, with ids of IDOK and IDCANCEL and map these to appropriate message handlers that will call EndDialog. Even if you do not have a Cancel button, you should still add a command handler for IDCANCEL, as the Close Button and the ESC key cause IDCANCEL to be sent to the dialog. 165 WTL Developer’s Guide Handling Dialog Messages DevStudio’s New Window Message and Event Handler wizard enables you to attach command handlers to a variety of messages for the dialog itself and all the controls it contains. This is similar to the message map feature in MFC. Per selected window/control, the list of possible messages is displayed, and you can select the Add Handler button to create a new command handler for that message. Sample : DialogDemo To explore dialogs, we will now create an example. Start up the WTL AppWizard and create a project called Dialogdemo. In Step 1, select modal dialog, and select the defaults in Step 2. In the dialog node in ResourceView, select the DialogDemo. Using the Controls palette, add a pushbutton to the dialog. Select the pushbutton and from the context menu bring up its properties. Set the id to IDC_MY_FIRST_BUTTON and give it a caption of “First Button”. We are now finished adding the button to the resource, and can move on to creating a handler to be activated when the button is pressed. In DevStudio’s ClassView, select the CMainDlg class which manages the dialog. In the context menu for it select Add Windows Message Handler. The following dialog will be displayed. 166 WTL Developer’s Guide In the Class or object to handle list, select IDC_MY_FIRST_BUTTON. The messages that are most commonly needed for a window of the selected type are displayed. In this case, a button is selected, and the messages are BN_CLICKED and BN_DOUBLECLICKED. Select BN_CLICKED and then select Add Handler. You are given an opportunity to enter a name for the handler function, but normally you should select the default. The message map will automatically have this line added: COMMAND_HANDLER(IDC_MY_FIRST_BUTTON, BN_CLICKED, OnClickedMy_first_button) A default implementation of the handler will also be added, and you can add a call to Win32’s Beep function. LRESULT OnClickedMy_first_button(WORD wNotifyCode, WORD wID, HWND hWndCtl, BOOL& bHandled){ ::Beep(200,200); return 0; } DDX Passing data in and out of dialog controls is very frequently needed. It can be done using low-level window messages, but that involves many lines of code if done manually. Naturally developers wished to minimize the effort involved, and out of this grew the concept of dynamic data exchange (DDX). MFC developers will be quite familiar with DDX, as it has been used there for many years. WTL’s DDX support is 167 WTL Developer’s Guide modeled on MFC’s support. These are some differences, but in general, if you understand DDX in MFC you will very quickly understand WTL’s also. To get started with DDX in a WTL application you will need to carry out a few manual steps. Firstly you will have to include atlddx.h in your project. Adding it to stdafx.h is the recommended location. Then you will have to derive your dialog class from WTL’s CWinDataExchange, set up a DDX Map, call DoDataExchange when needed and finally think about error handling. It is likely that in future the wizards will be updated to automate the necessary support. CWinDataExchange The heart of WTL’s DDX support is a WTL template called CWinDataExchange. It contains a method called DoDataExchange that the application has to over-ride (usually using a DDX Map). This contains the list of dialog-specific exchanges needed. CWinDataExchange also contains methods called OnDataExchangeError and OnDataValidateError for error handling, and the application may over-ride these if desired. The bulk of the methods inside CWinDataExchange are methods to transfer data in and out of controls. These include: • DDX_Text – text based on LPTSTR, BSTR, CComBSTR & WTL’s CString • DDX_Int – integer • DDX_Float – float and double • DDX_Control – control sub-classing (nothing to do with ActiveX controls) • DDX_Check – check box • DDX_Radio – group of radio buttons You derive your dialog box’s C++ class from the CWinDataExchange template and this gives you access to these methods. Each DDX method in CWinDataExchange takes in a Boolean parameter, which specifies whether a save or a load operation is to be carried out. Note that WTL’s CWinDataExchange does contain a default implementation of DoDataExchange – when called it simply asserts and throws you into the debugger. If you omit the DDX map then this will happen. To solve the problem you will simply have to add an appropriate DDX map. DDX Maps Instead of calling the data transfer methods, such as DDX_Int, directly, WTL provides the concept of DDX Maps. One of these maps should exist for each dialog class that uses DDX. The DDX Map implements the DoDataExchange function and its essential role is to map the control IDs of dialog controls to C++ data members. Data is transferred from the controls to the C++ data members when DoDataExchange is called with a TRUE parameter, and in the reverse direction when called with FALSE. 168 WTL Developer’s Guide Calling DoDataExchange The application developer needs to decide when to call DoDataExchange. Unlike MFC, it is not done automatically for you. The controls must actually exist when you call DoDataExchange. Typically you will add a call to DoDataExchange(FALSE) to OnInitDialog, and a call to DoDataExchange(TRUE) to OnOK. Sample : DataExchangeDemo To show a simple example of how to use WTL’s data exchange, create a project called DoDataExchange. Add this line at the bottom of stdafx.h: #include In the dialog resource, add a text box with an id of IDC_MYFIRST_INT. Change the definition of CMainDlg so it derives from CWinDataExchange: class CMainDlg : public CDialogImpl, public CWinDataExchange Add an integer data member to the CMainDlg class: int m_num; Add a data map with an entry mapping m_num data member to the text box in the dialog: BEGIN_DDX_MAP(CMainDlg) DDX_INT(IDC_MYFIRST_INT, m_num) END_DDX_MAP() Put calls to DoDataExchange in OnInitDialog and OnOK. LRESULT OnInitDialog(. . .){ . . . m_num = 43; // initialize m_num here DoDataExchange(FALSE); // FALSE means copy TO the dialog return TRUE; } LRESULT OnOK(){ DoDataExchange(TRUE); // TRUE means copy FROM the dialog // use m_num here EndDialog(wID); return 0; } In OnInitDialog, the data is transferred from the C++ data member to the dialog controls using the call to DoDataExchange. This means that the C++ data member values may be set before the call as needed. As they are only C++ data members, then may also be set where the dialog class is instantiated. CMainDlg dlg; Dlg.m_num = 20; Dlg.DoModal(); 169 WTL Developer’s Guide The actual window for the control does not need to exist at this point. It does need to exist when DoDataExchange is called. If you wish to carry out data exchange for a particular control, you can pass its id as the second parameter to DoDataExchange. If our DDX_MAP had many entries and for some reason we only wished to perform data exchange for the IDC_MYFIRST_INT control, we could call DoDataExchange as follows: DoDataExchange(TRUE, IDC_MYFIRST_INT); DDX With Integers The member function CWinDataExchange::DDX_Int is responsible for handling integer exchange. Its prototype is: template BOOL DDX_Int(UINT nID, Type& nVal, BOOL bSigned, BOOL bSave, BOOL bValidate = FALSE, Type nMin = 0, Type nMax = 0); At a minimum, its needs a control’s id and variable as parameters. The macros to put inside the DDX map which using DDX_Int are: • DDX_INT(nID, var): exchanges signed integer values, no validation • DDX_INT_RANGE(nID, var, min, max): exchanges signed integer values and for validation ensures it is between min and max (the max and min values themselves are valid in the range) • DDX_UINT(nID, var): exchanges unsigned integer values, no validation • DDX_UINT_RANGE(nID, var, min, max): exchanges unsigned integer values and for validation ensures it is between min and max (the max and min values themselves are valid in the range) For each transfer method in CWinDataExchange, there are multiple DDX macros, which call it using various combinations of parameters. If validation is turned on (i.e. using the DDX_INT_RANGE or DDX_UINT_RANGE macros), then when loading var must be between nMin and nMax (inclusive) – this is enforced by an assertion; when saving if var is not in the nMin/nMax range then the OnDataValidateError method is called. Note the difference between loading and saving – with loading the condition must be true – if false (and running in debug mode) an assertion is activated, whereas with saving the condition may be false and if so causes a call to the error handler but the application continues running. DDX with Checkboxes and Radio-Buttons The DDX member functions for check boxes and radio buttons are: void DDX_Check(UINT nID, int& nValue, BOOL bSave) void DDX_Radio(UINT nID, int& nValue, BOOL bSave) For the checkbox, the valid values are 0, 1 and 2, which correspond to unchecked, checked and indeterminate (for tri-state buttons). When transferring data to the check 170 WTL Developer’s Guide box control, if the value is not 0,1 or 2, then ATLTRACE2 is called to print out an error message in the output window – note that OnDataExchangeError is not called. When loading, the nValue must be 0, 1 or 2 – this is enforced by an assertion. For a radio button group, the members of the group are delimited by the control id of the first radio button in the group, and the next control with the WS_GROUP attribute set (this is one past the end). You can arrange the ordering of controls by selecting tab order from the layout menu in resource view. You will need to set the WS_GROUP property for the first radio button in the group and for the control one past the last radio button in tab order. If there are members of the group that are not radio-buttons (this should not be the case), then this message is sent to the output window ATL: Warning - skipping non-radio button in group. Only one radio button in the group can be selected and the control id parameter must be that of a radio button control (both of these should be the case) – this is enforced by assertions. The DDX macros are: • DDX_CHECK(nID, var) : exchanges between an unsigned integer and a check box • DDX_RADIO(nID, var) : exchanges data between an unsigned integer and radio button group DDX with Floating Point Values The DDX member functions for floating point values are: BOOL DDX_Float(UINT nID, float& nVal, BOOL bSave, BOOL bValidate = FALSE, float nMin = 0.F, float nMax = 0.F) BOOL DDX_Float(UINT nID, double& nVal, BOOL bSave, BOOL bValidate = FALSE, double nMin = 0., double nMax = 0.) Both floats and doubles are supported. At a minimum a control id and a data member are needed as parameters. The macros for floats/doubles are: • DDX_FLOAT(nID, var) – data exchange only (no validation) • DDX_FLOAT_RANGE(nID, var, min, max) : data exchange and validation The floating point functionality requires the C-Run-Time library. If you wish to use floating point DDX you will need to add a definition to the preprocessor of _ATL_USE_DDX_FLOAT to both the debug and release build settings, and remove the definition of _ATL_MIN_CRT from the release build settings (it was put these automatically by the WTL AppWizard). You cannot have both of these defined in the preprocessor – as one is saying include the CRT and the other is saying exclude the CRT. If you forget, these lines in atlddx.h will remind you: 171 WTL Developer’s Guide #if defined(_ATL_USE_DDX_FLOAT) && defined(_ATL_MIN_CRT) #error Cannot use floating point DDX with _ATL_MIN_CRT defined #endif //defined(_ATL_USE_DDX_FLOAT) && defined(_ATL_MIN_CRT) If validation is turned on (i.e. using the DDX_FLOAT_RANGE macro), then when loading the following must be true (this is enforced by an assertion): nVal >= nMin && nVal <= nMax When saving if nVal < nMin || nVal > nMax is true, then the OnDataValidateError method is called. Similarly to integer processing, if validation is turned on (i.e. using the DDX_FLOAT_RANGE macro), then when loading var must be between nMin and nMax (inclusive) – this is enforced by an assertion; when saving if var is not in the nMin/nMax range then the OnDataValidateError method is called. Also an assertion ensures that nMin is not equal to nMax. DDX with Strings CWinDataExchange offers a number of member functions dealing with data exchange for text. They are: BOOL DDX_Text(UINT nID, BOOL bValidate = BOOL DDX_Text(UINT nID, BOOL bSave, BOOL BOOL DDX_Text(UINT nID, BOOL bSave, BOOL BOOL DDX_Text(UINT nID, BOOL bSave, BOOL LPTSTR lpstrText, int nSize, BOOL bSave, FALSE, int nLength = 0); BSTR& bstrText, int /*nSize*/, bValidate = FALSE, int nLength = 0); CComBSTR& bstrText, int /*nSize*/, bValidate = FALSE, int nLength = 0); CString& strText, int /*nSize*/, bValidate = FALSE, int nLength = 0); The string data types they support are LPTSTR, BSTR, CComBSTR and WTL::CString. Note that WTL::CString is only supported if _ATL_TMP_NO_CSTRING is NOT defined (and usually it is not), and if the header file atlmisc.h is included. Each of these exchange functions transfer string data to a textbox control. The optional validation is the length of the string. If the string is to accept numeric characters only, then the NUMBER property should be set in the resource editor for the text-box. 172 WTL Developer’s Guide The DDX macros for text are: • DDX_TEXT(nID,var)- data exchange between text box and string data member (no validation) • DDX_TEXT_LEN(nID, var, len) - data exchange between text box and string data member, and also validation of string length The var field can be one of LPTSTR, BSTR, CComBSTR or CString and depending on its data type the appropriate DDX_Text member function will be called. If saving and validation is turned on, then the length field must be greater than zero (this is enforced by an assertion). If loading, and validation is turned on, and the data or type is one of LPTSTR, BSTR or CComBSTR (but not CString), then the number of characters in the string must be greater than the nLength parameter (this is enforced by an assertion). DDX For Control Sub-Classing To support subclassing of controls, CWinDataExchange offers the DDX_Control method. It simply calls the CWindowImplBaseT< TBase, TWinTraits >::SubclassWindow, which we discussed in the coverage of ATL Windowing. One macro is provided to support subclassing: • DDX_CONTROL(nID, Control) – the control class subclasses the control window in the dialog box Sample: CallMacrosDlg in DataExchangeDemo We will now extend the DataExchangeDemo with a dialog box that shows all these macros in use. As we wish to use floating-point, we first add _ATL_USE_DDX_FLOAT as a preprocessor definition in both the debug and release build settings. We also remove _ATL_MIN_CRT from the release build preprocessor definitions. As we wish to use the WTL’s CString functionality and the DDX functionality, we add atlmisc.h and atlddx.h to stdafx.h. 173 WTL Developer’s Guide In the ResourceView, create a new dialog template, with the id of IDD_ALL_MACROS_DLG and the title of All Macros. Lay out a series of controls as follows: Create a new dialog class called CAllMacrosDlg (say, in a file called allmacrosdlg.h). Add variables of each type we need: int m_nSignedIntNoValidation; int m_nSignedIntValidation; unsigned int m_nUnsignedIntNoValidation; unsigned int m_nUnsignedIntValidation; int m_nCheckBoxValue; int m_nRadioButtonValue; float m_nFloatNoValidation; float m_nFloatValidation; CString m_nStringNoValidation; CString m_nStringValidation; Now manually construct a DDX map with all the entries: BEGIN_DDX_MAP(CAllMacrosDlg) DDX_INT(IDC_SIGNED_INT_NO_VAL, m_nSignedIntNoValidation) DDX_INT_RANGE(IDC_SIGNED_INT_VAL, m_nSignedIntValidation, 10,20) DDX_UINT(IDC_UNSIGNED_INT_NO_VAL, m_nUnsignedIntNoValidation) DDX_UINT_RANGE(IDC_UNSIGNED_INT_VAL, m_nUnsignedIntValidation, (unsigned int) 30, (unsigned int) 40) DDX_CHECK(IDC_CHECK_BOX, m_nCheckBoxValue) DDX_RADIO(IDC_RADIOBUTTON_GROUP, m_nRadioButtonValue) DDX_FLOAT(IDC_FP_NO_VAL, m_nFloatNoValidation) DDX_FLOAT_RANGE(IDC_FP_VAL, m_nFloatValidation, 50.0, 60.0) DDX_TEXT(IDC_STRING_NO_VAL, m_nStringNoValidation) 174 WTL Developer’s Guide DDX_TEXT_LEN(IDC_STRING_VAL, m_nStringValidation, 20) END_DDX_MAP() Note that the DDX_MAP is entirely separate from the message map and each control can appear in neither, or both or any one of the two maps. In OnInitDialog, initialize the variables, and then call DoDataExchange. m_nSignedIntNoValidation = 1; m_nSignedIntValidation = 15; m_nUnsignedIntNoValidation = 3; m_nUnsignedIntValidation = 35; m_nCheckBoxValue = 1; // set check box m_nRadioButtonValue = 2; // third entry - zero'd // based index; m_nFloatNoValidation = 10.0; m_nFloatValidation = 55.0; m_nStringNoValidation = TEXT("A Wonderful String"); m_nStringValidation = TEXT("Another String"); DoDataExchange(FALSE); Validation occurs when transferring data from C++ data members to dialog controls. So it is where we call DoDataExchange(FALSE), usually inside OnInitDialog, where validation can fail. In OnOK, call DoDataExchange(TRUE), which results in the updated values being copied from the dialog box controls into the member variables. Sample : CStringWithFloat While on the subject of support for floating point, we will now return to the CString class. WTL’s CString::Format provides printf-like formatting functions. Currently it does not support floating point values (e.g. “%f”). WTL’s DDX optionally does supports floating-point (based on a #define), but this requires the inclusion of the C Run Time (CRT) library. What happens if we wish to use floats with WTL’s CString::Format? It is likely that this support will be built into CString in future, but for now we have to do some work. The CStringWithFloat sample explores how to add this support. It has a class CStringWithFloat that derives from WTL’s CString. It has two member functions, Format and FormatV. Format is a wrapper function for FormatV. CStringWithFloat::Format is the same implementation as CString::Format. It is there to ensure that it calls CStringWithFloat::FormatV and not CString::FormatV. If the macro _ATL_USE_CSTRING_FLOAT is defined, we add support for floats (this is similar to how floats are supported in WTL’s DDX code). The implementation of CStringWithFloat::FormatV is very similar to that of CString::FormatV, with two modifications. Firstly, when walking the argument list to determine the length for the CString buffer, we add this code: #ifdef _ATL_USE_CSTRING_FLOAT 175 WTL Developer’s Guide va_arg(argList, float); nItemLen = 32; nItemLen = max(nItemLen, nWidth+nPrecision); #else ATLASSERT(!"Floating point (%%e, %%f, %%g, and %%G) is “ “not supported by the WTL::CString class."); #endif // !_ATL_USE_CSTRING_FLOAT Secondly, when actually formatting, we use the CRT function _vstprintf (which can handle floats and is TCHAR-friendly) rather than wvsprintf from the Win32 API (which does not support floats): #ifdef _ATL_USE_CSTRING_FLOAT int nRet = _vstprintf(m_pchData, lpszFormat, argListSave); #else int nRet = wvsprintf(m_pchData, lpszFormat, argListSave); #endif We can use this as follows: CStringWithFloat str; str.Format("This is a float = %f\n", 4354.56); The DDX Map Macros The aim of the DDX map macros is to simplify the construction of the DoDataExchange function. The begin and end macros start and finish the function, and the various data transfer macros call the transfer function when appropriate. The BEGIN_DDX_MAP macro is defined as: #define BEGIN_DDX_MAP(thisClass) \ BOOL DoDataExchange(BOOL bSaveAndValidate = FALSE, UINT nCtlID = (UINT)-1) \ { \ The data transfer macros are all defined similarly. Here is the one for DDX_INT: #define DDX_INT(nID, var) \ if(nCtlID == (UINT)-1 || nCtlID == nID) \ { \ if(!DDX_Int(nID, var, TRUE, bSaveAndValidate)) \ return FALSE; \ } # The END_DDX_MAP macro is defined as: define END_DDX_MAP() \ return TRUE; \ } So the DDX map . . . BEGIN_DDX_MAP(CMainDlg) DDX_INT(IDC_MYFIRST_INT, m_num) END_DDX_MAP() . . . gets converted to this code: BOOL DoDataExchange(BOOL bSaveAndValidate = FALSE, 176 WTL Developer’s Guide UINT nCtlID = (UINT)-1){ if(nCtlID == (UINT)-1 || nCtlID == nID) { if(!DDX_Int(nID, var, TRUE, bSaveAndValidate)) return FALSE; } return TRUE; } The vast majority of times, DoDataExchange will be called without the second control id parameter. This means it defaults to –1, which results in all the DDX transfer functions being called. For specialist needs you can call DoDataExchange and pass it in a control id, when you need to carry out data transfer for a single control. The first parameter, bSaveAndValidate, specifies if a save or a load is occurring. If saving is occurring, then validation is an optional extra – it is called using some macros and not with some others. In our previous listing of all the available DDX macros, we noted those that provided validation. Detailed Look at one DDX Transfer Method To gain a deeper understanding of how data transfer works, we will look in more detail at one sample from the WTL source code. DDX_Text is called when any of the DDX_TEXT macros are used. The last two parameters, bValidate and nLength, are only used from DDX macros which require validation. The save parameters states whether we are copying data to the control or from the control to the string. The control id specific that control in the dialog we wish to exchange data with. Its actual HWND is retrieved in a call to Win32’s GetDlgItem. The lpstrText parameter is a buffer, filled with text if loading, and allocated with nSize characters if saving. Here is the implementation (we added comments inline): BOOL DDX_Text ( UINT nID, // RETURN: Did operation succeed? // IN: The control id used in // the transfer LPTSTR lpstrText, // IN/OUT: the text string // (buffer must be already allocated) int nSize, // IN: the length of the buffer BOOL bSave, // IN: Saving or loading BOOL bValidate = FALSE, // IN: If saving, should we validate int nLength = 0 // IN: If validating, the length ) { T* pT = static_cast(this); BOOL bSuccess = TRUE; if(bSave){ // ----------------------------------------------// COPY data from the control to the string buffer 177 WTL Developer’s Guide // ----------------------------------------------// calls Win32’s ::GetDlgItem HWND hWndCtrl = pT->GetDlgItem(nID); int nRetLen = ::GetWindowText(hWndCtrl, lpstrText, nSize); nLength); // If buffer was not large enough, flag an error if(nRetLen < ::GetWindowTextLength(hWndCtrl)) bSuccess = FALSE; } else { // ----------------------------------------------// COPY data from the string buffer to the control // ----------------------------------------------ATLASSERT(!bValidate || lstrlen(lpstrText) <= bSuccess = pT->SetDlgItemText(nID, lpstrText); } // ----------------------------------------------// If transfer was unsuccessful, call the data // exchange error handler and return FALSE // ----------------------------------------------if(!bSuccess){ pT->OnDataExchangeError(nID, bSave); } else if(bSave && bValidate) { // -----------------------------------------// Perform validation if required - note that // many DDX macros do not mandate validation // -----------------------------------------ATLASSERT(nLength > 0); if(lstrlen(lpstrText) > nLength){ // ----------------------------------------------// If validation failed, fill in the _XData // structure and call the validation error handler // ----------------------------------------------_XData data; data.nDataType = ddxDataText; data.textData.nLength = lstrlen(lpstrText); data.textData.nMaxLength = nLength; pT->OnDataValidateError(nID, bSave, data); bSuccess = FALSE; } } return bSuccess; } Error Handling Errors can occur during data transfer. Four scenarios are catered for: 178 WTL Developer’s Guide • General data exchange problems • Validate upon loading • Data exchange error • Validate upon saving The initial two are covered by assertions in the WTL code. There are coding errors and ATLASSERTs are sprinkled around the DDX code to protect against these. These could include trying to validate that a string has a length of zero, setting the check box to a value outside the range 0-1-2, etc. In our discussion of the DDX macros/DDX data exchange functions we noted all the rules to do with each data type. To handle the latter two situations, CWinDataExchange provides two overrideable error handlers, OnDataExchangeError and OnDataValidateError. A data exchange error will occur if a window with the specific control id does not exist. This is common if creating a dialog resource with a control which has a particular control id, and you use this in a DDX map; and later you edit the dialog resource and change/delete the control, but do not change the DDX map. The DDX map is left with an entry for a non-existent control id. When a data exchange error occurs, this error is detected and OnDataExchangeError is called. The problem can easily be fixed by ensuring that all entries in the DDX map correspond to entries in the dialog. When validating upon saving, then a validation error occurs (e.g. when a string is longer than permitted, or a numeric value is outside the permitted range), then OnDataValidateError is called. The reason there is a difference in strategy between validating upon load (use asserts if an error occurs) and validate upon save (call OnDataValidateError if an error occurs), is that for loading, the data is coming from the application code and should be correct, whereas when saving the data is coming from the end-user and could well be incorrect. An alternative for text validation is to set the text limit for an edit box. WTL’s CEdit provides the SetLimitText method, which internally sends an EM_SETLIMITTEXT message. Note that this affect what the user types into the edit box, but not what might be added through WM_SETTEXT messages or what is already in the edit box. Error Handlers CWinDataExchange has two data exchange handlers, OnDataExchangeError and OnDataValidateError. Their default implementations are: // Overrideables void OnDataExchangeError(UINT nCtrlID, BOOL /*bSave*/){ // Override to display an error message ::MessageBeep((UINT)-1); T* pT = static_cast(this); ::SetFocus(pT->GetDlgItem(nCtrlID)); } void OnDataValidateError(UINT nCtrlID, BOOL /*bSave*/, 179 WTL Developer’s Guide _XData& /*data*/){ // Override to display an error message ::MessageBeep((UINT)-1); T* pT = static_cast(this); ::SetFocus(pT->GetDlgItem(nCtrlID)); } They simply beep and set the focus to the offending control. Note that if the control id is incorrect (a likely possibility for calls to OnDataExchangeError), then the call to SetFocus will fail (but the app will not crash). Applications can provide custom implementation of these error handlers. It could be argued that these are coding errors and not end-user errors causes calls to OnDataExchangeError, and therefore an ATLASSERT statement would be appropriate. These are serious errors, and should be highlighted during debugging. Alternatively, a message box could be displayed stating that an error occurred. The enduser cannot do anything to fix the problem. Maybe the problem will still let the user work with other parts of the application, but to fix it a new build is required from the developer. Calls to OnDataValidateError occur due to end-user action. These are to be expected, and the application should robustly handle all of the validation errors and definitely continue running. End-users often do not understand what has gone wrong from a simple beep and the focus. They will probably need more descriptive error description. You will have to provide this by overriding OnDataValidateError. Per dialog, the same validation error handler is called for all validation errors. WTL provides the handler with detailed identification of what went wrong, you’re your implementation can display it to the end-user. The handler is passed in a _XData variable. The nDataType field identifies the data type that is in error and the union contains additional information. struct _XData { _XDataType nDataType; union { _XTextData textData; _XIntData intData; _XFloatData floatData; }; }; The nDataType field will be set to one of: enum _XDataType { ddxDataNull = 0, ddxDataText = 1, ddxDataInt = 2, ddxDataFloat = 3, ddxDataDouble = 4 }; 180 WTL Developer’s Guide Note that ddxDataNull and ddxDataDouble are not used in the current implementation (even DDX_Float when called with doubles uses ddxDataFloat when a validation error occurs). The additional information structures identify the permissible and the actual values: struct _XTextData { int nLength; int nMaxLength; }; struct _XIntData { long nVal; long nMin; long nMax; }; struct _XFloatData { double nVal; double nMin; double nMax; }; The _XData is only used for validation errors – it is not used for data exchange errors. Custom Validation Error Handler It is recommended that you provide a custom validation error handler for your dialogs if you are using the DDX validation features. First you will need to add strings to the application’s string table in the resources. In our implementation we will use the following: IDS_VALIDATE_TEXTERR You entered a text string of length %d, whereas the maximum allowed is %d IDS_VALIDATE_INTERR Error detected: You entered integer of %d, whereas it must be between %d and %d inclusive IDS_VALIDATE_FLOATERR Error detected: You entered floating-point number of %f, whereas it must be between %f and %f IDS_VALIDATE_UNEXPECTEDERR Unexpected error during validation Then we will implement the error handler. First we set the focus to the errant control and we select all its text. Validation errors occur with edit boxes and the particular one with the problem is highlighted when its contents are selected. Then we switch on the data type, and display a message box with as much information describing the problem. Even when the data type is float, we wish to display the current and permissible range. However, WTL’s CString does not permit floating point in its Format method. Our specialization of CString, call CStringWithFloat, does, so we use it here. (An alternative would be to directly call the CRT’s sprintf). When we have formatted the error message we display it in a message box. 181 WTL Developer’s Guide void OnDataValidateError(UINT nCtrlID, BOOL /*bSave*/, _XData& data){ CString strFormat; CStringWithFloat strError; ::SetFocus(GetDlgItem(nCtrlID)); ::SendMessage(GetDlgItem(nCtrlID), EM_SETSEL, 0, -1); switch (data.nDataType){ case ddxDataText: strFormat.LoadString(IDS_VALIDATE_TEXTERR); strError.Format(strFormat, data.textData.nLength, data.textData.nMaxLength); break; case ddxDataInt: strFormat.LoadString(IDS_VALIDATE_INTERR); strError.Format(strFormat, data.intData.nVal, data.intData.nMin, data.intData.nMax); break; case ddxDataFloat: strFormat.LoadString(IDS_VALIDATE_FLOATERR); strError.Format(strFormat, data.floatData.nVal, data.floatData.nMin, data.floatData.nMax); break; case ddxDataDouble: // fall through to default case ddxDataNull: // fall through to default default: strError.LoadString(IDS_VALIDATE_UNEXPECTEDERR); } MessageBox(strError); } Typically the same implementation will work for all your dialogs. You could define it as one big macro, such as #define ON_MY_DATA_VALIDATE_ERROR \ CString strFormat; \ . . . \ MessageBox(strError); \ } and then add the macro to each of your dialogs: class CMyFirstManualDialog : CDialogImpl{ . . . ON_MY_DATA_VALIDATE_ERROR BEGIN_DDX_MAP . . . }; Alternatively, you could define it in a template and derive your dialog class from it. Processing the return value from DoDataExchange We have seen how to create a custom implementation of the DoDataExchange method using DDX maps populated with DDX macros. When data transfer errors occur in one of the transfer method, then the DDX macros immediately return false from 182 WTL Developer’s Guide DoDataExchange. The subsequent DDX macros are not called at all. The DoDataExchange method returns a Boolean specifying if the data exchange succeeded. It should be checked, and if false action taken. Imagine we have a dialog box with two edit controls, identified by IDC_FIRST_VAL and IDC_LAST_VAL. Imagine we used this DDX map to conduct data exchange with it. BEGIN_DDX_MAP(CMyDlg) DDX_INT(IDC_FIRST_VAL, m_nFirstVal) DDX_INT(IDC_MIDDLE_VAL, m_nMiddleVal) DDX_INT(IDC_LAST_VAL, m_nLastVal) END_DDX_MAP() As no control with the id IDC_MIDDLE_VAL exists, data exchange will fail. We have two problems, the DDX_Int call with IDC_MIDDLE_VAL failed, and the DDX call IDC_LAST_VAL is never actually made. Here the problem will occur when initializing the data. The OnDataExchangeError method will have been called from inside DDX_Int, and then it returns false, which causes DoDataExchange to return false. Such errors are very serious as they usually indicate a coding error. If an error occurred, then it usually is better to display an error message and not displayed the half-initialized dialog box. Some developer make the mistake of thinking that returning false from OnInitDialog will accomplish this, which is incorrect. The Platform SDK description for WM_INITDIALOG states: “The dialog box procedure should return TRUE to direct the system to set the keyboard focus to the control specified by wParam. Otherwise, it should return FALSE to prevent the system from setting the default keyboard focus.” If you do not want the dialog to be displayed, call EndDialog. In this snippet we assume we have declared an error string resource IDS_DATA_TRANSFER_ERR with a suitable error message. LRESULT OnInitDialog(. . .){ . . . if (!DoDataExchange(FALSE)){ strFormat.LoadString(IDS_DATA_TRANSFER_ERR); MessageBox(strError); EndDialog(wID); } return TRUE; } If we manage to transfer data to the controls by successfully calling DoDataExchange(FALSE) in OnInitDialog, then that probably means we will not have data exchange problems when calling DoDataExchange(TRUE), assuming no one have dynamically destroyed some of the controls! However, there can very easily be validation errors when saving. These validation errors are to be expected, and are not a programming error – that there are merely end-user mistakes. We need to be able to robustly handle them. We have seen how our validation handler will be called, and the offending control highlighted and focus set upon it. However, typically we call DoDataExchange(TRUE) in our OnOk handler – and afterwards calls EndDialog to dismiss the dialog. 183 WTL Developer’s Guide An interesting problem can occur here. If a validation error arises, we do set the focus etc., but if we then call EndDialog, the dialog will be immediately dismissed. Instead, what we need to do is check the return from DoDataExchange, and only if it is successful, should we call EndDialog. If it is not successful, then the dialog will remained displayed, with the offending control highlighted. LRESULT OnOK(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ if (DoDataExchange(TRUE)) EndDialog(wID); return 0; } As you certainly do not wish to trap users in the dialog until then get the right entries in all the fields, it is essentially from a usability viewpoint to provide them with a Cancel button, so that if they are encountering validation errors, and do not know/want to fix them, then at least they can select Cancel and dismiss the dialog. So to recap, when loading the dialog with values, we only display it if no problems occur. When dismissing the dialog, we only hide it if no errors occur. What WTL’s DDX Does Not Support There are a few desirable features missing from the current implementation of WTL’s DDX. It does not support DDX for list-boxes or combo-boxes. Neither does it support DDX with an ActiveX control – we examined in the ATL chapter how this has to be done. It does not have a masked edit-box functionality, which would allow on-the-fly validation (after each entered character). Differences between MFC DDX and WTL DDX The following table summarizes the differences between DDX within WTL and MFC. Feature In MFC In WTL Creation of DDX maps Automatic, when inserting dialog Manual Populating DDX maps Automatic, via MFC’s ClassWizard Manual (Wizard support expected in VC++ 7) Class support Built into CDialog Need to manually derive from CWinDataExchange Calling DoDataExchange Built in (which can sometimes be a problem) You decide when to call (a little extra work, but more flexible) Validation Location Separate to DDX (DDV) Part of DDX Validation Direction Occurs when retrieving data from dialogs Occurs when transferring data to and from dialog controls 184 WTL Developer’s Guide Range of DDX macros Widespread Text, ints, floats, checkboxes and radiobuttons – no support for listboxes and comboboxes WTL Wrappers For The Standard Controls WTL provides a range of wrapper classes for the standard Windows controls. These simplify the sending of messages to the control to manage its behavior. It should be noted that for most cases, the use of message maps to detect user interface actions (e.g. BN_CLOCKED) and DDX maps to exchange data are all that is needed. The use of these wrapper controls is only necessary when more advanced management of controls is needed. The wrapper classes are: • CStatic • CButton • CListBox • CComboBox • CEdit • CScrollBar Each wrapper consists of a constructor, a function member called GetWndClassName that returns the window class name of the control, and a method for each valid type of message that can be sent to that particular control type. The wrappers are in fact templates and have T appended to their names. The templates have a single template parameter, which is usually CWindow. For each control there is a typedef of the control using CWindow, and this is named without the ’T’. typedef CStaticT CStatic; The simplest wrapper is CStatic, and it exposes methods for each message it accepts. The method implementation simply calls SendMessage with the appropriate message. A sample implementation is as follows: HICON GetIcon() const { ATLASSERT(::IsWindow(m_hWnd)); return (HICON)::SendMessage(m_hWnd, STM_GETICON, 0, 0L); } The wrappers set up the wParam and lParam fields correctly and also provide type safety. 185 WTL Developer’s Guide Using the Wrappers To use the wrapper templates you will first have to include atlctrls.h, usually in stdafx.h. Then in the dialog box in which you wish to use the wrappers, you will need to add data members: CEdit m_Edit; Then you will need to attach to the windows: M_Edit.Attach(GetDlgItem(IDC_ANY_EDITBOX)); After that you can use the wrapper as needed to send message to the control. Note that you would normally not use the wrapper to react to incoming messages for a particular control. These are normally sent to the control’s parent (the dialog box itself), and as discussed in the previous section on dialog box message maps, the Add Message Wizard understands the messages individual controls will receive, and support their detection. Sample: Standard Controls Dialog to DataExchangeDemo Now we will add a dialog called StandardControls to our demo project. In ResourceView add a static based on a picture, a button based on a checkbox, a listbox, a combobox, an editbox and a scrollbar. Add member variables for each of these to the dialog’s definition. CStatic m_Static; CButton m_Button; CListBox m_ListBox; CComboBox m_ComboBox; CEdit m_Edit; CScrollBar m_ScrollBar; In the dialog’s OnInitDialog, attach the wrapper instances to the dialog controls. Also initialize the contents of the listboxes and combobox, and set the range for the scroll-bar. m_Static.Attach(GetDlgItem(IDC_STATIC_DEMO)); 186 WTL Developer’s Guide m_Button.Attach(GetDlgItem(IDC_BUTTON_DEMO)); m_ListBox.Attach(GetDlgItem(IDC_LIST_DEMO)); m_ListBox.AddString(TEXT("One")); m_ListBox.AddString(TEXT("Two")); m_ListBox.AddString(TEXT("Three")); m_ComboBox.Attach(GetDlgItem(IDC_COMBO_DEMO)); m_ComboBox.AddString(TEXT("One")); m_ComboBox.AddString(TEXT("Two")); m_ComboBox.AddString(TEXT("Three")); m_ComboBox.SetCurSel(2); m_Edit.Attach(GetDlgItem(IDC_EDIT_DEMO)); m_Edit.SetWindowText(TEXT("Some text")); m_ScrollBar.Attach(GetDlgItem(IDC_SCROLLBAR_DEMO)); m_ScrollBar.SetScrollRange(0,100); Finally add a push-button called ChangeAttr and a corresponding member function which exercises the wrappers. LRESULT OnClickedChangeAttr(WORD wNotifyCode, WORD wID, HWND hWndCtl, BOOL& bHandled){ HICON MyIcon; MyIcon = (HICON) LoadImage(_Module.GetModuleInstance(), MAKEINTRESOURCE(ID_MY_ICON), IMAGE_ICON, 0, 0, LR_DEFAULTCOLOR); m_Static.SetIcon(MyIcon); m_Button.SetCheck(1); m_ListBox.SetCurSel(1); m_ComboBox.SetEditSel(0,1); m_Edit.AppendText(TEXT("- Extra text")); m_ScrollBar.SetScrollPos(50); return 0; } Initializing a Combo-Box The ResourceView in VC++’s DevStudio allows entries in a combobox to be added to the resource template at design time. When you add a combobox, and bring up its properties page, then the “Data” tab is used to add entries. When you run an MFC app and display dialogs with combo-boxes who have had such entries defined, they are displayed with the combo-boxes populated as expected. With WTL, the combo-boxes get displayed, but are empty. What is happening? Information about the combobox (such as its control id and location on the dialog, but excluding its data entries) is stored as part of a resource, of type DIALOG, in the resource file. The Win32 dialog APIs understand this and load it directly before rendering the dialog. The information about the entries in a combobox is stored as a separate resource, of type DLGINIT. The Win32 dialog APIs do not understand DLGINIT. Other code in higher-level libraries or the application itself has to process DLGINIT resources. 187 WTL Developer’s Guide MFC provides functionality as part of its dialog classes to cater for DLGINIT. WTL does not. The reuse header file (rtl.h) that we have written contains a WTL-friendly function called RUiDlgInit, which loads the DLGINIT data. It is modeled on the equivalent MFC functionality. RUiDlgInit(IDD, m_hWnd); You need to pass it in the IDD of the dialog and the HWND of the existing dialog window. Typically it is called from OnInitDialog. RUiDlgInit just runs through the fields in the DLGINIT and sends a CB_ADDSTRING to the combo-box for each entry. Sample : FirstCalc The FirstCalc sample shows the use of the combo-box data entries. It is a calculator, which accepts two numbers and lets the user select an operator (‘+’, ‘-‘, ‘/’, ‘*’) from a combo-box. Its dialog is laid out as: In the resource file this is stored as: IDD_MAINDLG DIALOG DISCARDABLE 0, 0, 179, 62 STYLE WS_MINIMIZEBOX | WS_CAPTION | WS_SYSMENU CAPTION "FirstCalc" FONT 8, "MS Sans Serif" BEGIN DEFPUSHBUTTON "OK",IDOK,65,40,50,14 PUSHBUTTON "=",IDC_EQUALS_BTN,110,15,25,15 EDITTEXT IDC_RESULT_EDIT,140,15,35,15,ES_AUTOHSCROLL | ES_READONLY EDITTEXT IDC_FIRST_EDIT,5,15,30,15,ES_AUTOHSCROLL | ES_NUMBER COMBOBOX IDC_OP_COMBO,40,15,30,36,CBS_DROPDOWNLIST | WS_VSCROLL | WS_TABSTOP EDITTEXT IDC_SECOND_EDIT,75,15,30,15,ES_AUTOHSCROLL | ES_NUMBER LTEXT "Number:",IDC_STATIC,5,5,30,10 LTEXT "Operator:",IDC_STATIC,40,5,35,10 LTEXT "Number:",IDC_STATIC,75,5,30,10 END The combobox entries are set using the ResourceView (for this demo they are just one character each in length, but of course they can be any length you like): 188 WTL Developer’s Guide At the end of each entry do not press - this will dismiss the properties dialog – rather press and simultaneously, which will put the caret on the next line. This information is stored in the resource file in a DLGINIT: IDD_MAINDLG DLGINIT BEGIN IDC_OP_COMBO, 0x403, 0x002b, IDC_OP_COMBO, 0x403, 0x002d, IDC_OP_COMBO, 0x403, 0x002a, IDC_OP_COMBO, 0x403, 0x002f, 0 END 2, 0 2, 0 2, 0 2, 0 Developers new to DevStudio sometimes have problems sizing a combo-box. There are two sizes which need to be set: the (horizontal) size of the combo box when it is not in a drop-down state, and the (vertical) size when it is. To set the first, click the mouse in the combo-box outside of the drop-down arrow – to second the second click inside the arrow. To load this data, add a call to RUiDlgInit to the OnInitDialog method – also set the current selection to 0 (the first entry in the combo-box list): LRESULT OnInitDialog(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/){ . . . RUiDlgInit(IDD, m_hWnd); m_ctlOpCombo.Attach(GetDlgItem(IDC_OP_COMBO)); m_ctlOpCombo.SetCurSel(0); return TRUE; } The handler carries out the calculation is as follows: 189 WTL Developer’s Guide LRESULT OnClickedEquals_btn(WORD wNotifyCode, WORD wID, HWND hWndCtl, BOOL& bHandled){ CString strError; // get current values if ((!DoDataExchange(TRUE, IDC_FIRST_EDIT)) || (!DoDataExchange(TRUE, IDC_SECOND_EDIT))){ strError.LoadString(IDS_ERR_EXCHANGE); MessageBox(strError); return 0; } int sel = m_ctlOpCombo.GetCurSel(); switch (sel){ case 0: m_nResultNum = m_nFirstNum + m_nSecondNum; break; case 1: m_nResultNum = m_nFirstNum - m_nSecondNum; break; case 2: m_nResultNum = m_nFirstNum * m_nSecondNum; break; case 3: m_nResultNum = m_nFirstNum / m_nSecondNum; break; default: ATLASSERT(FALSE); // should not get here! } // Put result in result window if (!DoDataExchange(FALSE, IDC_RESULT_EDIT)){ strError.LoadString(IDS_ERR_EXCHANGE); MessageBox(strError); } return 0; } WTL Wrappers for Common Controls WTL provides the following classes for controls and helper classes: CListViewCtrl CTreeViewCtrl CTreeItem CHeaderCtrl CImageList CTreeViewCtrlEx CToolBarCtrl CToolInfo CProgressBarCtrl CStatusBarCtrl CToolTip CHotKeyCtrl CTabCtrl CTrackBarCtrl CAnimate CFlatScrollBar CUpDownCtrl CRichEdit CDragListBox CComboBoxEx CRichEditComamnds CDragListNotifyImpl CDataTime CIPAddressCtrl CReBarCtrl CMonthCalendar CPagerCtrl 190 WTL Developer’s Guide CFlatScrollBarImpl 191 CCustomDraw WTL Developer’s Guide Chapter 7 Graphical Primitives Objectives The objectives of this chapter are to: • Explain when we need graphical primitives • Describe the CSize/CPoint/CRect helper classes • Explain device contexts • Review graphical settings, such as pens, brushes and fonts Overview All modern operating systems provide windowed graphics, which operate in a similar manner. The screen is divided into a hierarchical set of windows, which can be controlled by multiple applications. An application can set up the co-ordinates and mapping mode for its windows, specific drawing attributes and then use a variety of primitives to render into the window. On Windows family of operating systems, the graphics technology is known as GDI. The most important type of GDI object is a device context. One can apply methods such as LineTo or Rectangle or TextOut to a device context to effect rendering. The output from device context rendering methods can be configured using other types of GDI objects such as pens, brushes, fonts and bitmaps. WTL provides a full range of classes and templates for graphical primitives that cover all aspects of the Win32 graphics facilities. The WTL support is a very thin layer above the Win32 GDI API, so it is highly efficient and also it is easy to mix WTL and direct Win32 GDI calls. Rendering Graphics In Windows Win32 graphics is based on the concept of a device context, which provides access to drawing primitives and stores drawing attributes and modes. Primitives consists of lines, rectangles, text, bitmaps and anything else which is to appear as pixels in a window. How these primitives are rendered is controlled by a set of attributes, some of which are simple values, such as drawing mode (e.g. R2_COPYPEN for plain or R2_NOT for inverting), and some of which are other objects, such as pens, brushes and fonts. These attributes can be assigned to the device context, and when rendering occurs the graphical primitives use the current attributes. Windows GDI provides a number of object types. Each object is identified by a handle, has construction and destruction functions and offers a variety of other graphical output 192 WTL Developer’s Guide and settings functions. The idea of providing wrappers as C++ templates for each GDI object types is appealing – and this is exactly what WTL does for all of them. The Windows GDI object types are: • Device Context (DC) – Rendering and current attribute selections • Font – Font to use for text • Pen – How to render edge of primitives • Brush – How to fill internals of primitives • Region – Collection of one or more rectangles, polygons, or ellipses, which together describes an area of the window (need not be contiguous) • Palette – A collection of color indices and values • Bitmap – A two-dimensional collection of pixels A DC is used for rendering inside windows. A DC is used to setup rendering attributes, either using discrete attribute settings (e.g. current-pen position) or attribute object types, such as a pen object, which color, width and dashing styles). A DC is used to store graphical modes, such as whether XOR is used for drawing or the co-ordinate management. Hence the DC is the main object and one can consider the others as helper objects Where to Render? Display device contexts can either cover the client area of window (the normal case), or the entire window including the window frame& title bar (rarely used). Typically, Win32 applications render graphics in the client-area of a window. By “client-area”, we mean the area inside the window frame, below the window title bar and above the statusbar, and inside scrollbars and command-bars if present. A display device context used for client-area rendering is known as a client DC. A Win32 function called GetDC will return a client DC. Normally all these other parts of a window, such as the frame and title bar, are rendered by the operating system itself, and applications rarely needed to render here. It is possible to retrieve a DC, known as a window DC, capable of rendering in the entire window. A Win32 function called GetWindowDC will return a window DC. However, note that the Microsoft Platform SDK documentation for GetWindowDC has this line: “Painting in non-client areas of any window is not recommended.” The Windows standard and common controls are normally responsible for rendering themselves. One exception is the concept of owner-draw controls, in which case your code is responsible for refreshing controls as needed. It is possible to use the Win32 graphical primitives to render anywhere in any window, but this is not frequently done. Two WTL templates that could help are CBitmapButton and COwnerDraw. 193 WTL Developer’s Guide Types of Device Context Think of device context as providing access to the destination for drawing. Very often, device contexts represent on-screen windows (display device context), but the can also represent printers, metafiles and off-screen memory (memory device contexts). Hence the types of device contexts are: • ClientDC • WindowDC • PaintDC • EnhancedMetafileDC • PrintDC A memory device context (for off-screen bitmaps) operate like the generic device context, so WTL need not provide anything special in this case. A memory device context is constructed by calling the Win32 function CreateCompatibleDC. During refreshing, the OS sending a WM_PAINT message to the window. To signify that the update has actually occurred, the application code must call BeginPaint and EndPaint. To facilitate this, WTL provides which is known as a PaintDC, which is a normal DC, but in addition calls BeginPaint in its constructor and EndPaint in its destructor. What to Render? The application needs to keep track of everything that appears in the client area of each window. When a window is hidden and later displayed, it is the application’s duty to redraw everything. WTL does not provide any support in this area. Typically the application maintains a hierarchy of one or more collection classes, such as arrays or linked lists, and during refreshes, it cycles through the data (in a painter’s algorithm, back to front) and rendering each data item. Each item has a bounding box, which identifies its perimeter, and when an update occurs often only a subsection of the window needs to be refreshed. Only items whose bounding box intersects the refresh area need be redrawn. When the application is about to shutdown, there is a need to persist such data to the hard disk. Again, WTL does not provide assistance in this area. Modern applications normally use XML here. When to Render? There are two schools of thought about when to do rendering. One opinion is that it should all be centralized inside the handling of WM_PAINT. This has the advantage of doing it in one place, but the disadvantage of redrawing everything within the region, which needs to be refreshed. The function will be needed anyway to refresh the screen after it has been de-iconized or brought in front of another window which was obscuring it, or re-sized or scrolled. However, in this first approach it can also be used even if the window is already the foreground window. The idea is when adding an item 194 WTL Developer’s Guide to appear on screen, it is appended to the internal hierarchical data structures, and then the area on screen it will cover is invalidated. This will result in a WM_PAINT message being sent to the window. Your windowing code will respond as usual by erasing the background and walking through the hierarchy redrawing everything that appears in the window. As the new object will be last in the hierarchy, it will be last to be drawn, so will appear on top of everything else. You might think this is an awful lot of work just to draw one object, but for very many applications it happens so fast the end-user will not notice. However, for applications will very complex output, yes, it can slow things down and there can be screen flicker. For such applications, the use of an off-screen bitmap can be helpful. All the graphical primitives can be rendered off-screen, and when finished, the off-screen bitmap can be bitblt'ed on-screen. The second school of thought on how to do rendering is to permit rendering as needed, and still have a complete WM_PAINT implementation. When adding an object, it can be done inside the handler for mouse button up. Some people disapprove of this technique, as it is spreading drawing code in multiple locations. However, most applications manage items, which appear on screen as objects, and each has a draw method, so the bulk of the drawing code in inside such methods. What appears inside your handler for WM_PAINT or WM_LBUTTONUP is just calls to such methods. Also note that for rubber banding, this will normally have to be done in mouse move handling, so this second approach will still need to be catered for. Whether you choose the first or the second approach usually depends on how complex the scene will be and whether the appearance of the new item will change existing items. For 3D graphics with shadowing etc., adding an item definitely will change other items, so it should be done in WM_PAINT. For an application drawing an item onto a window with 500,000 existing items, and if the item will just appear on top of these, then it would make sense to render it directly inside your WM_LBUTTONUP handler. WM_PAINT messages are considered of a low-priority. The reason for this is that if additional messages are in the message queue, it is likely that they will change what needs to be rendered, and hence it is more efficient to leave the painting until all the other messages are handled, and then update once. When the OS is ready to present a WM_PAINT message, it gathers all the outstanding messages and amalgamates their invalid regions, and presents a single WM_PAINT with the amalgamated region to the application code. Device Context Types There are two popular approaches to DC management. One is to use the DC pool and the other is to maintain a private DC per window. The OS maintains a pool of DCs and lends one to an application when it calls GetDC, and it is returned when ReleaseDC is called. What is provided is known as a common DC, as it is used by multiple windows (at different times). Using DCs from the pool is efficient and very often the Get/Release occurs inside the same function. In the distant past (e.g. Win3.1) there was a tiny limit to the number of DCs in the pool, but now it is only limited by system memory availability, and if memory is full you are going to have a lot more worries than simply refreshing the screen. 195 WTL Developer’s Guide It is possible to maintain a separate DC for a particular window. Passing CS_OWNDC among the styles to the Win32 API RegisterClass does this. A third approach, known as a class DC, is possible but not recommended. This approach uses one DC per class. Every window of that class will share the same DC. This can cause serious problems in a multi-threaded application. DCs and Threads A DC may only be used in one thread at a time. If an attempt is made to use it in multiple threads in parallel, one of the calls will fail. Another problem is that even though a DC might not be used in different threads at exactly the same time, it could happen that one thread has set up the DC with pens and brushes etc. and renders with it, but another thread sets it up with different attributes, and thus the first thread will render un-expectedly. If you wish to share DCs across threads, then you are responsible for ensuring each DC is used in only one thread at a time and you must use synchronization constructs such as mutexes or critical sections. GDI itself does not do this for you. Most applications do not share DCs across threads because of this. Hence most applications do not use class DCs. Rather they use common device DCs, or if they can guarantee that all rendering to a particular window will be from the same thread, a private DC. It is noted that with the Windows Template Library, there is an option to use multiple threads with the SDI approach, and critically these use one thread per window, so using private device contexts will work. However, the default setting is that a common device context will be used. To use a private DC, add CD_OWNDC to the WNDCLASS passed to RegisterClass. In WTL/ATL, adding a custom implementation of GetWndClassInfo as follows can do this: static CWndClassInfo& GetWndClassInfo() { static CWndClassInfo wc = { { sizeof(WNDCLASSEX), CS_HREDRAW | CS_VREDRAW|\ CS_DBLCLKS | CS_OWNDC, StartWindowProc, 0, 0, NULL, NULL, NULL, (HBRUSH)(COLOR_WINDOW + 1), NULL, TEXT("myWindow"), NULL }, NULL, NULL, IDC_ARROW, TRUE, 0, _T("") }; return wc; } Selecting Attribute Objects GDI attribute objects are selected into a device context using SelectObject. When finished, the object has to be de-selected by calling SelectObject again with a default value or any other valid object handle. It is deleted by calling DeleteObject. Note that an object that is selected into a DC may not be deleted. If you are using a common DC, every time you retrieve it its attributes will be set to default values. You will have to set it up with settings suitable for your application. 196 WTL Developer’s Guide With a private DC, the first time you retrieve a DC it will have default values, but all subsequent retrieves will return the exact same DC and the attributes. Minimize DC Alterations For high-performance graphics applications, it is important to minimize the amount of changing of DC attributes. If you are drawing ten elements on-screen it makes no difference. If you are drawing a million elements, it certainly does. For each element, it is ill advised to select a new pen, new brush, new font etc., as this is a waste of CPU resources. Instead, only as needed should these be selected. Hence we need to minimize the amount of DC alterations. One idea is to use a private DC. It is likely that application default attributes can be different from system default, and with a private DC we could set the default once. Also, certain attributes are likely not to change, and they only need to be set in the private DC once. The use of the concept of a style is another way of encouraging users to share attributes among multiple elements. One idea is to have a bitmask identifying desired attributes – and default attributes – only where bits are different, is there a need to select objects into DC. One final possibility is that you could use multiple DCs per window. If you were managing a small number of graphical styles, you could create a DC for each style, and never release it until the application was shut down. Therefore your redraw code would never have to change a DC, thus speeding it up considerably. The number of times a style is usually changed is small compare to the number of times it is used (e.g. one per each element based on it per refresh). Getting Started with WTL Graphics Now we cam move on to WTL graphics. WTL Header Files When using WTL’s graphics features you will have to include the WTL header file atlgdi.h. If you plan to also use the helper classes such as CSize, CRect and CPoint, then you will have to include atlmisc.h. The entries in atlgdi.h do not depend on the entries in atlmisc.h, and vice versa, so it is quite possible to include one but not the other header file. Atlgdi.h is included in atlapp.h, which in turn is included in every WTL project by the WTL AppWizard, so unless you change something you can be sure it is always included. Sample : First Line To draw your first line in WTL, generate a sample project, say of type SDI Application. In the view.h file, there will be an implementation of the class CView. It will have a default OnPaint method with an 197 WTL Developer’s Guide instantiation of a device context based on the view window. Add a call to CPaint::LineTo method, and build and run the application. LRESULT OnPaint(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/){ CPaintDC dc(m_hWnd); dc.LineTo(200, 200); // Add this line only return 0; } The view window will contain a line, from the upper-left corner, down to the dimensions you have specified as parameters to LineTo. As you see, using WTL for graphics is very simple, especially if you have a prior knowledge of Win32 graphics programming. All WTL provides is an efficient set of wrappers for Win32 graphical constructs, and in some cases adds functionality, which can be quite useful and timesaving. Using WTL for graphics also makes your code more easily tie in with other parts of a WTL application. WTL Helper Classes – CSize, CPoint and CRect WTL provides three helper classes to manage size, rectangle and point information. These classes have no data members. Instead then inherit from Win32 data types, which is where the data is stored. CSize/CPoint/CRect are not used in any WTL GDI templates – so for example, to draw a rectangle, you may either pass in four integer parameters representing the boundary, or a variable of a Win32 data type LPCRECT, which points to the Win32 RECT structure. typedef struct tagRECT { // Win32 LONG left; LONG top; LONG right; LONG bottom; } RECT; The result of this is that as the WTL GDI functionality is based on the same Win32 data types, there is no direct dependency between the WTL helper classes and the WTL GDI templates/classes. As the WTL helper classes derive from the Win32 structures, then are interchangeable. You may either continue to use the Win32 structures, or the WTL helper classes. CSize CSize manages rectangular dimensions. It stores the width and height values as LONGs. WTL’s CSize derives from Win32’s SIZE structure. CSize provides the following constructors: CSize(); CSize(int initCX, int initCY); CSize(SIZE initSize); CSize(POINT initPt); 198 WTL Developer’s Guide CSize(DWORD dwSize); CSize provides these operations: BOOL BOOL void void void operator==(SIZE operator!=(SIZE operator+=(SIZE operator-=(SIZE SetSize(int CX, size) const; size) const; size); size); int CY); CSize has these operators which return CSize values: CSize operator+(SIZE size) const; CSize operator-(SIZE size) const; CSize operator-() const; CSize has these operators which return CPoint values: CPoint operator+(POINT point) const; CPoint operator-(POINT point) const; Finally, CSize has these operators which return CRect values: CRect operator+(const RECT* lpRect) const; CRect operator-(const RECT* lpRect) const; CPoint CPoint manages a point. It stores the x and y co-ordinates as LONGs. WTL’s CPoint derives from Win32’s POINT structure. CPoint provides the following constructors: CPoint(); CPoint(int initX, int initY); CPoint(POINT initPt); CPoint(SIZE initSize); CPoint(DWORD dwPoint); CPoint provides these operations: void void void BOOL BOOL void void void void void Offset(int xOffset, int yOffset); Offset(POINT point); Offset(SIZE size); operator==(POINT point) const; operator!=(POINT point) const; operator+=(SIZE size); operator-=(SIZE size); operator+=(POINT point); operator-=(POINT point); SetPoint(int X, int Y); CPoint has these operators which return CPoint values: CPoint CPoint CPoint CPoint operator+(SIZE size) const; operator-(SIZE size) const; operator-() const; operator+(POINT point) const; CPoint has this operator which return a CSize value: 199 WTL Developer’s Guide CSize operator-(POINT point) const; CPoint has these operators which return CRect values: CRect operator+(const RECT* lpRect) const; CRect operator-(const RECT* lpRect) const; CRect CRect manages a rectangle. It stores the upper-left and lower-right co-ordinates as LONGs WTL’s CRect derives from Win32’s RECT structure. CRect provides the following constructors: CRect(); CRect(int l, int t, int r, int b); CRect(const RECT& srcRect); CRect(LPCRECT lpSrcRect); CRect(POINT point, SIZE size); CRect(POINT topLeft, POINT bottomRight); Note that the CRect constructor, which takes four ints, expects left, top, right and bottom parameters, and not left, top, width and height. It one assumed the latter and create a CRect with say (10,20,10,200), it would occupy no area of screen. CRect provides these additional methods to access the stored data: int Width() const; int Height() const; CSize Size() const; CPoint& TopLeft(); CPoint& BottomRight(); const CPoint& TopLeft() const; const CPoint& BottomRight() const; CPoint CenterPoint() const; CRect provides these conversion methods: operator LPRECT(); operator LPCRECT() const; CRect provides these querying methods: BOOL BOOL BOOL BOOL IsRectEmpty() const; IsRectNull() const; PtInRect(POINT point) const; EqualRect(LPCRECT lpRect) const; CRect provides these editing methods: void void void void SetRect(int x1, int y1, int x2, int y2); SetRect(POINT topLeft, POINT bottomRight); SetRectEmpty(); CopyRect(LPCRECT lpSrcRect); CRect provides these methods to inflate, deflate and offset the rectangle: void InflateRect(int x, int y); void InflateRect(SIZE size); void InflateRect(LPCRECT lpRect); 200 WTL Developer’s Guide void void void void void void void void void InflateRect(int l, int t, int r, int b); DeflateRect(int x, int y); DeflateRect(SIZE size); DeflateRect(LPCRECT lpRect); DeflateRect(int l, int t, int r, int b); OffsetRect(int x, int y); OffsetRect(SIZE size); OffsetRect(POINT point); NormalizeRect(); CRect provides these rectangle positioning methods: void void void void MoveToY(int y); MoveToX(int x); MoveToXY(int x, int y); MoveToXY(POINT point); CRect provides these Boolean operator methods: BOOL IntersectRect(LPCRECT lpRect1, LPCRECT lpRect2); BOOL UnionRect(LPCRECT lpRect1, LPCRECT lpRect2); BOOL SubtractRect(LPCRECT lpRectSrc1, LPCRECT lpRectSrc2); CRect provides these overloaded operators: void BOOL BOOL void void void void void void void void operator=(const RECT& srcRect); operator==(const RECT& rect) const; operator!=(const RECT& rect) const; operator+=(POINT point); operator+=(SIZE size); operator+=(LPCRECT lpRect); operator-=(POINT point); operator-=(SIZE size); operator-=(LPCRECT lpRect); operator&=(const RECT& rect); operator|=(const RECT& rect); CRect provides these overloaded operators which return a CRect: CRect CRect CRect CRect CRect CRect CRect CRect CRect operator+(POINT point) const; operator-(POINT point) const; operator+(LPCRECT lpRect) const; operator+(SIZE size) const; operator-(SIZE size) const; operator-(LPCRECT lpRect) const; operator&(const RECT& rect2) const; operator|(const RECT& rect2) const; MulDiv(int nMultiplier, int nDivisor) const; Sample : HelperClasses The HelperClasses sample shows a simple use of these helper classes. It is a simple SDI project. The header file “atlmisc.h” is added to atdafx.h. This method is added to show CPoint, CSize and CRect in action. void HelperDemo(){ CPoint pt1(100,100); 201 WTL Developer’s Guide CPoint pt2(300,300); CRect rect(pt1, pt2); CSize sz; sz.SetSize(100,100); pt1 += sz; BOOL b = rect.PtInRect(pt2); } GDI Objects and Handle Management GDI objects are a totally separate concept to Win32 kernel objects, C++ objects and COM objects. They are internal OS-managed pieces of memory, which are used to render graphics. Applications refer to them via handles. The OS frees GDI objects automatically when the process, which uses them, dies. However, it is considered good practice to release them when no longer needed. A GDI object gets created with a call to Win32 APIs ::Create and get freed with a call to the Win32 API DeleteObject. Consider a GDI handle as an opaque pointer to the memory. WTL provides templates to all the GDI objects to manage their lifetime and to expose type-safe wrappers to access all their functionality. Who owns the handle One issue that needs to be considered is how the GDI object handle is managed. Is the GDI object created inside the WTL template, or outside? Who looks after object destruction? How is access to the handle provided through the WTL template, so that direct Win32 calls can be made if needed? Each WTL template for a GDI object works through this set of issues in the same way. Here we will take the example of a WTL’s CPenT template. The same applies to CBrushT, CFontT, CPaletteT, CBitmapT, CPaletteT, CRgnT, CEnhMetaFileT and CDCT and all its derivatives (CPaintDCT, CClientDC, CWindowDC and CEnhMetaFileDC). There is one Boolean template parameter, which specifies how handle management is to be done. Each template has two typedefs, which specify that the class will be used to manage its own handle, or will use an existing handle created elsewhere. template class CPenT { . . . }; typedef CPenT CPenHandle; typedef CPenT CPen; The constructor can take in a handle as an optional parameter (it defaults to NULL). CPenT(HPEN hPen = NULL) : m_hPen(hPen) { } The destructor will destroy the object represented by the handle only WTL is in change of handle management. ~CPenT(){ if(t_bManaged && m_hPen != NULL) 202 WTL Developer’s Guide DeleteObject(); } An assignment operator is provided. CPenT& operator=(HPEN hPen){ m_hPen = hPen; return *this; } Methods are provided to attach and detach the handle. void Attach(HPEN hPen){m_hPen = hPen;} HPEN Detach(){ HPEN hPen = m_hPen; m_hPen = NULL; return hPen; } Methods are provided to access the internal handles and determine if it is NULL. operator HPEN() const { return m_hPen; } bool IsNull() const { return (m_hPen == NULL); } A method is provided to delete the object. This is called from the destructor and also may be called directly. BOOL DeleteObject(){ ATLASSERT(m_hPen != NULL); BOOL bRet = ::DeleteObject(m_hPen); if(bRet) m_hPen = NULL; return bRet; } Parameters for WTL GDI Member Functions WTL’s GDI is a thin wrapper around Win32 GDI functionality. The design decision was taken to use standard Win32 data types for parameters into most WTL template methods. Hence one will see a HPEN rather that a &CPen being used. One exception is CString, which is supported in some templates for string inputs. Device Context Templates WTL provides a number of templates for device contexts. CDC is the main one and the other inherit from it. CDC is a big template – it is by far the biggest within WTL. It offers about 240 methods, which mostly wrap the familiar Win32 drawing functions such as LineTo and TextOut. Their implementations simply check that the DC handle is valid and then execute the Win32 function. For example, WTL’s CDC::LineTo is implemented as: BOOL LineTo(int x, int y){ ATLASSERT(m_hDC != NULL); return ::LineTo(m_hDC, x, y); } 203 WTL Developer’s Guide There are a small number of methods within CDC, which contain more substantial functionality. We will shortly examine all of CDC’s methods. The other WTL templates for device contexts are small – they essentially perform a few extra duties and use CDC for the entire graphics rendering. CDC CPaintDC CClientDC CEnhMetaFileDC CPrintDC CWindowDC CPaintDC CPaintDC is a class that derives from CDC. It is aimed at provided refresh functionality – so it manages a Win32 PAINTSTRUCT. In its constructor it calls Win32’s BeginPaint and in its destructor it calls Win32’s EndPaint. Apart from that, it is exactly like a CDC. It should be used inside an application’s handling of WM_PAINT messages. It should not be used elsewhere. CClientDC CClientDC derives from CDC and it is a DC based a client area of a window. Therefore it is ideal when rendering into a window – especially the client-area. Hence the name – CClientDC. In its constructor it calls Win32’s GetDC and in its destructor it calls Win32’s ReleaseDC. Apart from that, it is exactly like a CDC. CClientDC is used when an application wishes to render outside of refresh handling. Examples would be inside button down, mouse move and button up handlers. CWindowDC CWindowDC derives from CDC and in its constructor it calls Win32’s GetWindowDC and in its destructor calls ReleaseDC. Apart from that, it is exactly like a CDC. The CWindowDC template can be used to render in all parts of a window – including the non-client areas such as the title bar. It is rarely used, as most applications render only into the client-area (using CClientDC) do not need to render in the non-client area of a window. The DC retuned is a common device context – regardless of whether CS_OWNDC is set or not. CEnhMetaFileDC CEnhMetaFileDC is used when creating enhanced meta files. We will discuss this template further when covering enhanced metafiles. 204 WTL Developer’s Guide CPrintDC CPrintDC is used when printing. We will discuss this template further in the chapter on printing. Managing Attributes Objects CDC provides a number of methods to work with attribute objects. To retrieve the current attribute value, use: CPenHandle GetCurrentPen() const CBrushHandle GetCurrentBrush() const CPaletteHandle GetCurrentPalette() const CFontHandle GetCurrentFont() const CBitmapHandle GetCurrentBitmap() const To set and get the origin of the brush in use: BOOL GetBrushOrg(LPPOINT lpPoint) const BOOL SetBrushOrg(int x, int y, LPPOINT lpPoint = NULL) BOOL SetBrushOrg(POINT point, LPPOINT lpPointRet = NULL) To list the pens and brushes available use: int EnumObjects(int nObjectType, int (CALLBACK* lpfn)(LPVOID, LPARAM), LPARAM lpData) When selecting attribute objects, use: HPEN SelectPen(HPEN hPen) HBRUSH SelectBrush(HBRUSH hBrush) HFONT SelectFont(HFONT hFont) HBITMAP SelectBitmap(HBITMAP hBitmap) int SelectRgn(HRGN hRgn) The OS provides a number of standard object types, known as stock objects. When selecting stock objects use: HPEN SelectStockPen(int nPen) HBRUSH SelectStockBrush(int nBrush) HFONT SelectStockFont(int nFont) HPALETTE SelectStockPalette(int nPalette, BOOL bForceBackground) Stock pens may be one of: WHITE_PEN, BLACK_PEN or (Win98/2000 only) DC_ PEN. Stock brushes may be WHITE_BRUSH, HOLLOW_BRUSH or (Win98/2000 only) DC_BRUSH. Stock fonts may be one of the range from OEM_FIXED_FONT to SYSTEM_FIXED_FONT or DEFAULT_GUI_FONT. The stock palette must be DEFAULT_PALETTE. Lines and Pens WTL’s CDC offers these member functions to render lines, arcs, polylines and bezier curves. BOOL GetCurrentPosition(LPPOINT lpPoint) const; 205 WTL Developer’s Guide BOOL BOOL BOOL BOOL BOOL MoveTo(int x, int y, LPPOINT lpPoint = NULL); MoveTo(POINT point, LPPOINT lpPointRet = NULL); LineTo(int x, int y); LineTo(POINT point); Arc(int x1, int y1, int x2, int y2, int x3, int y3, int x4, int y4); BOOL Arc(LPCRECT lpRect, POINT ptStart, POINT ptEnd); BOOL Polyline(LPPOINT lpPoints, int nCount); BOOL AngleArc(int x, int y, int nRadius, float fStartAngle, float fSweepAngle); BOOL ArcTo(int x1, int y1, int x2, int y2, int x3, int y3, int x4, int y4); BOOL ArcTo(LPCRECT lpRect, POINT ptStart, POINT ptEnd); int GetArcDirection() const; int SetArcDirection(int nArcDirection); BOOL PolyDraw(const POINT* lpPoints, const BYTE* lpTypes, int nCount); BOOL PolylineTo(const POINT* lpPoints, int nCount); BOOL PolyPolyline(const POINT* lpPoints, const DWORD* lpPolyPoints, int nCount); BOOL PolyBezier(const POINT* lpPoints, int nCount); BOOL PolyBezierTo(const POINT* lpPoints, int nCount); Some functions use and update the current position and some don’t. For some drawing primitives, two versions of the function are provided – one ending in “To” and one not. The “To” versions (e.g. ArcTo, PolylineTo, PolyBezierTo) use the current point as the starting point of the rendering and when finished update the current position. The alternatives (Arc, Polyline, PolyBezier) ignore the current point, by starting the rendering at the first point in the point list parameter and not updating the current point when finished. Some pairs of member functions provide access to the same underlying Win32 API, but support multiple data types – e.g. there is a MoveTo which takes integer x and y, and another MoveTo which takes POINTs. The PolyDraw member function is not supported on Windows 9x. Check out Microsoft's KB article id: Q135059 to see how to provide a custom implementation for Win9x. CPenT A pen determines the line-rendering attributes, such as line color, dimensions and dashing style. The CPenT template maintains a single data member, of type HPEN, which is a GDI handle to a pen object. There are two types of pens: cosmetic and geometric. Cosmetic means that the pen’s width is fixed and does not change with scaling, whereas with geometric pens the width can be scaled. 206 WTL Developer’s Guide Three Create methods are provided: // creates a cosmetic pen HPEN CreatePen(int nPenStyle, int nWidth, COLORREF crColor); //creates a geometric pen HPEN CreatePen(int nPenStyle, int nWidth, const LOGBRUSH* pLogBrush, int nStyleCount = 0, const DWORD* lpStyle = NULL); // Creates cosmetic pen HPEN CreatePenIndirect(LPLOGPEN lpLogPen); CPenT also provides these methods to access pen attributes: int GetLogPen(LOGPEN* pLogPen) const; bool GetLogPen(LOGPEN& LogPen) const; int GetExtLogPen(EXTLOGPEN* pLogPen) const; bool GetExtLogPen(EXTLOGPEN& ExtLogPen) const; These are wrappers for the Win32 API GetObject. They return information about the currently selected pen. Note that the Width field of the LOGPEN structure is actually a POINT, and only the x field is used. Sample : LinesAndPens The LinesAndPens sample shows how to use the CDCT and CPenT templates. It is an SDI application. Menu-items have been added for each important line member function of CDC and each CPen member. We wish to use a private DC, so that DC settings will be retained across multiple GetDC calls. To enable this, in the view class, one line was commented out: //DECLARE_WND_CLASS(NULL) and this implementation of CWndClassInfo added: static CWndClassInfo& GetWndClassInfo() { static CWndClassInfo wc = { { sizeof(WNDCLASSEX), CS_HREDRAW | CS_VREDRAW|\ CS_DBLCLKS | CS_OWNDC, StartWindowProc, 0,0,NULL,NULL,NULL,(HBRUSH)(COLOR_WINDOW+1), NULL, TEXT("myWindow"), NULL }, NULL, NULL, IDC_ARROW, TRUE, 0, _T("") }; return wc; } The menu handlers are in the CMainFrame class, but the rendering is to be done in the view class, which is identified inside the CMainFrame class by m_hWndClient. Hence, when accessing the DC we use the line: CClientDC dc(m_hWndClient); There are three types of handlers. One type calls rendering methods, such as LineTo, Arc, or PolyBezier. Here is one sample handler for PolyDraw: LRESULT OnPloyDraw(WORD /*wNotifyCode*/, WORD /*wID*/, 207 WTL Developer’s Guide HWND /*hWndCtl*/, BOOL& /*bHandled*/) { #define NUM_BEZIER_POINTS 4 POINT ptList1[NUM_BEZIER_POINTS]={{20,20}, {250, 900}, {450,145}, {600,320}}; BYTE typeList1[NUM_BEZIER_POINTS]={PT_MOVETO, PT_BEZIERTO, PT_BEZIERTO, PT_BEZIERTO}; dc.PolyDraw(ptList1, typeList1, NUM_BEZIER_POINTS); return 0; } Another type of handler is used to retrieve information from the device context. In this project, the data is simply sent to the output window. LRESULT OnGetLogPen(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ CClientDC dc(m_hWndClient); CPenHandle penhandle = dc.GetCurrentPen(); LOGPEN logpen; penhandle.GetLogPen(logpen); CString str; str.Format("Output from GetLogPen: style = %d, \ width = %d, color = (%d, %d, %d)\n", logpen.lopnStyle, logpen.lopnWidth.x, GetRValue(logpen.lopnColor), GetGValue(logpen.lopnColor), GetBValue(logpen.lopnColor)); OutputDebugString(str); return 0; } The final type of handler is used to create a new pen. LRESULT OnCreatePen(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ CClientDC dc(m_hWndClient); // check if old pen existed if (!m_pen.IsNull()){ // Before deleting pen, should de-select it from DC dc.SelectStockPen(BLACK_PEN); m_pen.DeleteObject(); } // Generate logical pen m_pen.CreatePen(PS_DASHDOT, 5, RGB(150,50,100)); dc.SelectPen(m_pen); return 0; } Note that the m_Pen parameter is a member variable of the CMainFrame class. If it were a local variable inside OnCreatePen, upon exiting the function the destructor for CPen would be called, and it calls DeleteObject, which is definitely not what is wanted in this situation. 208 WTL Developer’s Guide Filled Shapes and Brushes WTL’s CDC offers these methods to draw filled rectangles. BOOL Rectangle(int x1, int y1, int x2, int y2); BOOL Rectangle(LPCRECT lpRect); BOOL FillRect(LPCRECT lpRect, HBRUSH hBrush); BOOL FrameRect(LPCRECT lpRect, HBRUSH hBrush); BOOL InvertRect(LPCRECT lpRect); BOOL RoundRect(int x1, int y1, int x2, int y2, int x3, int y3); BOOL RoundRect(LPCRECT lpRect, POINT point); BOOL DrawEdge(LPRECT lpRect, UINT nEdge, UINT nFlags); BOOL DrawFrameControl(LPRECT lpRect, UINT nType, UINT nState); The Rectangle method draws the outline with the current pen and fills with the current brush. FillRect draws using the brush parameter, does not use the pen, and draw the top and left borders, but does not touch the bottom and right borders. FrameRect uses the brush to draw the outline of the rectangle. InvertRect performs a logical NOT operation on each pixel inside the rectangle. RoundRect draws a rounded rectangle. DrawEdge renders the edges of a rectangle according to a variety of styles, such as sunken or raised, specified in the flags parameter. DrawFrameControl renders the face of a standard control, such as a pushbutton or a radiobutton. CDC offers these methods to render chords, pies and ellipses. BOOL Chord(int x1, int y1, int x2, int y2, int x3, int y3, int x4, int y4); BOOL Chord(LPCRECT lpRect, POINT ptStart, POINT ptEnd); void DrawFocusRect(LPCRECT lpRect); BOOL Ellipse(int x1, int y1, int x2, int y2); BOOL Ellipse(LPCRECT lpRect); BOOL Pie(int x1, int y1, int x2, int y2, int x3, int y3, int x4, int y4); BOOL Pie(LPCRECT lpRect, POINT ptStart, POINT ptEnd); The Pie method draws a pie shape. Chord draws part of an ellipse bounded by a line. Ellipse draws the entire ellipse. DrawFocusRect draws a rectangle in XOR mode. CDC offers these methods to draw polygons: BOOL Polygon(LPPOINT lpPoints, int nCount); BOOL PolyPolygon(LPPOINT lpPoints, LPINT lpPolyCounts, int nCount); The Polygon method draws a single polygon, whereas the PolyPolygon method draws multiple polygons. CDC offers these methods to rendered already loaded icons: BOOL DrawIcon(int x, int y, HICON hIcon); BOOL DrawIcon(POINT point, HICON hIcon); CDC offers these methods to render an image with an effect. 209 WTL Developer’s Guide BOOL DrawState(POINT pt, SIZE size, HBITMAP hBitmap, UINT nFlags, HBRUSH hBrush = NULL); BOOL DrawState(POINT pt, SIZE size, HICON hIcon, UINT nFlags, HBRUSH hBrush = NULL); BOOL DrawState(POINT pt, SIZE size, LPCTSTR lpszText, UINT nFlags, BOOL bPrefixText = TRUE, int nTextLen = 0, HBRUSH hBrush = NULL); BOOL DrawState(POINT pt, SIZE size, DRAWSTATEPROC lpDrawProc, LPARAM lData, UINT nFlags, HBRUSH hBrush = NULL); The image can be a bitmap, an icon, some text with or without a mnemonic, or what is known as complex, which is a callback function you provide to draw it. The effects can be hidden, hide-prefix, mono, normal (none), prefix only, right aligned or union (dithering). CBrushT A brush is used to specify how filling is to be done in filled shapes. The WTL template CBrushT manages one data member: an HBRUSH instance. CBrushT offers the following Create methods: HBRUSH HBRUSH HBRUSH HBRUSH HBRUSH HBRUSH CreateSolidBrush(COLORREF crColor); CreateHatchBrush(int nIndex, COLORREF crColor); CreateBrushIndirect(const LOGBRUSH* lpLogBrush); CreatePatternBrush(HBITMAP hBitmap); CreateDIBPatternBrush(HGLOBAL hPackedDIB, UINT nUsage); CreateDIBPatternBrush(const void* lpPackedDIB, UINT nUsage); HBRUSH CreateSysColorBrush(int nIndex); CBrushT provides these methods to access brush attributes: int GetLogBrush(LOGBRUSH* pLogBrush) const bool GetLogBrush(LOGBRUSH& LogBrush) const Sample : FilledShapesAndBrushes The FilledShapesAndBrushes sample exercises the CDC methods for filled shapes and the CBrush template. It is a modal-dialog based application. It contains a series of pushbuttons, which have been mapped to handlers that called the various methods. It has a static with an id of IDC_OUTPUT_WINDOW which is used as a destination for rendering. We wish to use a common DC, so we do nothing special when constructing the static. A member variable of the dialog class identifies this. CStatic m_OutputWindow; LRESULT OnInitDialog(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/){ . . . m_OutputWindow.Attach(GetDlgItem(IDC_OUTPUT_WINDOW)); } One sample handler when creating a brush is implemented as follows: LRESULT OnCreateBrushIndirect(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ 210 WTL Developer’s Guide LOGBRUSH logbrush; logbrush.lbColor = RGB(190,90,70); logbrush.lbHatch = HS_CROSS; logbrush.lbStyle = BS_HATCHED; CBrush br; br.CreateBrushIndirect(&logbrush); CClientDC dc(m_OutputWindow); dc.SelectBrush(br); dc.Rectangle(190,190,130,130); return 0; } Here is one sample of the handlers for a drawing method: LRESULT OnDrawState(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/){ CClientDC dc(m_OutputWindow); CPoint pt(150,150); CSize size(0,0); dc.DrawState(pt, size, TEXT("Hello there!"),DST_TEXT); return 0; } Text and Fonts WTL’s CDC offers these methods to render text: BOOL TextOut(int x, int y, LPCTSTR lpszString, int nCount =-1); BOOL ExtTextOut(int x, int y, UINT nOptions, LPCRECT lpRect, LPCTSTR lpszString, UINT nCount = -1, LPINT lpDxWidths=NULL); SIZE TabbedTextOut(int x, int y, LPCTSTR lpszString, int nCount = -1, int nTabPositions = 0, LPINT lpnTabStopPositions = NULL, int nTabOrigin = 0); int DrawText(LPCTSTR lpszString, int nCount, LPRECT lpRect, UINT nFormat); The TextOut method renders the string at the specified location using attributes of the DC. ExtTextOut is similar to TextOut, but also provides for clipping and opaquing. TabbedTextOut is also similar to TextOut, but expands tabs using the dimensions in the tab positions array. DrawText renders formats (e.g. justifies) and renders text. CDC offers these methods to get and set text-related information: BOOL GetTextExtent(LPCTSTR lpszString, int nCount, LPSIZE lpSize) const; BOOL GetTabbedTextExtent(LPCTSTR lpszString, int nCount, int nTabPositions, LPINT lpnTabStopPositions) const; BOOL GrayString(HBRUSH hBrush, BOOL (CALLBACK* lpfnOutput)(HDC, LPARAM, int), LPARAM lpData, int nCount, int x, int y, int nWidth, int nHeight); UINT GetTextAlign() const; UINT SetTextAlign(UINT nFlags); int GetTextFace(LPTSTR lpszFacename, int nCount) const; int GetTextFaceLen() const; 211 WTL Developer’s Guide BOOL GetTextMetrics(LPTEXTMETRIC lpMetrics) const; int SetTextJustification(int nBreakExtra, int nBreakCount); int GetTextCharacterExtra() const; int SetTextCharacterExtra(int nCharExtra); GetTextExtent will return the space occupied by a string when rendered in this DC. GetTabbedTextExtent does the same and takes tabbing into consideration. GrayString renders a string (or a bitmap) with a grayed background. GetTextMetrics returns a Win32 TEXTMETRIC structure filled in with lots of detailed information about the DC’s current font. Get/SetTextAlign manages the alignment of the text relative to the bounding rectangle of the text. GetTextFace returns the typeface name of the currently selected font. There are versions of GetTextFace that return a BSTR (if _ATL_NO_COM is not defined) and a WTL::CString (if __ATLSTR_H__ is defined). GetTextFaceLen returns the length of the name. Often applications will call this first, allocate enough memory for the name, and then call GetTypeFace. There is no Win32 function called GetTypeFaceLen – the way WTL::CDC::GetTypeFaceLen is implemented is by calling Win32’s GetTypeFace with a NULL buffer. Note that there are some problems with the implementation of CDC::GetTextFace(CString &str); See the later “Bugs and Suggestions” section for details. SetTextJustification sets the spacing to be added for break characters when a string is later rendered with TextOut etc. Get/SetTextCharacterExtra manages the inter-character spacing. This is the space used between each character when rendering. CDC offers these methods to manage fonts: BOOL GetCharWidth(UINT nFirstChar, UINT nLastChar, LPINT lpBuffer) const; DWORD SetMapperFlags(DWORD dwFlag); BOOL GetAspectRatioFilter(LPSIZE lpSize) const; BOOL GetCharABCWidths(UINT nFirstChar, UINT nLastChar, LPABC lpabc) const; BOOL GetCharABCWidths(UINT nFirstChar, UINT nLastChar, LPABCFLOAT lpABCF) const; BOOL GetCharWidth(UINT nFirstChar, UINT nLastChar, float* lpFloatBuffer) const; DWORD GetFontData(DWORD dwTable, DWORD dwOffset, LPVOID lpData, DWORD cbData) const; int GetKerningPairs(int nPairs, LPKERNINGPAIR lpkrnpair) const; UINT GetOutlineTextMetrics(UINT cbData, LPOUTLINETEXTMETRIC lpotm) const; DWORD GetGlyphOutline(UINT nChar, UINT nFormat, LPGLYPHMETRICS lpgm, DWORD cbBuffer, LPVOID lpBuffer, const MAT2* lpmat2) const; GetCharWidth returns the character widths of characters from a font. SetMapperFlags is used to get the DC to match the aspect ratio when mapping logical to physical fonts. GetAspectRatioFilter returns the aspect-ratio filter, which is a CSize structure, which specifies the desirable font size in pixels, when selecting fonts. GetCharABCWidths returns the ABC width of characters in a 212 WTL Developer’s Guide Truetype font. A stands for the space before the character glyph, B the width of what is drawn in the glyph and C the space after the drawn glyph. A+B+C is the space the current position is advanced when a character is drawn. GetFontData returns information about the metrics of a Truetype font. GetKerningPairs returns the kerning pairs for the current font. When certain pairs of characters are to be rendered together, then it is sometimes the case that the inter-character distance should be different than the standard. This is known as kerning. Usually the distance is less. GetOutlineTextMetrics returns TrueType font metrics. GetGlyphOutline returns the outline for a character, e.g. as bezier curve data. The following methods are only available when _WIN32_WINNT >= 0x0500 (i.e. Windows 2000 or later): DWORD GetFontUnicodeRanges(LPGLYPHSET lpgs) const; DWORD GetGlyphIndices(LPCTSTR lpstr, int cch, LPWORD pgi, DWORD dwFlags) const; BOOL GetTextExtentPointI(LPWORD pgiIn, int cgi, LPSIZE lpSize) const; BOOL GetTextExtentExPointI(LPWORD pgiIn, int cgi, int nMaxExtent, LPINT lpnFit, LPINT alpDx, LPSIZE lpSize) const; BOOL GetCharWidthI(UINT giFirst, UINT cgi, LPWORD pgi, LPINT lpBuffer) const; BOOL GetCharABCWidthsI(UINT giFirst, UINT cgi, LPWORD pgi, LPABC lpabc) const; GetFontUnicodeRanges returns data about the UNICODE characters in a font. GetGlyphIndices converts a string into indices for glyphs. GetTextExtentPointI determines the dimensions of an array of glyph indices. GetCharABCWidthsI returns the ABC widths of glyph indices. CFontT A font is used when rendering text. The WTL template CFontT manages one data member, a HFONT. CFontT provides these Create methods: HFONT CreateFontIndirect(const LOGFONT* lpLogFont); HFONT CreateFont(int nHeight, int nWidth, int nEscapement, int nOrientation, int nWeight, BYTE bItalic, BYTE bUnderline, BYTE cStrikeOut, BYTE nCharSet, BYTE nOutPrecision, BYTE nClipPrecision, BYTE nQuality, BYTE nPitchAndFamily, LPCTSTR lpszFacename); HFONT CreatePointFont(int nPointSize, LPCTSTR lpszFaceName, HDC hDC = NULL); HFONT CreatePointFontIndirect(const LOGFONT* lpLogFont, HDC hDC = NULL); When running on Windows 2000 or later, it also offers: HFONT CreateFontIndirectEx(CONST ENUMLOGFONTEXDV* penumlfex); CFontT provides these methods to access attrobites: int GetLogFont(LOGFONT* pLogFont) const 213 WTL Developer’s Guide bool GetLogFont(LOGFONT& LogFont) const Sample : TextandFonts The TextAndFonts sample explores the text methods in WTL’s CDC and the CFont template. It uses treeview to present the names of the methods, and when one item in the treeview is selected, the relevant methods are called. A sample handling of one API is as follows: m_pDC->TabbedTextOut(40,40, TEXT("Tabbed\tText\tOut"), -1, TAB_COUNT, tabList, 20 ); A sample creation of a font is: m_Font.CreateFont( 16, 0, 900, 900, FW_DONTCARE, 0, 0, 0,DEFAULT_CHARSET, OUT_TT_PRECIS, CLIP_DEFAULT_PRECIS, PROOF_QUALITY, DEFAULT_PITCH|FF_DONTCARE, "Helvetica" ); m_pDC->SelectFont(m_Font); 214 WTL Developer’s Guide Chapter 8 Internals of WTL Objectives • Identify the templates/classes in WTL’s source code Overview As it is a template library, WTL consists of a set of header files only (atl*.h). It has no .cpp files. It includes some header files from ATL but in general you can use WTL independently of using ATL. Header Files WTL consists of the following files: File Name Comments Length (text lines) Length (count of ‘;’) AtlApp.h AppModel, messagepump, reflectioning 872 244 AtlCrack.h Windows message crackers 1835 748 AtlCtrls.h Wrapper code for each of the Windows controls (The old code from ATLCON have evolved into this file) 7134 2732 AtlCtrlw.h Command Bars 2218 886 AtlCtrlx.h Bitmap buttons, checklistview, hyperlink, wait cursor and multipane status bar 1427 523 AtlDdx.h DDX Support 563 191 AtlDlgs.h Wrappers for common selection dialogs for File, folder, font, wrappers for common find/replace, printer setup, and property pages 2382 813 AtlFrame.h MDI/SDI frame windows, MDI child & UpdateUI 2259 823 AtlGdi.h Wrappers for pens, brushes, fonts, bitmaps, palettes and device contexts; OpenGL pixel formats and wgl wrappers 2847 1095 AtlMisc.h Wrappers for common windows structures such 3026 1211 215 WTL Developer’s Guide as SIZE, POINT, RECT, a string class called CString and helpers for Find File, recent documents, etc. AtlPrint.h Printing support 831 363 AtlRes.h List of #defines of Resource IDs 245 0 AtlScrl.h Scrolling 1020 364 AtlSplit.h Splitter windows 724 246 AtlUser.h Menu implementation 291 99 27,674 10,338 Total Contents of Each Header File CMessageLoop m_aMsgFilter : CSimpleArray m_aIdleHandler : CSimpleArray m_msg MSG AddMessageFilter (pMessageFilter : CMessageFIlter*) : BOOL CIdleHandler RemoveMessageFilter (pMessageFilter : CMessageFIlter*) : BOOL AddIdleHandler(pIdleHandler: CIdleHandler*) : BOOL OnIdle():BOOL {abstract} RemoveIdleHandler(pIdleHandler: CIdleHandler*) : BOOL AddUpdateUI(pIdleHandler: CIdleHandler*) : BOOL RemoveUpdateUI(pIdleHandler: CIdleHandler*) : BOOL CMessageFilter Run() : int IsIdleMessage (pMsg : MSG*) PreTranslaterMessage (pMsg : MSG*) : BOOL OnIdle (int) : BOOL ATLApp.h 216 PreTranslateMessage (pMsg: MSG*):BOOL {abstract} WTL Developer’s Guide CComModule ATLApp.h CAppModule m_dwMainThreadID m_pMsgLoopMap : CSimpleMap* m_pSettingChangeNotify : CSimpleArray* Init (pObjMap : _ATL_OBJMAP_ENTRY*, hInstance : HINSTANCE, * pLibID: const GUID* = NULL) Term() AddMessageLoop (pMsgLoop : CMessageLoop*) : BOOL RemoveMessageLoop () : BOOL GetMessageLoop (dwThreadID: DWORD = GetCurrentThreadId) : CMessageLoop* AddSettingChangeNotify (hWnd : HWND) : BOOL RemoveSettingChangeNotidy (hWnd : HWND) : BOOL _SettingChangeDlgProc (hwnd :HWND, uMsg : UINT, wParam : WPARAM, lParam : LPARAM) BOOL {callback} CServerAppModule m_hEventShutdown HANDLE m_bActivity bool m_dwTimeOut DWORD m_dwPause DWORD Init (pObjMap : _ATL_OBJMAP_ENTRY*, hInstance : HINSTANCE, * pLibID: const GUID* = NULL) Term() Unlock() : LONG MonitorShutdown () StartMonitor () : bool ParseCommandLine(lpCmdLine : LPCTSTR, nResId : UINT, pnRetCode : HRESULT bool *) : bool ParseCommandLine(lpCMdLine : LPCTSTR, LPCTSTR pAppId, pnRetCode : HRESULT *) : FindOneOf(p1: LPCTSTR, p2 : LPCTSTR) : LPCTSTR RegisterAppID(pAppId : LPCTSTR) : HRESULT UnregisterAppID 217 WTL Developer’s Guide CUpdateUIBase ATLFrame.h ... T CSimpleArray <_AtlUpdateUIElement> m_UI Elements COwnerDraw UIAddMenuBar ... UIAddToolBar IsMsgHandled UIAddStatusBar SetMsgHandled UIAddChildWIndowContainer OnDrawItem OnInitMenuPopup OnMeasureItem UIEnable OnCompareItem UISetCheck OnDeleteItem UISetRadio DrawItem UISetText MeasureItem UISetState CompareItem UIGetState DeleteItem UIUpdateMenuBar ATLCrack.h contains message crackers UIUpdateToolBar UIUpdateStatusBar UIUpdateChildWIndows UIUpdateMenuBarElement ATLRes.h UIUpdateToolBarElement contains #defines UIUpdateStatusBarElement UIUpdateChildWindow T CUpdateUI CupdateUI() 218 WTL Developer’s Guide CWindowImplBaseT TBase TWinTraits ATLFrame.h +m_pfnSuperWindowProc : WNDPROC +GetWndStyle() +GetWndExStyle() +StartWindowProc() +WindowProc() +Create() +DestroyWindow() +SubclassWindow() +UnsubclassWindow() DefWindowProc OnFinalMessage T TBase TWinTraits CFrameWindowImpl ... Create CreateEx CFrameWindowImplBase CreateSimpleToolBar TBase TWinTraits OnReBarAutoSize OnChevronPushed ... Create CreateSimpleToolBarCtrl CreateSimpleReBarCtrl CreateSImpleRebar AddSimpleRebarCtrl AddSimpleRebarBand SizeSimpleReBarBands CreateSimpleStatusBar UpdateLayout UpdateBarsPosition PreTranslateMessage OnEraseBackground OnMenuselect OnSetFocus OnDestroy OnToolTipTextA/W T TBase CMDIFrameWindowImpl TWinTraits ... Create CreateEx CreateSimpleToolBar GetWindowProc MDIFrameWindowProc {callback} DefWindowProc PreTranslateMessage CreateMDIClient OnSize OnSetFocus OnMDISetMenu OnReBarAutoSize CMDIChild -WindowImpl T TBase TWinTraits ... Create CreateEx CreateSimpleToolBar UpdateClientEdge OnSize OnWindowPosChanging OnMenuSelect OnMDIActivate DisplayChevro OnDestroy CleanupChevronMenu OnChevronPushed 219 WTL Developer’s Guide CWindow CMenuT CMenuT() CMDIWindow ~CMenuT() ... operator = MDIGetActive Attach MDIActivate Detach MDINext CreateMenu MDIMaximize CreatePopupMenu MDIRestore LoadMenu MDIDestroy LoadMenuIndirect MDICascade DestroyMenu MDITile DeleteMenu MDIIconArrange TrackPopupMenu MDISetMenu TrackPopupMenuEx MDIRefreshMenu AppendMenu GetStandardWindowMenu CheckMenuItem SetMDIFrameMenu EnableMenuItem GetMDIFrame GetMenuItemCOunt CFrameWindowClassInfo GetMenuItemID Register GetMenuState GetMenuString GetSubMenu ATLFrame.h InsertMenu ModifyMenu MENUINFO {struct which is part of Win32} RemoveMenu ATLUser.h SetMenuItemBitmaps CheckMenuRadioItem SetMenuItemInfo CMenuItemInfo InsertMenuItem SetMenuContextHelpId CMenuItemInfo() GetMenuContextHelpId 220 bManaged WTL Developer’s Guide ATLMisc.h CSize CPoint CRect CRecentDocumentList CString CFindFile ATLGdih CPenT CBrushT CFontT CPaletteT CBitmapT CRgnT CDCT CClientDC CPaintDC CWindowDC CEnhMetaFileDC 221 CEnhMetaFileInfo CEnhMetaFileT WTL Developer’s Guide CFileDialogImpl CFolderDialogImpl CFileDialog CFolderDialog CFontDialogImpl CFontDialog ATLDlgs.h CRichEditFontDialogImpl CRichEditFontDialog CColorDialogImpl CPrintDialogImpl CFindReplaceDialogImpl CColorDialog CPrintDialog CPrintDialogExImpl CFindReplaceDialog CPageSetupDialogImpl CPropertySheetWindow CPropertyPageWindow CPageSetupDialog CPropertySheetImpl CPropertyPageImpl CPropertySheet CWinDataExchange CPropertyPage TBase TWinTraits ... DoDataExchange DDX_Text DDX_Int ATLDdx.h _AtlSimpleFloatParse DDX_Float DDX_Control DDX_Check DDX_Radio Lots of DDX macros! OnDataExchangeError OnDataValidateError 222 WTL Developer’s Guide CBitmapButtonImpl T Tbase TWinTraits CBitmapButton CCheckListViewCtrlImpl T Tbase TWinTraits CCheckListViewCtrl CHyperLinkImpl T Tbase TWinTraits CHyperLink CWaitCursor ATLCtrlx.h CMultiPaneStatusBarCtrlImpl T Tbase CMultiPaneStatusBarCtrl ATLCtrlwh CCommandBarCtrlImpl T Tbase TWinTraits CCommand 223 WTL Developer’s Guide CStaticT CEditT CListViewCtrlT CButtonT CEditCommands CTreeViewCtrlT CListBox CScrollBarT CTreeItem CComboBoxT CImageList CHeaderCtrlT CToolInfo CToolBarCtrlT CToolTip CStatusBarCtrlT CTrackBarCtrl CTabCtrlT CRichEditCommands CDragListBoxT CDragListNotifyImpl CReBarCtrlT ATLCtrlsh CTreeViewCtrlExT CProgressBarCtr CHotKeyCtrlT CAnimate CUpDownCtrlT CRichEdit CComboBoxEx CFlatScrollBarT CDateTimePicke CIPAddressCtrlT CMonthCalendarCtrlT CPagerCtrlT CFlatScrollBarImpl CCustomDraw 224 WTL Developer’s Guide CSplitterImpl ATLSplit.h CSplitterWindowImpl CSplitterWindowT CScrollImpl ATLScrl.h CScrollWindowImpl CMapScrollImpl CMapScrollWindowImpl CFSBWindowT CPrinterInfo CPrintJobInfo CPrinterT CPrintJob CDevModeT CPrintPreview CPrinterDC CPrintPreviewWindowImpl CPrintPreviewWindow 225 ATLPrint.h

Source Exif Data:
File Type                       : PDF
File Type Extension             : pdf
MIME Type                       : application/pdf
PDF Version                     : 1.4
Linearized                      : No
Create Date                     : 2008:03:27 15:48:47+01:00
Creator                         : Writer
Modify Date                     : 2008:03:27 15:54:08+01:00
Producer                        : OpenOffice.org 2.3
Title                           : WTL Developer's Guide
Page Count                      : 225
Page Mode                       : UseOutlines
EXIF Metadata provided by EXIF.tools

Navigation menu