KEMBAR78
Learn Visual C++ | PDF | Component Object Model | Windows Registry
0% found this document useful (0 votes)
1K views977 pages

Learn Visual C++

Learn Visual c++

Uploaded by

manicheese
Copyright
© Attribution Non-Commercial (BY-NC)
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PDF, TXT or read online on Scribd
0% found this document useful (0 votes)
1K views977 pages

Learn Visual C++

Learn Visual c++

Uploaded by

manicheese
Copyright
© Attribution Non-Commercial (BY-NC)
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PDF, TXT or read online on Scribd
You are on page 1/ 977

Programming

Windows 98/NT

Viktor Toth

201 West 103rd Street Indianapolis, IN 46290

Unleashed

Programming Windows 98/NT Unleashed Copyright 1998 by Sams Publishing


All rights reserved. No part of this book shall be reproduced, stored in a retrieval system, or transmitted by any means, electronic, mechanical, photocopying, recording, or otherwise, without written permission from the publisher. No patent liability is assumed with respect to the use of the information contained herein. Although every precaution has been taken in the preparation of this book, the publisher and author assume no responsibility for errors or omissions. Neither is any liability assumed for damages resulting from the use of the information contained herein. International Standard Book Number: 0-672-31353-7 Library of Congress Catalog Card Number: 98-85231 Printed in the United States of America First Printing: June 1998 00 99 98 4 3 2 1

Executive Editor
Brad Jones

Acquisitions Editor
Matt Purcell

Development Editors
Matt Purcell Erik Dafforn

Managing Editor
Jodi Jensen

Project Editor
Tonya Simpson

Copy Editor
Heather Urschel

Trademarks
All terms mentioned in this book that are known to be trademarks or service marks have been appropriately capitalized. Sams Publishing cannot attest to the accuracy of this information. Use of a term in this book should not be regarded as affecting the validity of any trademark or service mark. Windows is a registered trademark of Microsoft Corporation. Windows NT is a registered trademark of Microsoft Corporation.

Indexer
Bruce Clingaman

Technical Editor
Greg Guntle

Software Development Specialist


Dan Scherf

Production
Marcia Deboy Michael Dietsch Jennifer Earhart Cynthia Fields Susan Geiselman

Overview
Preface xxvii

Part I Introduction to the Development System


1 Using the Development Environment 3

Part II Under the Hood: Windows and the Win32 API


2 Operating System Overview 3 The Message Loop 4 Windows, Dialog Boxes, and Controls 5 Resource Files 6 Drawing and Device Contexts 7 Threads and Processes 8 Memory Management 9 File Management 10 The Windows Clipboard 11 The Registry 12 Exception Handling 35 49 63 97 109 139 157 177 193 205 219

Part III Microsoft Foundation Classes


13 Exploring an MFC Skeleton Application 14 Working with Documents and Views 15 Dialogs and Property Sheets 16 MFC Support for Common Dialogs and Common Controls 17 Using ActiveX Controls 18 Device Context and GDI Objects 19 Serialization: File and Archive Objects 20 Collection Classes 21 Internet Support Classes 22 Exceptions, Multithreading, and Other MFC Classes 235 257 273 297 321 333 351 367 387 397

Part IV OLE, COM, and MFC Applications


23 OLE, ActiveX, and the Component Object Model 24 OLE Servers 417 439

iv

Programming Windows 98/NT UNLEASHED

25 26 27 28 29 30 31

OLE Containers OLE Drag and Drop Automation Building ActiveX Controls with MFC Using the ActiveX Template Library ActiveX Documents Distributed COM

455 477 493 511 537 559 571

Part V Client/Server Solutions


32 33 34 35 36 Database Programming Through ODBC Data Access Objects OLE DB and ADO Writing a Windows NT Service MTS and the Three-Tiered Model 581 607 625 639 653

Part VI Networks and Communications


37 38 39 40 41 Writing Messaging Applications with MAPI TCP/IP Programming with WinSock Using the WinInet API Telephony Applications with TAPI Named Pipes and Remote Procedure Calls 665 679 699 711 731

Part VII Graphics and Multimedia


42 Multimedia Applications 43 The OpenGL Graphics Library 44 High-Performance Graphics and Sound: DirectX 749 767 783

Part VIII Other Topics


45 46 47 48 Implementing Context-Sensitive Help Creating Installation Programs User Interface Extensions Localization: Creating International Applications Bibliography Index 803 827 847 865 889 891

Contents
Part I Introduction to the Development System
1 Using the Development Environment 3 Using the Compiler .......................................................................... 4 Using Command-Line Tools ....................................................... 4 Resources ..................................................................................... 5 Executables and DLLs ................................................................. 6 Integrated Development Environments ............................................ 8 The Integrated Editor .................................................................. 8 The Debugger ............................................................................. 8 Resource Editing ......................................................................... 9 Project Workspaces .................................................................... 10 Using Project Templates ............................................................ 10 Components .............................................................................. 11 Miscellaneous Tools .................................................................. 11 Profiling .................................................................................... 12 Version Control ......................................................................... 12 Using Visual C++ Features ............................................................. 13 Creating Projects with AppWizards ........................................... 13 Adding Functionality with the ClassWizard............................... 20 Using the Component Gallery ................................................... 28 Using the Visual C++ Debugger ................................................ 29 Summary ........................................................................................ 31

Part II Under the Hood: Windows and the Win32 API


2 Operating System Overview 35 Windows and Messages .................................................................. 36 Applications, Threads, and Windows ........................................ 36 Window Classes ......................................................................... 38 Message Types ........................................................................... 38 Messages and Multitasking ............................................................. 40 Message Queues ........................................................................ 40 Processes and Threads ................................................................ 40 Threads and Messages ................................................................ 41 Windows Function Calls ................................................................ 41 Kernel Services ........................................................................... 42 User Services .............................................................................. 44 GDI Services ............................................................................. 44 Other APIs ................................................................................ 46

vi

Programming Windows 98/NT UNLEASHED

Error Reporting ......................................................................... 47 Using Standard C/C++ Library Functions ................................. 47 Platform Differences ...................................................................... 48 Summary........................................................................................ 48 3 The Message Loop 49 The Real Hello, World Program ................................................. 50 A Simple Message Loop: Sent and Posted Messages ....................... 51 Window Procedures ....................................................................... 53 Comparison with generic.c ......................................................... 56 Multiple Message Loops and Window Procedures.......................... 57 Summary........................................................................................ 60 4 Windows, Dialog Boxes, and Controls 63 The Window Hierarchy ................................................................. 64 Window Management.................................................................... 67 The RegisterClass Function and the WNDCLASS Structure ...... 67 Creating a Window Through CreateWindow ............................ 69 Extended Styles and the CreateWindowEx Function ................. 70 Painting Window Contents ............................................................ 70 The WM_PAINT Message ............................................................. 71 Repainting a Window by Invalidating Its Contents ................... 71 Window Management Messages .................................................... 72 Window Classes ............................................................................. 74 The Window Procedure ............................................................ 74 Subclassing ................................................................................ 75 Global Subclassing ..................................................................... 78 Superclassing ............................................................................. 80 Dialog Boxes .................................................................................. 81 Modal Dialogs ........................................................................... 81 Modeless Dialogs ....................................................................... 82 Message Boxes ........................................................................... 82 Dialog Templates ...................................................................... 83 The Dialog Box Procedure ........................................................ 83 Common Dialogs ........................................................................... 84 The Open and Save As Dialogs .................................................. 84 The Choose Color Dialog .......................................................... 85 The Font Selection Dialog ......................................................... 86 Dialogs for Printing and Print Setup.......................................... 87 Text Find and Replace Dialogs .................................................. 88 Common Dialogs Example ........................................................ 89 OLE Common Dialogs ............................................................. 91 Controls ......................................................................................... 91 Summary........................................................................................ 94

Contents

vii

5 Resource Files 97 Resource File Components ............................................................. 98 Resource File Preprocessing ....................................................... 99 Single-Line Statements ............................................................ 100 Multiline Resource Statements ................................................ 101 User-Defined and Raw Data Resources ................................... 106 Compiling and Using Resource Scripts ........................................ 107 Running the Resource Compiler .............................................. 107 Running the Linker ................................................................. 107 Resource DLLs ........................................................................ 107 Summary ...................................................................................... 108 6 Drawing and Device Contexts 109 The GDI, Device Drivers, and Output Devices ........................... 110 Device Contexts ........................................................................... 111 Device Context Types ............................................................. 111 Memory and Metafile Device Contexts ................................... 112 Information Contexts .............................................................. 112 Coordinates .................................................................................. 113 Logical and Device Coordinates............................................... 113 Constrained Mapping Modes .................................................. 114 World Coordinate Transforms ................................................ 115 Drawing Objects .......................................................................... 122 Pens ......................................................................................... 123 Brushes .................................................................................... 123 Fonts ....................................................................................... 124 Palettes .................................................................................... 125 Bitmap Objects ........................................................................ 129 Clipping ....................................................................................... 129 Drawing Functions ...................................................................... 132 Lines ........................................................................................ 134 Curves ..................................................................................... 134 Filled Shapes ............................................................................ 135 Regions .................................................................................... 135 Bitmaps ................................................................................... 135 Paths ........................................................................................ 136 Text Output ............................................................................ 137 Notes About Printing ................................................................... 137 Summary ...................................................................................... 138

viii

Programming Windows 98/NT UNLEASHED

7 Threads and Processes 139 Multitasking in the Win32 Environment ..................................... 140 Multitasking Concepts............................................................. 140 Cooperative Multitasking ........................................................ 142 Preemptive Multitasking in Windows NT ............................... 142 Windows 95/98: A Mixed Bag of Multitasking Tricks ............. 143 Programming with Processes and Threads .................................... 145 Cooperative Multitasking: Yielding in the Message Loop......... 145 Processing Messages During Lengthy Processing ..................... 145 Using a Secondary Thread ....................................................... 148 Thread Objects ........................................................................ 151 Creating and Managing Processes ............................................ 151 Synchronization Objects .......................................................... 152 Programming with Synchronization Objects ........................... 153 Summary...................................................................................... 154 8 Memory Management 157 Processes and Memory ................................................................. 158 Separate Address Spaces ........................................................... 158 Address Spaces ......................................................................... 159 Virtual Memory ....................................................................... 160 32-Bit Programs ........................................................................... 161 Integer Size .............................................................................. 162 Type Modifiers and Macros ..................................................... 162 Address Calculations ................................................................ 163 Library Functions .................................................................... 163 Memory Models ...................................................................... 163 Selector Functions ................................................................... 163 Simple Memory Management ...................................................... 164 Memory Allocation via malloc and new .................................. 164 The Problem of Stray Pointers ................................................. 165 Sharing Memory Between Applications ................................... 165 Virtual Memory and Advanced Memory Management ................ 166 Win32 Virtual Memory Management ..................................... 166 Virtual Memory Functions ...................................................... 167 Heap Functions ....................................................................... 170 Windows API and C Runtime Memory Management ............. 171 Miscellaneous and Obsolete Functions .................................... 171 Memory-Mapped Files and Shared Memory ............................ 171 Shared Memory and Based Pointers ......................................... 173 Threads and Memory Management ............................................. 174 Interlocked Variable Access ...................................................... 174 Thread-Local Storage ............................................................... 174

Contents

ix

Accessing Physical Memory and I/O Ports ................................... 175 Summary ...................................................................................... 175 9 File Management 177 File System Overview ................................................................... 178 Supported File Systems ............................................................ 178 CD-ROM ............................................................................... 179 Network Volumes ................................................................... 179 File and Volume Compression ................................................. 179 Win32 File Objects ...................................................................... 179 Creating and Opening Files ..................................................... 180 Simple Input and Output ........................................................ 180 Asynchronous I/O Operations ................................................. 181 Low-Level I/O.............................................................................. 187 File Descriptors ....................................................................... 187 Standard File Descriptors ......................................................... 187 Low-Level I/O Functions ........................................................ 187 Stream I/O ................................................................................... 188 Stream I/O in C ...................................................................... 188 Stream I/O in C++ (The iostream Classes) ............................ 188 Special Devices ............................................................................. 189 Console I/O............................................................................. 189 Communication Ports ............................................................. 191 Summary ...................................................................................... 192 10 The Windows Clipboard 193 Clipboard Formats ....................................................................... 194 Standard Clipboard Formats .................................................... 194 Registered Formats .................................................................. 196 Private Formats ........................................................................ 196 Clipboard Operations .................................................................. 196 Transferring Data to the Clipboard ......................................... 196 Delayed Rendering .................................................................. 197 Pasting Data from the Clipboard ............................................. 197 Controls and the Clipboard ..................................................... 198 Clipboard Messages ................................................................. 198 Clipboard Viewers ................................................................... 198 A Simple Implementation ............................................................ 199 Summary ...................................................................................... 203 11 The Registry 205 Registry Structure ......................................................................... 206 Registry Values ........................................................................ 206 Registry Capacity ..................................................................... 207 Predefined Registry Keys ......................................................... 207

Programming Windows 98/NT UNLEASHED

Manually Editing the Registry ...................................................... 208 Commonly Used Registry Keys .................................................... 209 Subtrees in HKEY_LOCAL_MACHINE ........................................... 209 Subtrees in HKEY_CLASSES_ROOT ............................................. 210 Subtrees in HKEY_USERS .......................................................... 211 Subtrees in HKEY_CURRENT_USER ............................................. 211 The Registry and INI Files ...................................................... 212 Application Programs and the Registry ......................................... 212 Opening a Registry Key ........................................................... 212 Querying a Value ..................................................................... 213 Setting a Value ......................................................................... 213 Creating a New Key................................................................. 213 Other Registry Functions ......................................................... 214 A Working Example ................................................................ 214 Summary...................................................................................... 218 12 Exception Handling 219 Exception Handling in C and C++ ............................................... 220 C Exceptions ........................................................................... 220 C Termination Handling ......................................................... 224 C++ Exception Handling ......................................................... 226 Termination Handling in C++ ................................................ 226 C++ Exception Classes ............................................................. 227 Mixing C and C++ Exceptions ..................................................... 228 The Ellipsis Handler ................................................................ 228 Translating C Exceptions ......................................................... 230 Summary...................................................................................... 231

Part III Microsoft Foundation Classes


13 Exploring an MFC Skeleton Application 235 MFC and Applications ................................................................. 236 Foundation Class Fundamentals .................................................. 237 A Simple MFC Application Skeleton ........................................... 238 Creating the YAH Project ........................................................ 239 Exploring the Application Object ............................................ 239 The Message Map .................................................................... 243 The Frame, the Document, and the View ................................ 245 Adding Code to the Application ................................................... 253 Adding a String Resource......................................................... 253 Modifying the Document ........................................................ 253 Modifying the View ................................................................. 254 Summary...................................................................................... 255

Contents

xi

14 Working with Documents and Views 257 The CDocument Class ................................................................... 258 Declaring a Document Class in Your Application .................... 258 CDocument Member Functions ................................................ 259 Documents, Events, and Overridable Functions ...................... 260 Document Data ....................................................................... 262 CCmdTarget and CDocItem ..................................................... 265 The CView Class ........................................................................... 266 Declaring a View Class ............................................................ 267 CView Member Functions ....................................................... 268 Views and Messages ................................................................. 269 Variants of CView .................................................................... 270 Dialog-Based Applications ....................................................... 270 Summary ...................................................................................... 271 15 Dialogs and Property Sheets 273 Constructing Dialogs ................................................................... 274 Adding a Dialog Template ....................................................... 275 Constructing the Dialog Class ................................................. 275 Adding Member Variables ....................................................... 277 ClassWizard Results ................................................................. 278 Invoking the Dialog ................................................................. 279 Modeless Dialogs ..................................................................... 280 More on Dialog Data Exchange ................................................... 283 Dialog Data Exchange ............................................................. 283 Dialog Data Validation ............................................................ 283 Using Simple Types ................................................................. 284 Using Control Data Types ....................................................... 285 Implementing Custom Data Types .......................................... 285 Dialogs and Message Handling .................................................... 285 Property Sheets ............................................................................ 286 Constructing Property Pages .................................................... 287 Adding a Property Sheet Object .............................................. 291 CPropertyPage Member Functions ........................................ 291 Modeless Property Sheets ......................................................... 292 Summary ...................................................................................... 294 16 MFC Support for Common Dialogs and Common Controls 297 Common Dialogs ......................................................................... 298 CColorDialog ......................................................................... 299 CFileDialog ........................................................................... 299 CFindReplaceDialog ............................................................. 300 CFontDialog ........................................................................... 302

xii

Programming Windows 98/NT UNLEASHED

302 303 304 Common Controls ....................................................................... 305 Animation Control .................................................................. 306 Date Time Picker Control ....................................................... 307 Header Control ....................................................................... 307 Hotkey Control ....................................................................... 307 IP Address Control .................................................................. 308 List Control ............................................................................. 309 Month Calendar Control ......................................................... 310 Progress Bar ............................................................................. 311 Rich-Text Edit Control ........................................................... 311 Slider Control .......................................................................... 312 Spin Button ............................................................................. 313 Status Window ........................................................................ 313 Tab Control ............................................................................ 314 Toolbar.................................................................................... 315 Tooltip Control ....................................................................... 316 Tree Control ............................................................................ 316 Summary...................................................................................... 318 17 Using ActiveX Controls 321 Adding ActiveX Controls to Your Application ............................. 323 Creating a Control Container .................................................. 323 Adding an ActiveX Control to a Dialog Template ................... 323 Setting Control Properties ....................................................... 325 Adding Member Variables ....................................................... 326 Handling Messages .................................................................. 329 ActiveX Controls Supplied with Visual C++ ................................. 330 Summary...................................................................................... 331 18 Device Context and GDI Objects 333 Device Contexts ........................................................................... 334 The Basic CDC Class ................................................................. 335 Creating a Device Context ....................................................... 335 Paint-Time Device Contexts .................................................... 336 Client-Area Device Contexts ................................................... 337 Window Device Contexts ........................................................ 337 Metafile Device Contexts ......................................................... 337 CDC Attributes .......................................................................... 338 Coordinate Mapping and Views .............................................. 339 Simple Drawing Functions ...................................................... 341

CPageSetupDialog ................................................................. CPrintDialog ......................................................................... COleDialog ............................................................................

Contents

xiii

Selecting GDI Objects ............................................................. 341 Basic Lines and Shapes............................................................. 342 Bitmaps and Scrolling .............................................................. 343 Text and Font Functions ......................................................... 344 Clipping Operations ................................................................ 345 Printing ................................................................................... 345 Path Functions ........................................................................ 345 GDI Object Support in MFC ...................................................... 346 Pens ......................................................................................... 347 Brushes .................................................................................... 347 Bitmaps ................................................................................... 348 Fonts ....................................................................................... 348 Palettes .................................................................................... 348 Regions .................................................................................... 349 Summary ...................................................................................... 350 19 Serialization: File and Archive Objects 351 The CFile Class ........................................................................... 352 CFile Initialization ................................................................. 354 Reading from and Writing to a CFile Object ......................... 354 File Management ..................................................................... 354 Error Handling ........................................................................ 355 Locking ................................................................................... 355 Using a CFile in a Simple Application .................................... 355 The CStdioFile Class ............................................................ 356 The CInternetFile Class ...................................................... 356 The CMemFile Class ................................................................ 356 The COleStreamFile Class .................................................... 357 The CSocketFile Class .......................................................... 358 CArchive ..................................................................................... 358 Creating a CArchive ............................................................... 358 Reading and Writing Objects .................................................. 359 The Overloaded >> and << Operators ..................................... 359 The CObject::Serialize Member Function ........................ 360 Error Handling ........................................................................ 361 Using CArchive in Simple Applications .................................. 361 Serialization in MFC Framework Applications ............................. 362 Serialization in Documents ...................................................... 363 Helper Macros ......................................................................... 363 Serialization, OLE, and the Clipboard ..................................... 364 Summary ...................................................................................... 364

xiv

Programming Windows 98/NT UNLEASHED

20 Collection Classes 367 CObject Collections .................................................................... 368 The CObList Class and the POSITION Type ......................... 369 The CObArray Class ................................................................ 371 Other List Collections .................................................................. 373 The CPtrList Class ................................................................ 373 The CStringList Class .......................................................... 373 Other Array Collections ............................................................... 374 The CPtrArray Class .............................................................. 374 Integral Array Classes ............................................................... 374 The CStringArray Class ........................................................ 375 Mappings ..................................................................................... 375 The CMapStringToString Class ............................................. 375 The CMapStringToOb Class .................................................... 377 The CMapStringToPtr Class .................................................. 378 The CMapPtrToPtr Class ........................................................ 378 The CMapPtrToWord Class ...................................................... 379 The CMapWordToOb Class ........................................................ 379 The CMapWordToPtr Class ...................................................... 379 Template-Based Object Collections ............................................. 379 Collection Class Helper Functions ........................................... 380 The CList Template ............................................................... 381 The CArray Template ............................................................. 382 The CMap Template ................................................................. 383 The CTypedPtrList Template ................................................ 384 The CTypedPtrArray Template .............................................. 385 The CTypedPtrMap Template ................................................. 385 Summary...................................................................................... 385 21 Internet Support Classes 387 The MFC Internet Class Architecture .......................................... 388 Internet Sessions ...................................................................... 389 Internet Connections ............................................................... 389 Internet Files ........................................................................... 391 Other Support Classes ............................................................. 391 Using MFC Internet Classes in Applications ................................ 392 Communicating with an FTP Server ....................................... 392 Communicating with a Gopher Server .................................... 393 Communicating with an HTTP Server.................................... 394 Summary...................................................................................... 395

Contents

xv

22 Exceptions, Multithreading, and Other MFC Classes 397 Using Exceptions in MFC Applications ....................................... 398 Exception Handling with Macros ............................................ 398 C++ Exceptions and the CException Class ............................. 399 The CMemoryException Class ................................................. 401 The CFileException Class .................................................... 401 The CArchiveException Class ............................................... 402 The CNotSupportedException Class ..................................... 403 The CResourceException Class ............................................. 403 The CDaoException Class ...................................................... 404 The CDBException Class ........................................................ 404 The CInternetException Class ............................................. 404 The COleException Class ...................................................... 405 The COleDispatchException Class ....................................... 405 The CUserException Class .................................................... 405 Throwing an MFC Exception .................................................. 405 MFC and Multithreading............................................................. 406 Thread-Safe MFC .................................................................... 406 Creating Threads in MFC ....................................................... 407 Thread Synchronization .......................................................... 408 The CEvent Class ................................................................... 409 The CMutex Class .................................................................... 409 The CCriticalSection Class ................................................. 409 The CSemaphore Class ............................................................ 409 Synchronization with CSingleLock and CMultiLock ............. 410 Miscellaneous MFC Classes ......................................................... 410 Simple Value Types ................................................................. 410 Structures and Support Classes ................................................ 411 Summary ...................................................................................... 413

Part IV OLE, COM, and MFC Applications


23 OLE, ActiveX, and the Component Object Model 417 OLE Basics and the Component Object Model ........................... 418 Interfaces and Methods ............................................................ 418 Methods and Memory Allocation ............................................ 419 Inheritance and Reusing Objects ............................................. 420 Interface Identifiers .................................................................. 420 Interface Definition Through IUnknown .................................. 420 Class Objects and Registration ................................................. 421 Inter-Object Communication .................................................. 421

xvi

Programming Windows 98/NT UNLEASHED

Monikers ................................................................................. 422 COM and Threads .................................................................. 422 COM and Compound Documents .............................................. 422 Structured Storage ................................................................... 423 Data Transfer .......................................................................... 423 Compound Documents ........................................................... 424 Applications of COM and OLE ................................................... 425 OLE Document Containers and Servers .................................. 425 Automation ............................................................................. 425 OLE Drag and Drop ............................................................... 426 ActiveX Controls ..................................................................... 426 Custom Interfaces .................................................................... 426 A Simple Example ........................................................................ 426 Functional Description ............................................................ 427 The Hello Server Application .................................................. 428 Registering and Running the Server ......................................... 435 Accessing the Server from C++ ................................................ 436 Summary...................................................................................... 437 24 OLE Servers 439 Server Concepts ........................................................................... 440 Full-Servers and Mini-Servers .................................................. 440 In-Place Editing ....................................................................... 440 Server Activation ..................................................................... 440 Creating a Server Application with MFC ..................................... 440 Using AppWizard to Create an Application Skeleton............... 441 The OLE Server Skeleton Application ..................................... 441 The Server Item ....................................................................... 442 COleDocument and Document Items ...................................... 444 The In-Place Frame Window................................................... 445 Modes of Operation and Resources ......................................... 445 Running the Server Skeleton.................................................... 447 Customizing a Skeleton Server ..................................................... 447 Modifying the Document ........................................................ 448 Adding Drawing Code ............................................................. 448 Adding a Dialog ...................................................................... 450 Serialization ............................................................................. 452 Registering the New Application ............................................. 452 Summary...................................................................................... 453 25 OLE Containers 455 Creating a Container Application Through AppWizard ............... 456 Creating the Skeleton Application ........................................... 456

Contents

xvii

The OLE Container Skeleton Application ............................... 457 Running the Skeleton Container ............................................. 458 The Skeleton Container Code ................................................. 460 Container Menus ..................................................................... 466 Customizing the Application ........................................................ 468 Object Positions ...................................................................... 469 Drawing All Objects ................................................................ 471 Object Selection ...................................................................... 472 Other Features ......................................................................... 474 Summary ...................................................................................... 475 26 OLE Drag and Drop 477 Drag-and-Drop Basics .................................................................. 478 Creating a Container Application ................................................. 478 Creating the Application .......................................................... 479 Adding Positioning Support .................................................... 479 Adding Selection Support ........................................................ 482 Adding Drag-and-Drop Support .................................................. 482 Implementing a Drag Source ................................................... 483 Implementing a Drop Target ................................................... 484 Summary ...................................................................................... 492 27 Automation 493 Building an Automation Server .................................................... 494 Constructing the ASRV Application Skeleton .......................... 495 Implementing the Calculation ................................................. 495 Adding Automation Support ................................................... 499 The Type Library .................................................................... 503 Testing the Application ........................................................... 504 Standard Methods and Properties ................................................. 506 The Application Object ........................................................... 507 The Documents Collection ..................................................... 508 The Document Object ............................................................ 508 The Objects Collection ............................................................ 509 Summary ...................................................................................... 510 28 Building ActiveX Controls with MFC 511 Creating a Skeleton Control with AppWizard .............................. 513 Creating the Skeleton Control ................................................. 513 ActiveX Control Code Overview ............................................. 515 Customizing the Control ............................................................. 523 Changing the Controls Bitmap ............................................... 524 Adding Properties .................................................................... 524

xviii

Programming Windows 98/NT UNLEASHED

Making a Property Persistent ................................................... 527 Adding Methods ...................................................................... 528 Adding Events ......................................................................... 528 Drawing the Control ............................................................... 530 Adding a Property Page Interface ................................................. 531 Editing the Property Page ........................................................ 531 Connecting the Property Page with Control Properties ........... 531 Additional Property Pages ........................................................ 533 Testing, Distributing, and Using a Custom Control......................................................................... 534 Testing an ActiveX Control ..................................................... 534 ActiveX Control Distribution .................................................. 534 Using ActiveX Controls in Applications .................................. 535 ActiveX Controls on the Web .................................................. 535 Summary...................................................................................... 535 29 Using the ActiveX Template Library 537 Why ATL? ............................................................................... 538 Building an ActiveX Control with ATL........................................ 539 Creating a Skeleton ATL COM Project ................................... 540 Adding a Control..................................................................... 542 Adding Properties and Methods .............................................. 543 Adding Drawing Code ............................................................. 546 Adding a Property Page ........................................................... 547 Event Handling ....................................................................... 550 Adding a Bitmap ..................................................................... 554 Testing the Control ................................................................. 556 Summary...................................................................................... 557 30 ActiveX Documents 559 Overview ...................................................................................... 560 The ActiveX Document Interface ................................................. 561 ActiveX Document Servers ...................................................... 561 ActiveX Document Containers ................................................ 562 Creating ActiveX Document Applications .................................... 562 Creating an ActiveX Document Container .............................. 562 Creating an ActiveX Document Server .................................... 567 Converting an Existing OLE Server to Support ActiveX Documents ............................................................................ 569 Summary...................................................................................... 570

Contents

xix

31 Distributed COM 571 COM and DCOM: An Evolution................................................ 572 Configuring the Client Workstation ........................................ 572 Using DCOMCNFG.EXE .............................................................. 573 Explicit Coding Practices ............................................................. 575 Beyond DCOM: COM+ ............................................................. 577 Summary ...................................................................................... 578

Part V Client/Server Solutions


32 Database Programming Through ODBC 581 ODBC in Action .......................................................................... 582 The ODBC Setup Applet ........................................................ 582 ODBC API Concepts .............................................................. 583 A Simple ODBC Example ....................................................... 586 Other ODBC Calls ................................................................. 589 The SQL Standard and ODBC .................................................... 591 Data Manipulation Statements ................................................ 591 Views ....................................................................................... 592 Data Definition Statements ..................................................... 593 ODBC in MFC Applications ....................................................... 593 Setting Up a Data Source ........................................................ 594 Creating an ODBC Application Skeleton Through App Wizard .................................................................................. 596 Customizing the ODBC Application ....................................... 601 ODBC Classes in MFC ........................................................... 602 Summary ...................................................................................... 605 33 Data Access Objects 607 DAO Overview ............................................................................ 608 Building a DAO Application ........................................................ 608 The Database ........................................................................... 609 Creating the Skeleton Application ........................................... 612 Exploring the DAO Application Skeleton ................................ 613 Customizing the Application ................................................... 617 DAO Classes ................................................................................ 619 The CDaoRecordset Class ...................................................... 620 The CDaoDatabase Class ........................................................ 621 The CDaoWorkspace Class ...................................................... 621 The CDaoQueryDef Class ........................................................ 622 The CDaoTableDef Class ........................................................ 622 Miscellaneous DAO Classes ..................................................... 622 Summary ...................................................................................... 623

xx

Programming Windows 98/NT UNLEASHED

34 OLE DB and ADO 625 OLE DB ...................................................................................... 626 The OLE DB SDK .................................................................. 627 Basic Concepts ........................................................................ 627 A Working Example ................................................................ 628 ActiveX Data Objects ................................................................... 633 ADO Objects Overview .......................................................... 633 A Working Example ................................................................ 635 Summary ...................................................................................... 638 35 Writing a Windows NT Service 639 Services in the Windows NT Environment .................................. 640 The Service Control Manager .................................................. 640 Starting and Stopping Services ................................................. 641 Services and the Win32 API .................................................... 641 Creating a Windows NT Service Application ............................... 642 Using the Windows NT SDK Service Sample ......................... 642 Chat Service Application Architecture ..................................... 643 The Main Service Module ....................................................... 644 The Service Class ..................................................................... 647 The CSlot Class ...................................................................... 649 Compiling and Running the Service ........................................ 651 Summary ...................................................................................... 651 36 MTS and the Three-Tiered Model 653 Dynamic HTML ......................................................................... 654 The Three-Tiered Client/Server Model ........................................ 657 The Tiers ................................................................................. 658 Sample Implementations ......................................................... 658 The Microsoft Transaction Server ................................................ 659 Scalability Issues ...................................................................... 660 MTS Concepts ........................................................................ 660 MTS Programming: An Overview ........................................... 661 Summary ...................................................................................... 662

Part VI Networks and Communications 663


37 Writing Messaging Applications with MAPI 665 The MAPI Architecture ............................................................... 666 Types of MAPI Support .......................................................... 668 Service Providers ...................................................................... 668 MAPI Profiles .......................................................................... 669

Contents

xxi

MAPI APIs ................................................................................... 670 Simple MAPI ........................................................................... 670 Common Messaging Calls ....................................................... 673 Extended MAPI ....................................................................... 675 Active Messaging ..................................................................... 675 MAPI Support in MFC ................................................................ 677 MAPI Support in CDocument ................................................ 677 MAPI and AppWizard ............................................................. 677 Summary ...................................................................................... 678 38 TCP/IP Programming with WinSock 679 TCP/IP Networks and OSI .......................................................... 680 The Internet Protocol Suite ..................................................... 680 IP Datagrams ........................................................................... 682 IP Headers ............................................................................... 682 IP Host Addresses and Routing ............................................... 683 Hostnames............................................................................... 684 TCP and UDP Packets, Port Numbers, and Sockets ............... 684 Internet Services ...................................................................... 685 The WinSock API ........................................................................ 685 WinSock Initialization ............................................................. 686 Creating and Using Sockets ..................................................... 686 Name Service ........................................................................... 687 Byte Ordering .......................................................................... 687 Communication Through Sockets ........................................... 687 The Blocking Problem and the select Call ............................ 689 Asynchronous Socket Calls ...................................................... 690 A Simple WinSock Example......................................................... 690 Socket Programming and the Microsoft Foundation Classes ........ 692 CAsyncSocket Example .......................................................... 692 Synchronous Operations and Serialization ............................... 694 Further Information ..................................................................... 695 Summary ...................................................................................... 696 39 Using the WinInet API 699 Internet Protocols ......................................................................... 700 The File Transfer Protocol (FTP) ............................................ 701 The Gopher Protocol ............................................................... 702 The Hypertext Transfer Protocol (HTTP)............................... 704 The WinInet Library .................................................................... 705 Retrieving Files from an FTP Server ........................................ 706 Retrieving Files from a Gopher Server ..................................... 707

xxii

Programming Windows 98/NT UNLEASHED

Retrieving Files from an HTTP Server .................................... 708 Other WinInet Features and Functions ................................... 709 Summary...................................................................................... 710 40 Telephony Applications with TAPI 711 TAPI Overview ............................................................................ 712 Assisted TAPI: The Simplest TAPI Application ....................... 712 TAPI Concepts ........................................................................ 713 TAPI Devices .......................................................................... 714 TAPI Addresses ....................................................................... 714 TAPI Software Architecture ......................................................... 716 Synchronous and Asynchronous Operations ............................ 716 Variable-Length Structures ...................................................... 718 TAPI Services ............................................................................... 719 The TAPI Programming Model .............................................. 720 TAPI Media Modes ................................................................. 722 Multiple Applications .............................................................. 722 A Data Communication Example ................................................ 723 Summary...................................................................................... 728 41 Named Pipes and Remote Procedure Calls 731 Communicating with Pipes .......................................................... 732 Creating Pipes ......................................................................... 732 Connecting to Named Pipes .................................................... 733 Transferring Data Through Pipes ............................................ 734 A Working Example ................................................................ 734 Microsoft Remote Procedure Calls ............................................... 736 RPC Fundamentals.................................................................. 736 A Simple Example ................................................................... 737 Specifying the Interface ............................................................ 738 Implementing the Server .......................................................... 740 Implementing the Client ......................................................... 742 RPC Exception Handling ........................................................ 743 Advanced RPC Features .......................................................... 744 Summary...................................................................................... 745

Part VII Graphics and Multimedia 747


42 Multimedia Applications 749 Video Playback with One Function Call ...................................... 750 Fundamentals of Multimedia Programming ................................. 752 Multimedia Data Formats ....................................................... 752 Multimedia Interfaces .............................................................. 753

Contents

xxiii

Programming with MCIWnd ........................................................... 754 The MCIWnd Window Class ..................................................... 754 MCIWnd Functions .................................................................... 755 MCIWnd Macros ........................................................................ 756 MCIWnd Notifications ............................................................... 758 The Media Control Interface ....................................................... 759 MCI Command String Syntax ................................................. 761 MCI Command Sets ............................................................... 762 MCI Functions and Macros ..................................................... 762 MCI Notifications ................................................................... 763 Advanced Interfaces ...................................................................... 763 AVIFile and AVIStream Functions ......................................... 763 Custom File and Stream Handlers ........................................... 763 DrawDib Functions .................................................................. 763 The Video Compression Manager ........................................... 764 Video Capture ......................................................................... 764 Waveform Audio Recording and Playback ............................... 764 The Audio Compression Manager ........................................... 764 MIDI Recording and Playback ................................................ 764 Audio Mixers ........................................................................... 765 Miscellaneous Multimedia Services .......................................... 765 Summary ...................................................................................... 765 43 The OpenGL Graphics Library 767 OpenGL Overview ....................................................................... 768 Basic OpenGL Concepts ......................................................... 769 Initialization ............................................................................ 770 Drawing with OpenGL ........................................................... 771 Additional Libraries ................................................................. 772 Writing OpenGL Windows Applications in C ............................. 772 OpenGL Initialization ............................................................. 775 The Window Procedure .......................................................... 775 Compiling and Running the Application ................................. 776 OpenGL in MFC Applications .................................................... 776 OpenGL Initialization ............................................................. 777 Drawing the Cube ................................................................... 778 Running the Application ......................................................... 780 Summary ...................................................................................... 781

xxiv

Programming Windows 98/NT UNLEASHED

44 High-Performance Graphics and Sound: DirectX 783 The DirectX APIs ........................................................................ 785 DirectX and the Component Object Model ............................ 785 DirectDraw ............................................................................. 786 Direct3D ................................................................................. 788 DirectSound ............................................................................ 788 DirectPlay................................................................................ 790 DirectInput ............................................................................. 791 DirectShow ............................................................................. 791 DirectAnimation ..................................................................... 791 DirectSetup ............................................................................. 792 A Working Example ..................................................................... 792 Summary...................................................................................... 799

Part VIII Other Topics


45 Implementing Context-Sensitive Help 803 Help File Development ................................................................ 805 Help Topics and the Rich Text Format ................................... 806 The Help Project File .............................................................. 809 The Help Contents File ........................................................... 810 Compiling Help ...................................................................... 811 Macros and DLLs .................................................................... 811 Invoking Help from Applications ............................................ 812 The Microsoft Help Workshop .................................................... 813 Editing a Help Project ............................................................. 813 Editing a Help Contents File ................................................... 818 Testing and Running Help ...................................................... 820 AppWizard-Generated Help File Skeletons .................................. 822 HTML Help ................................................................................ 822 The HTML Help System ........................................................ 822 Table of Contents and Index ................................................... 823 The HTML Help API ............................................................. 824 The HTML Help Workshop ................................................... 824 Summary...................................................................................... 824 46 Creating Installation Programs 827 Installation Program Requirements .............................................. 828 Media Types ............................................................................ 829 Multiple Pieces of Media ......................................................... 829 Efficient Storage ...................................................................... 829 User-Selectable Options ........................................................... 829 Registration and/or Authentication .......................................... 829

Contents

xxv

Conditional Copying ............................................................... 830 Configuration Updates ............................................................ 830 Shared Components ................................................................ 830 Uninstallation .......................................................................... 830 InstallShield 5 .............................................................................. 830 Creating a Project with the Project Wizard .............................. 831 Adding Files ............................................................................ 835 Editing Resources .................................................................... 839 The Setup Script ...................................................................... 840 Building Distribution Sets ....................................................... 842 Testing and Debugging ........................................................... 844 Summary ...................................................................................... 846 47 User Interface Extensions 847 Interfacing with the Shell ............................................................. 848 Namespace .............................................................................. 848 Application Desktop Toolbars (Appbars) ................................. 848 Interacting with the Taskbar .................................................... 848 Creating File Viewers ............................................................... 849 Writing Shell Extension DLLs ................................................. 849 Other Extensions ..................................................................... 850 Examples ...................................................................................... 850 Installing a Taskbar Icon ......................................................... 851 Adding a Shell Property Page ................................................... 853 Summary ...................................................................................... 864 48 Localization: Creating International Applications 865 Preparing for Nationalization: Programming Practices ................. 867 Text Size .................................................................................. 867 Depending on the Grammar .................................................... 868 Isolating Language-Specific Information .................................. 869 User Input ............................................................................... 869 Non-European Languages ....................................................... 870 Tools for International Programming ........................................... 870 Locales ..................................................................................... 870 Character Sets .......................................................................... 872 Writing a Unicode Application .................................................... 874 Unicode in Console Applications ............................................. 874 Unicode in Windows Applications .......................................... 875 Unicode and MFC .................................................................. 876 Multilanguage Resources .............................................................. 877 Translating the User Interface into a Foreign Language ........... 878 Multiple Application Versions ................................................. 879

xxvi

Programming Windows 98/NT UNLEASHED

Satellite DLLs .......................................................................... 879 Multiple-Language Resources .................................................. 879 Help Files ................................................................................ 885 Installation Issues ..................................................................... 885 Summary ...................................................................................... 886 Bibliography 889 Index 891

xxvii

Preface
If your background is similar to mine and you, too, have written thousands of lines of code in a non-graphical environment, youll know that Windows programming is very different. Gone is the simple, linear execution model of a command-line program; instead, we deal with multiple, simultaneous paths of execution as our programs respond to events, we manage concurrency in a multithreaded environment, we design a graphical presentation instead of printing simple text messages. The programming is different, so it shall come as no surprise that the tools are also different. For todays Windows programmer a development environment must provide many features in addition to the simple capability to compile and link code. In response to this, over the years numerous companies developed a variety of development tools, many of which are still actively in use worldwide in the construction of high-quality applications. These tools implement a variety of programming languages from Fortran to Pascal, from BASIC to Java. Yet among all these languages C, and its younger cousin, C++, stand out as the languages of choice for the professional Windows developer. No surprise here: all Windows application programming interfaces, or APIs, have been developed with the C programmer in mind, and the ability to access them from other languages often exists as a mere afterthought. If C and C++ are the languages of choice for many programmers, the compiler of choice is Microsofts flagship development tool, Visual C++. Market research by Microsoft and others indicates that the majority of C/C++ programmers use this tool for Windows software development. Makes perfect sense: if an operating systems manufacturer releases the same tool that is used in the development of the operating system itself, why would you use third-party implementations instead? One reason, of course, would be quality. Yet in this department Microsoft has little to be ashamed of; Visual C++ is, if not the best, then one of the best C/C++ development tools around. (The same isnt always true; for example, when it comes to creating installation programs, InstallShield beats anything Microsoft has to offer. Which might explain why Microsoft also packages InstallShield as the setup toolkit of choice with Visual C++.) In this book, the focus is not on the tools but on the tasks you accomplish with them. Consequently, a single tool will be used throughout most chapters to develop examples. After what I just wrote about Visual C++, it shall come as no surprise that the tool I chose is Visual C++. However, most examples are easily recompilable using other development environments (especially if they are packaged with an up-to-date version of the Microsoft Foundation Classes

xxviii

Programming Windows 98/NT UNLEASHED

library). Furthermore, the principles of Windows programming that are revealed here are directly applicable even when you use other languages; for example, I regularly utilize Win32 API functions in the occasional snippets of Visual Basic code that I write during the course of my daily work. Visual C++ has many versions. Curiously, I find myself using three different versions of Visual C++ regularly. The most recent production version at the time of this writing is version 5 (with Visual C++ 6, or Visual C++ 98, whichever name it will be called, on its way). This product contains all the latest features but can no longer be used to develop 32-bit applications that run under Windows 3.1 (using the Win32s extensions). For this reason, I also installed a copy of Visual C++ 4.1, the last 32-bit version that supports targeting Win32s. Finally, I use Visual C++ 1.52c to create 16-bit applications. Of course, if you are only targeting 32-bit Windows platforms, this is a non-issue. You can safely use the latest features in the newest development systems, with only one possible downside: if what you are using is not yet a standard operating system feature, you might need to redistribute a large number of components. But in the age of CD-ROMbased software distribution this is unlikely to cause you to change your development strategy. Programming in C or C++ is not easy, not even with the help of the latest development tools. However, the task need not be as difficult as some would like you to believe. For example, there is an oft-quoted myth that even the simplest Windows program is many hundreds of lines long and requires weeks of study to understand. If you ever hear this again from diehard DOS or UNIX programmers, just show them the following program:
#include <windows.h> int WINAPI WinMain(HINSTANCE d1, HINSTANCE d2, LPSTR d3, int d4) { MessageBox(NULL, Hello, World!, , MB_OK); }

Although the lines themselves are longer, this program has the exact same number of lines as the command-line original from the classic K&R book by Kernighan and Ritchie:
#include <stdio.h> void main(void) { printf(Hello, World!\n); }

As this example so well demonstrates, Windows programming can be made simple, which is exactly the philosophy I followed when developing the examples for this book. Take, for example, MAPI. If you have never done MAPI programming and want to add some messaging features to your application, what would you prefer as your first exposure to MAPI? A comprehensive MFC-based example with 100KB of source code? Here is what I offer instead:
#include <windows.h> #include <stdio.h>

Preface

xxix

#include <mapi.h> LPMAPILOGON lpfnMAPILogon; LPMAPISENDMAIL lpfnMAPISendMail; LPMAPILOGOFF lpfnMAPILogoff; MapiRecipDesc recipient = { 0, MAPI_TO, Bill Clinton, SMTP:president@whitehouse.gov, 0, NULL }; MapiMessage message = { 0, Greetings, Hello, Mr. President!\n, NULL, NULL, NULL, 0, NULL, 1, &recipient, 0, NULL }; void main(void) { LHANDLE lhSession; HANDLE hMAPILib; hMAPILib = LoadLibrary(MAPI32.DLL); lpfnMAPILogon = (LPMAPILOGON)GetProcAddress(hMAPILib, MAPILogon); lpfnMAPISendMail = (LPMAPISENDMAIL)GetProcAddress(hMAPILib, MAPISendMail); lpfnMAPILogoff = (LPMAPILOGOFF)GetProcAddress(hMAPILib, MAPILogoff); (*lpfnMAPILogon)(0, NULL, NULL, MAPI_ALLOW_OTHERS, 0, &lhSession); (*lpfnMAPISendMail)(lhSession, 0, &message, 0, 0); (*lpfnMAPILogoff)(lhSession, 0, 0, 0); printf(Message to the White House sent.\n); FreeLibrary(hMAPILib); }

This program, in all its 42 lines, is a fully functional command-linebased MAPI application. It demonstrates how to load the MAPI library, use MAPI data structures, and call MAPI functions. Nor does compiling this program require a 1000-line makefile (I always considered such huge makefiles somewhat obscene in examples); you can compile it from the command line simply by typing cl cmdmsg.c. (But for goodness sake, before you do so, change the e-mail address in line 12!) I actually use simple examples like this one regularly when I try to understand a new programming topic. Even when I find a suitable sample application somewhere, I often mutilate it by removing all nonessential fluff. After all, when I want to learn about MAPI, I am not really interested in menus, resource files, or how to construct fancy dialogs or use elegant techniques for error handling.

xxx

Programming Windows 98/NT UNLEASHED

Bloated, complex examples are just one of the reasons why Windows programming often appears more difficult than it seems. Another reason, paradoxically, is the existence of excellent tools that manage to hide so many of the ugly details. If you never used anything other than the MFC or high-level development tools like Visual Basic, you might never have heard of the concept of the message loop. That while loop, lying at the heart of practically all Windows programs, obtains messages from the operating system and dispatches them to event handler functions. Yet without a grasp of this simple concept, you will never be able to program Windows efficientlyyou will just blindly follow programming recipes without ever understanding how they work. Which is why I devoted several chapters in this book to some of the fundamental topics of Windows programming. Not that this book is meant exclusively for those with little prior Windows programming exposure. Hopefully, even seasoned Windows programmers among you will find some of these chapters useful when you want to understand more about such Win32 topics as structured exceptions, virtual memory, or programming with the Registry. Let me say a few words about the topics and organization of this book. Part I is a brief overview of the capabilities of the Visual C++ development system, which is used throughout the book in examples. Part II is about the fundamentals of Windows programming. The chapters cover the basics as well as several more advanced topics, such as thread management, virtual memory, structured exceptions, and the Registry. Part III introduces the Microsoft Foundation Classes Library. Documents and views, dialogs, controls (including the use of ActiveX controls), device contexts and GDI objects, serialization, collections, and miscellaneous classes are the topics of the chapters that comprise this part. Part IV is about ActiveX and the underlying Component Object Model (COM) technology. In accordance with my philosophy of providing the fundamentals before presenting cookbooklike MFC recipes, I begin this part with a review of COM fundamentals. A non-MFC example of a mere 300-some lines that is nevertheless a fully functional Automation server highlights some of the basic ideas presented in the first chapter. In the remaining four chapters, the use of MFC is explored in developing OLE containers, OLE component servers, OLE drag-and-drop applications, and Automation servers. Part V is about client/server programming. It provides a review of the Microsoft Data Access Components (MDAC); specifically, it covers topics such as ODBC, DAO, ADO, and OLE DB. The use of MFC to build ODBC and DAO applications is also demonstrated. Part VI is about networking and communications. Topics include TCP/IP programming, the WinInet API, TAPI, named pipes, and RPC. Part VII is about graphics and multimedia. The three chapters in this part review the basic multimedia support in Windows, the OpenGL graphics library, and the DirectX API.

Preface

xxxi

Part VIII contains an assortment of chapters about programming topics not covered elsewhere. These include creating context-sensitive help, building installation programs, or programming the Windows 95 and Windows NT shell. Last but not least, I should say a word to those of you who have had a chance to read my book, Visual C++ Unleashed. As you might have noticed, the volume you hold in your hands inherited a lot of material from that book. No, this is not some vile deception; it is my publishers recognition that the material I wrote, suitably expanded and revised, goes beyond the use of a single development system and provides a solid foundation for Windows programming in general. If this recognition allows my work to reach a broader audience, I am truly grateful.

xxxii

Programming Windows 98/NT UNLEASHED

About the Author


Viktor Toth is a Hungarian-born author and self-employed software developer. His professional career started in 1979, when he wrote his first Hungarian-language book on Ern Rubiks Magic Cube. Between that time and 1986, he developed many scientific and business applications for clients in Hungary, Austria, Germany, and the United Kingdom. He wrote applications in Fortran, a variety of assemblers, Simula-67, C, and Pascal, just to name a few languages. In 1986 he authored his second book, a technical reference for programmers of the Commodore 16 home computer. Viktor Toth became a resident of Canada in 1987. There, he continued his self-employed career. He co-authored several studies as a consultant for the Canadian government and wrote numerous applications in C and C++, assembler, dBase, and other environments. After briefly experimenting with other graphical systems such as the long-forgotten GEM, he eventually wrote his first Windows application in 1990 and has not looked back since. In addition to being the author of Visual C++ Unleashed, Viktor also co-authored Sams Publishings Windows 95 Programming Unleashed and Windows NT Workstation 4 Unleashed. When he is not writing books, he earns his keep as a Windows developer. Most recently, he had more than his fair share of fun writing Visual C++ code that controls radio frequency spectrum analyzer instruments. Viktor lives with his wife in the capital of Canada, Ottawa, surrounded by several hundred pounds of computing equipment (his), knitting yarn (hers), an unruly cat, and books. He spends his copious amounts of free time managing one of the oldest multiuser games in existence, MUD2, at the Internet site mud2.com.

IN THIS PART
s Using the Development Environment 3

PART Introduction to the Development System

Using the Development Environment CHAPTER 1

Using the Development Environment

1
USING THE DEVELOPMENT ENVIRONMENT

IN THIS CHAPTER
s Using the Compiler 4 s Integrated Development Environments 8 s Using Visual C++ Features 13

Introduction to the Development System PART I

A modern development environment like Visual C++ is a complex collection of tools that can be used in many ways. Just about all compilers (and associated tools such as linkers and librarians) can be used from the command line. Indeed, until recently this was the only way to invoke the compiler. This type of command-line invocation still has its uses for compiling simple programs. Compilers can also be invoked from the graphical environment (often called the Integrated Development Environment, or IDE). This is most useful for large, complex projects; it is also very convenient to have a single user interface that controls the compiler and linker, provides facilities for code editing, and perhaps offers a built-in debugger and other tools as well.

Using the Compiler


If you never used a C/C++ compiler before Visual C++ or one of its modern competitors, you might think that the graphical environment or IDE is the only way to access the compilers functions. That is not so. Frequently it is advantageous to use the compiler from the command line. Many simple applications can be easily recompiled by typing a simple command (cl hello.c, for example) and you dont have to go through the plethora of confusing, often conflicting options that the IDE has to offer. Of course, the downside of this is that you lose the convenience of the IDEs integrated features for compiler optimization, editing, and debugging. Still, command-line tools have their place in the developers toolbox; furthermore, knowing how to use your command-line tools will also make your life significantly easier when you are tweaking the IDEs settings because you know whats going on behind the scenes, so to speak.

Using Command-Line Tools


Compiling C/C++ code is a process of multiple phases. First, the preprocessor analyzes and resolves directives such as #include or #define, evaluates constant expressions, and performs all macro substitution. Second, the preprocessed result is fed to the compiler itself, which places the compiled result into an intermediate file called the object file. Third, the linker takes over; it combines the components from several object files, resolves symbolic references, and places the result, now an executable program or library, into the target file.

NOTE
In most modern compilers, the preprocessor and the compiler proper are thoroughly integrated. Visual C++ is no exception.

Two additional tools are used frequently: the make utility (which, for historical reasons, is called nmake in Visual C++) and the librarian. The make tool is used to automate the process of compiling complex projects efficiently. It uses an instruction file (the makefile) that contains

Using the Development Environment CHAPTER 1

information about the dependencies of the project. For example, if a change to header.h requires a recompile of program.c, the makefile might contain an entry like this:
program.obj: header.h program.c cl -c program.c

1
USING THE DEVELOPMENT ENVIRONMENT

The first of these two lines tells the make tool that program.obj must be re-created if either header.h or program.c has changed (determined by comparing their file date to the date of program.obj ); the second line simply contains the instructions needed to create a new program.obj. Notice that none of this is specific to C/C++ or, indeed, to compiling source code; the make tool is universal. The librarian is used to package multiple object files into a single file, called a library. Thus, when you are using functions from the C runtime library, for example, you dont need to individually reference several hundred object files that are the modules which comprise this library; you can refer to a single library file instead. Figure 1.1 demonstrates the process of creating an executable file.

FIGURE 1.1.
The process of compiling and linking.

Source file (.C, .CPP)

Source file (.C, .CPP)

Source file (.C, .CPP)

Compiler (CL.EXE)

Compiler (CL.EXE)

Compiler (CL.EXE)

Object file (.OBJ)

Object file (.OBJ)

Object file (.OBJ)

Librarian (LIB.EXE)

Library file (.LIB)

Linker (LINK.EXE)

Executable or DLL (.EXE, .DLL, .OCX)

Resources
Regardless of the programming language you use, a Windows program also utilizes resources. Resources define the visual appearance of the program. They include dialog templates, menus, icons, bitmaps, strings, and more.

Introduction to the Development System PART I

Although there are a variety of tools you can use to create an applications resources visually, eventually the result is deposited into a human-readable source file: a resource script. Most development systems like Visual C++ contain a resource compiler; the compiled resource file can then be linked with other object components of your project to create the final executable.

Executables and DLLs


Usually when we speak of the function of a compiler or other development system components, we implicitly assume that the target of the exercise is an executable program, or .EXE file: something that the user can invoke from the command line by typing its name. In reality, in addition to executable programs, a compiler can be used to generate dynamic-link libraries or object modules (which can optionally be packaged into a static library). The execution of your typical, run-of-the-mill C or C++ program begins with the entry-point function, main. Indeed, omitting this function from your program usually results in a linker error. Yet Windows programs have no main function; it is instead replaced with WinMain. How can such programs compile and link without errors? The secret, of course, lies in the startup code that is linked with your program implicitly. Depending on which startup routine is used, it passes execution to either main or WinMain. The startup routine can be selected using compiler switches, but often it is not necessary; modern compilers can automatically identify text-mode programs (console applications in Win32 parlance) and Windows programs and select the appropriate startup code. Dynamic-link libraries are a different issue altogether. Although they, too, contain executable code, they have no entry-point function. Instead, they are merely a collection of functions or other objects (for example, compiled resources) that can be used by other programs. They say a picture is worth a thousand words; in a book about programming, the same must be true for code samples. And just as with pictures, the simpler, the better: If a 5-line example can demonstrate a programming principle without oversimplification, a 5-line example should do! Well, 18 lines in this case; a 7-line source file for a simple DLL and an 11-line program that uses it. The library, shown in Listing 1.1, contains the definition of a single function, hello. Notice that the function definition is preceded with the _ _declspec directive; this is the Microsoftspecific way of instructing your compiler to export this function. Other compilers may use a different syntax, for example a keyword like _ _export.

Listing 1.1. A simple DLL function.


#include <stdio.h> _ _declspec(dllexport) void hello(void) { printf(Hello, World!\n); }

Using the Development Environment CHAPTER 1

This library can be compiled from the command line using cl -LD hdll.c. (Again, this is how the Visual C++ compiler is invoked; from now on, Ill stop mentioning this, just keep in mind that if you use another vendors compiler the syntax might be different.) This creates several files, including hl.dll, which is our new dynamic-link library. This DLL can be loaded into the address space of another program using the LoadLibrary function (part of the Win32 API); the starting address of individual library functions can be obtained using GetProcAddress. A program that loads the library, obtains the address of the hello function, and invokes this function is shown in Listing 1.2.

1
USING THE DEVELOPMENT ENVIRONMENT

Listing 1.2. Using a DLL.


#include <windows.h> void main(void) { HINSTANCE hLib; void (*hello)(void); hLib = LoadLibrary(HL.DLL); hello = (void (*)(void))GetProcAddress(hLib, hello); hello(); }

This program can be compiled by typing cl

hello1.c.

Explicitly loading a library and obtaining the start address of each function within it can be inconvenient and can also be the source of errors. Fortunately, theres an easier way. Notice that when you compiled your DLL the compiler also generated a library file, hdll.lib. This library contains stub implementations for each of the exported functions in your DLL. If you link your executable with this library, you can directly invoke the functions contained within the DLL. So your main program can be simplified, as shown in Listing 1.3.

Listing 1.3. A simpler way to use a DLL.


void hello(void); void main(void) { hello(); }

This time you must tell the compiler which library to link with: cl hello2.c hdll.lib. Note that you also had to declare the function hello before its use; normally, such declarations would be stored in a separate header file that is created and maintained along with the DLL to which it refers.

Introduction to the Development System PART I

Integrated Development Environments


After you have a basic set of command-line tools for compiling and linking programs and a simple editor (even something like EDIT.EXE or the Windows Notepad application will do), technically you can create applications of arbitrary complexity. In reality, however, it is often not practical to use these simple tools when working with complex projects. Nevertheless, it is important to recognize that nothing that a graphical environment does is essential to application development; these, however important, are merely tools of convenience. In everyday practice, two features of an IDE stand out as the most often used by programmers: the integrated editor and the debugger. Modern IDEs also offer a variety of other great features, such as integration with class libraries, support for browsing classes and member variables, or online documentation.

The Integrated Editor


The integrated editor can have features that are specific to the programming language used; for example, it might be capable of highlighting keywords, matching pairs of instruction brackets, or marking the beginning of function definitions. Most importantly, it might also be able to invoke the compiler and other tools and parse error lists that are generated by them, so that problem lines will be automatically marked or highlighted for easy identification.

The Debugger
Not long ago runtime debugging of an application was the exclusive domain of interpreted languages such as BASIC. Of course, it was easy to interrupt and resume the flow of an interpreted program: it was up to the interpreter itself to enable user interaction between executing two program instructions. Unless the users interactions altered the program itself that was being debugged, there was no reason why program execution could not be resumed. Compiled programs are different. They execute as native machine language programs on the target systems processor, not under the control of any runtime interpreter. Indeed, reliable debuggers appeared only after processors began to support their implementation through special instructions and features. Easily the most fundamental feature of a debugger is its capability to interrupt the execution of the program under test, enable the examination and perhaps modification of the programs memory space or the processors registers, and then enable the execution to be resumed. Modern debuggers, of course, do a lot more than that. First, most modern debuggers are language-aware; they can match the machine code that is being executed to corresponding source lines, and enable the examination and modification of variables and expressions by name. This is usually accomplished through compiler support: the compiler adds debugger information to the generated code, so that later the debugger can match program instructions and memory addresses with program symbols and source lines.

Using the Development Environment CHAPTER 1

Another very useful feature is the debuggers capability to place breakpoints in the code and attach specific conditions to them. This enables you to interrupt the programs execution in a controlled manner without having to explicitly add instructions for this purpose, possibly altering the programs behavior. It is also very important that a debugger does not modify the environment in which the target program runs. For this reason, it is sometimes desirable to run debuggers using a second screen or keyboard. Otherwise, even in a multitasking environment like Windows, the debugger may interfere with the program under test. For example, if you are testing a full-screen application that changes display modes, it might not be possible to display a debugger window in the first place, and if you do, it might cause your applications display area to be redrawn, altering the applications program flow.

1
USING THE DEVELOPMENT ENVIRONMENT

Resource Editing
Most integrated development environments today provide a means to edit your applications resources. At the very least, you are provided with a graphical dialog editor; quite possible, the resource editor can do a lot more and can graphically edit menus, toolbars, bitmaps, icons, and more (see Figure 1.2).

FIGURE 1.2.
Editing a bitmap resource.

A modern IDE can also link the visual presentation of resources with your code. For example, in Visual C++ it is possible to jump directly from a dialog template to create a corresponding C++ class, assign member variables that correspond with the dialogs controls, or create functions that handle dialog events.

10

Introduction to the Development System PART I

Project Workspaces
A makefile is often sufficient to manage a simple project (indeed, for really simple ones, like some of the examples presented in this book, its not even necessary to use a makefile) but for more complex projects, the use of makefiles can be cumbersome. This fact, of course, did not escape the attention of designers of integrated environments. In Visual C++ you work with the concepts of projects and workspaces. A project is a collection of files and rules to compile them; in fact, you could almost say that a project is equivalent to an old-fashioned makefile, except that it contains more information. For example, a project can contain several sets of build rules, such as build rules for release versus debug builds, or ASCII versus Unicode builds. A workspace is a collection of projects. For example, a recent package I was working on consisted of two DLLs and two executable programs; these four components existed as four separate projects within the workspace. I also created two additional projects that were not C/C++ projects at all: one was used to automate the creation of the installation file images (using InstallShield 3) and the other was used to create a set of miscellaneous information files in WinHelp format.

Using Project Templates


One of the main strengths of any IDE is its capability to provide a quick start launching point for your application development efforts. Rather than starting from scratch, they enable you to create a skeleton application that already contains many user interface features of your finished product. In Visual C++, these starting points, or project templates, are available through a special tool: the AppWizard. Or I should say AppWizards, because more recent versions of Visual C++ contain several them for a variety of projects (for example, MFC-based applications or ActiveX controls). AppWizards are essentially a sophisticated version of the good old GENERIC.C, the sample application that was used many years ago as an introduction to the basics of Windows programming. Back then, GENERIC.C (which implemented a blank window with a menu bar and some trivial menu functions, such as Exit and About) was used as a starting point for application development. You started with GENERIC.C, began by modifying its menus and adding dialog resources, and proceeded by adding the code supporting the new features. Rather than a single, inflexible skeleton application, Microsofts AppWizards produce a customized, tailored version. You can basically pick and choose from a shopping list of features. (Is it an executable or a DLL? Do you want database support? Is it an ActiveX server? And so on.) AppWizards also automate some of the more mechanical steps of this process. For example, you no longer must manually rename GENERICThis and GENERICThat to MyAppThis or MyAppThat; the AppWizard will do it for you. The result is a skeleton application that, even if it does nothing, already has some of your planned projects look and feel.

Using the Development Environment CHAPTER 1

11

Components
Project templates such as the AppWizards of Visual C++ can be of great value when creating the initial skeleton of your application, but are of little utility afterward. In the old days, this meant that if you wanted to add a specific feature to your application, the solution was to find a suitable example, extract the required code, and insert it in the appropriate place in your program. Of course, this is a fairly mechanical process, so it can be automated. In Visual C++, the Component Gallery provides this functionality. The development system comes complete with a set of standardized components ranging from the simple (such as adding a splash dialog to an MFC application) to the complex (such as adding components to an ActiveX project).

1
USING THE DEVELOPMENT ENVIRONMENT

Miscellaneous Tools
Any modern development system worth its salt comes with a series of miscellaneous tools. Some you could live without, some have high-quality replacements that are available as freeware or shareware, but a few are essential for application development. Perhaps the most often used such tool is the message spy. This tool (Spy++ is the name in the Visual C++ distribution) lets you monitor messages sent to or posted to a specific window or a set of windows in the system. Needless to say, when you are trying to understand the behavior of user interface objects in your application or trying to trace the cause of some unexplained behavior, this tool is invaluable. An equally important tool is the process viewer that lets you see running processes in the system. My favorite is actually a version that does not come with Visual C++: I am talking about the program PVIEW.EXE (the Process Exploder) that comes with the Windows NT Resource Kit (see Figure 1.3). In addition to listing processes, this tool provides a huge amount of information about their memory use, threads, and security settings. Furthermore, this is the only process viewer tool I know of that actually lets you change the security settings of a process. For example, when you are unable to shut down a stubborn Windows NT service application that entered into an endless loop, you might find that even as administrator you do not have the rights to do so. With PVIEW.EXE you can change the security settings of the process, thereby enabling administrators to shut it down. PVIEW.EXE is an NT-only tool, but then again, many of the problems for which it provides a solution do not exist under Windows 95/98. If you are working with COM objects or creating ActiveX controls, you might find the ActiveX Control Text Container or the COM Object Viewer very handy. These programs, distributed with Visual C++ 5 and other development tools, enable you to test your ActiveX controls in a testbench environment, and enable you to monitor COM object behavior. Earlier versions of Visual C++ (up to and including version 4.x) also included the program DISPTEST.EXE. This program, essentially a dumbed-down version of Microsofts Visual Basic programming environment, was very handy for testing Automation projects. This tool was unfortunately removed from the Visual C++ 5 distribution, but I have high hopes that this was merely an oversight, and the tool will be back with Visual C++ 6.

12

Introduction to the Development System PART I

FIGURE 1.3.
The Process Explode from the Windows NT Resource Kit.

Finally, there is Microsofts Help Workshop. If you distribute your application with Windows Help-style help files (this is still the norm even though HTML is quickly becoming a serious alternative), this program can help you compile and debug your help files by providing a graphical front-end to the Windows Help Compiler.

Profiling
Profilers are tools used for performance analysis: in particular, they are used to identify performance bottlenecks in your programs. What profilers do is really simple: During execution of your program (compiled with the appropriate flags) they collect statistical information about various portions of your code. When execution is complete, this statistical information can be analyzed.

Version Control
Most real-life projects today are not the work of a single programmer. When several programmers work on the same project, managing their work and ensuring that they use a consistent set of source files becomes a potentially major problem. Version control systems or revision control systems are designed to solve this problem by ensuring that at any given time, only one programmer has the right to modify a source file. All other programmers receive a read-only copy. The version control system also makes it possible to track ownership of a file, in other words to determine which programmer has checked it out from the version control system.

Using the Development Environment CHAPTER 1

13

Pesky programmers are, of course, well known for their tendencies to ignore rules or circumvent them. If we want a version control system to work well in a production environment, there must be incentives for its use other than threats from the boss. A good version control system is preferably well integrated with the development environment itself, so at the very least it does not make programmers lives any harder. Visual C++ is packaged with Microsofts own version control system, Visual SourceSafe. This product is available with most enterprise versions of Microsofts development tools. The tool is thoroughly integrated with Visual Studio.

1
USING THE DEVELOPMENT ENVIRONMENT

Using Visual C++ Features


The previous section reviewed some of the key features that a modern C/C++ development system is expected to provide. What follows is a more in-depth review of the implementation of some of the most important features in Microsofts Visual C++.

Creating Projects with AppWizards


When you select the New command from Visual Studios File menu, you are presented with the New dialog. When the Files tab is selected in this dialog, you are presented with the choice of creating a new file that can be a text file, a source or header file, bitmap, or any of a variety of other file types (see Figure 1.4).

FIGURE 1.4.
The New dialog with the Files tab selected.

However, if you select the Projects tab (and this is the default if you currently have no workspaces open), you are presented instead with a variety of project types (see Figure 1.5).

14

Introduction to the Development System PART I

FIGURE 1.5.
The New dialog with the Projects tab selected.

Several choices presented here can be used to create skeleton applications for Windows executable programs and for Windows DLLs.

Types of Windows Applications Created Through AppWizard


If you select MFC AppWizard (exe) as the type of your new project, you are presented with step one of a multistep wizard process (see Figure 1.6). In this first step you can determine the basic characteristic of your new application: whether it will have a single-document based, multiple-document based, or dialog-based user interface.

FIGURE 1.6.
MFC AppWizard: Selecting your projects type.

Using the Development Environment CHAPTER 1

15

A single-documentbased application can present only one file to the user at any given time. A good example for such an application is the Windows NotePad. A multiple-documentbased application, in contrast, can present several documents at once, each in its own child window. Many word processing applications, such as Microsoft Word, are multiple-documentbased applications. A dialog-based application presents a single dialog as its user interface. These applications are used when all user interaction can take place through a single dialog template. An example for a dialog-based application is the Windows Character Map.

1
USING THE DEVELOPMENT ENVIRONMENT

NOTE
AppWizard-generated, dialog-based applications offer substantially fewer MFC features than MFC-based SDI or MDI applications. If you plan to use a document class, view class, or other MFC features, consider creating an SDI application based on the CFormView view class instead of creating a dialog-based application.

During this first AppWizard step, you can also specify the language of your application. Your language selection defines which standard MFC resource set will be included with your project.

Document-Based MFC Applications


The procedures to create single- or multiple-documentbased (SDI or MDI) applications through AppWizard are nearly identical, and consist of the same AppWizard steps. The project files created by AppWizard for SDI and MDI projects are somewhat different; in particular, for an MDI project AppWizard generates an additional class, CChildFrame, that represents MDI child windows. After you select one of these document-based options, clicking the Next button takes you to step two of a six-step process. In this step, you must specify the level of database support your application will provide (see Figure 1.7). Database support is in the form of MFCs open database connectivity (ODBC) and data access objects (DAO) classes. AppWizard step three is about support for object linking and embedding and ActiveX features (see Figure 1.8). You can specify whether your application supports OLE compound document functionality as a server, container, mini-server, or container-server. You can also specify ActiveX document server support and add Automation server and ActiveX control container support; the latter is important if you want to use ActiveX controls in your applications dialogs.

16

Introduction to the Development System PART I

FIGURE 1.7.
MFC AppWizard: Selecting database support.

FIGURE 1.8.
MFC AppWizard: OLE features.

Step four of AppWizard contains a variety of miscellaneous options (see Figure 1.9). The check boxes for toolbar support, status bar support, and support for 3D controls require little explanation. Of particular interest in AppWizard step four is the Advanced Options dialog, which is invoked when you click the Advanced button. Through this dialog you can specify a variety of additional options that affect your applications appearance and execution in subtle ways.

Using the Development Environment CHAPTER 1

17

FIGURE 1.9.
MFC AppWizard: Miscellaneous options in step four.

1
USING THE DEVELOPMENT ENVIRONMENT

The Advanced Options dialog is a tabbed dialog. The first tab, Document Template Strings, enables you to specify several string values defining your applications default filename extension, filename filter, file type identifier, and more (see Figure 1.10). These strings are combined and stored in your applications string table (in the resource file) under the identifier IDR_MAINFRAME. For example, the following string corresponds to the settings shown in Figure 1.10:
TEST\n\nTEST\nTEST Files (*.tst)\n.TST\nTEST.Document\nTEST Document

FIGURE 1.10.
MFC AppWizard: Advanced document template string settings.

The second tab in this dialog, Window Styles, can be used to specify the window styles for the applications main frame window and child frame windows. You can also select split window support; when this check box is set, your applications views will have a splitter bar.

18

Introduction to the Development System PART I

Step five of AppWizard enables you to specify two simple options: whether the AppWizardgenerated skeleton source should contain comments, and whether the MFC Library should be linked to your project as a static library or a DLL (see Figure 1.11).

FIGURE 1.11.
MFC AppWizard step five.

The final step in the AppWizard process, step six, lists the classes that the AppWizard is about to create for your application (see Figure 1.12). You can also modify some aspects of class creation at this stage. For example, you can modify the base class of your applications view class to be based on the CFormView or CScrollView class, instead of the default CView.

FIGURE 1.12.
MFC AppWizard: Class overview.

Using the Development Environment CHAPTER 1

19

To complete the AppWizard process, click the Finish button. This displays the New Project Information dialog (which, incidentally, gives you one more chance to back out and cancel the AppWizard procedure); this dialog displays information about the skeleton project. This information is also saved in your project directory as your projects readme.txt file. The classes generated for a basic, single-document base application include a document class, a view class, a frame window class, and a CWinApp-derived class representing the application. The declaration and definition of a CDialog-derived class, representing your applications About dialog, will also be included in the implementation file of your applications CWinApp-derived class. Depending on the additional options you specified (MDI support, database support, OLE features), additional classes may be generated by AppWizard.

1
USING THE DEVELOPMENT ENVIRONMENT

Dialog-Based MFC Applications


The AppWizard process for creating a dialog-based MFC application is much shorter than creating document-based applications. Selecting a dialog-based application in AppWizard step one and clicking on the Next button takes you to step two of a four-step process (see Figure 1.13).

FIGURE 1.13.
MFC AppWizard: Creating a dialog-based application.

Step three of the AppWizard process for dialog-based applications is identical to step five for document-based applications (refer to Figure 1.11). Similarly, step four for dialog-based applications is the same as step six for document-based ones; however, the list of generated classes is different. Only two classes are created: one representing the application object, the other representing the applications dialog. In many cases, you might find that a dialog-based, AppWizard-generated skeleton application lacks the features that you want to see in your program. In such cases, consider using an SDI application based on the CFormView class.

20

Introduction to the Development System PART I

Using AppWizard to Create MFC-Based DLLs


If you use the AppWizard to create an MFC-based DLL skeleton, you are presented with a one-step procedure in which you can specify your DLLs characteristics (see Figure 1.14).

FIGURE 1.14.
MFC AppWizard: Creating an MFCbased DLL.

Other Project Types


The New command in Visual Studios File menu, in addition to letting you create skeleton MFC-executable and DLL projects with AppWizard, also enables you to create many other types of projects (refer to Figure 1.5).

NOTE
Not all project types can appear in your version of Visual C++. Some project types are available only in the Professional and Enterprise versions of the product.

Adding Functionality with the ClassWizard


The ClassWizard is typically invoked from Visual Studios View menu. It can also be invoked from many pop-up menus; for example, you can invoke the ClassWizard from the pop-up menu that appears during dialog editing. In such cases, the ClassWizard is invoked in a special mode. It usually appears with the relevant class (that is, the class associated with the dialog template you were editing) preselected. If such a class does not exist, the ClassWizard is invoked in class creation mode, where a new class for the selected object (dialog) can be created. The ClassWizard appears to the user as a large dialog with five tabs. Each of these tabs represents a special ClassWizard function. The Message Map tab presents options for defining and editing message-handler functions. The Member Variables tab is where you assign member

Using the Development Environment CHAPTER 1

21

variables to dialog controls in classes that are associated with a dialog template and to table columns in database applications. The Automation tab offers a choice of Automation methods and properties of an Automation server application or DLL. The ActiveX Events tab is where you add events to an ActiveX control. Finally, the Class Info tab presents a review of some of the overall characteristics of your class.

1
USING THE DEVELOPMENT ENVIRONMENT

Message Maps
The MFC implements a special mechanism for the processing of Windows messages. These messages, after being received by the applications message loop, travel through the applications CCmdTarget-derived class objects in accordance with a set of specific rules. When an object receives a message it can handle, it does so; alternatively, it can also pass on the message for further routing. Whether an object can handle a message or not depends on whether it has an entry for that particular type of message in its message map. The message map for MFC classes is maintained through the ClassWizard. Figure 1.15 shows the message map for the property page of an ActiveX control. (I decided to use a simple ActiveX control project for the illustrations in this chapter for the simple reason that ActiveX controls have all the necessary classes that enable me to demonstrate all features of ClassWizard.)

FIGURE 1.15.
Editing message maps through ClassWizard.

The list of selectable messages on the right side, in the Messages column, varies depending on which item you pick on the left side. If you select the class name, this list shows all Window Manager (WM_) messages that the class may respond to; if the class has overridable functions, those are also shown here.

22

Introduction to the Development System PART I

If you select a control identifier in the Object IDs column, the list on the right side changes; it shows the list of WM_COMMAND messages that the control can send to its parent. You can add a handler function for a specific message by clicking the Add Function button or double-clicking the message item in the Messages column. For example, when you add a member function to the CMCTLPropPage class (in this example, an event handler for a control, OnDblclkCbshape), the ClassWizard automatically inserts the functions declaration in the header file:
// Message maps protected: //{{AFX_MSG(CMCTLPropPage) afx_msg void OnDblclkCbshape(); //}}AFX_MSG DECLARE_MESSAGE_MAP()

The ClassWizard identifies the location of the message map in your code by the special comments that enclose it. Message map declarations, for example, are marked by comments that begin with //{{AFX_MSG. Under normal circumstances, you should never modify code that is marked by such comments. It is easy to notice these ClassWizard-specific sections of code in your source files; by default, the Developer Studio editor shows these portions of your application with a special color. The message map declaration is not the only place that the ClassWizard touched. It also modified the message map definition in the implementation file of CMCTLPropPage:
/////////////////////////////////////////////////////////////////// // Message map BEGIN_MESSAGE_MAP(CMCTLPropPage, COlePropertyPage) //{{AFX_MSG_MAP(CMCTLPropPage) ON_CBN_DBLCLK(IDC_CBSHAPE, OnDblclkCbshape) //}}AFX_MSG_MAP END_MESSAGE_MAP()

Most importantly, the ClassWizard also added a skeleton implementation of the new function OnDblclkCbshape. This implementation is simply appended to the file; it is your responsibility to move it elsewhere in order to keep your source file well organized:
/////////////////////////////////////////////////////////////////// // CMCTLPropPage message handlers void CMCTLPropPage::OnDblclkCbshape() { // TODO: Add your control notification handler code here }

Member Variables
In the second tab in the ClassWizard dialog, you can add and modify member variables (see Figure 1.16).

Using the Development Environment CHAPTER 1

23

FIGURE 1.16.
Editing member variables with ClassWizard.

1
USING THE DEVELOPMENT ENVIRONMENT

The member variables you can edit here are those that are associated with controls in the dialog template for your class. (Needless to say, the class must represent a dialog or property page, otherwise it would not have a dialog template.) In database applications, member variables associated with table columns can also be edited here. This ClassWizard page lists all the control identifiers with which member variables can be associated. Any existing member variables are also shown. To add a new member variable to a control, select its identifier and click the Add Variable button (or simply double-click the control name). In the Add Member Variable dialog shown in Figure 1.17, you can specify the name and type of the new variable.

FIGURE 1.17.
Adding a member variable.

24

Introduction to the Development System PART I

Generally, two kinds of member variables can be assigned to a control. Either a member variable representing the controls value or a member variable representing the control object can be specified. However, for many control types (buttons, for example), only the latter kind of variable is available. You can select whether the variable is to represent the controls value or the control itself in the Category field. Variables that represent a controls value are typically of a simple type, such as CString or int. Variables that represent controls are of a class representing the controls type; for example, CButton or CComboBox. You can normally add only one member variable for a specific control. However, if a control is of a type that enables member variables representing both the controls value and the object itself, you can add both to the control. Thus, it is possible, for example, to have an edit control with a CEdit member variable through which the controls appearance and behavior are controlled, while at the same time another member variable of type CString represents the text typed into the same control. For ActiveX controls, a new member variable added to the controls property page can represent a control property. This property is identified in the Optional Property Name field. You can assign a user-defined property name to the member variable, or you can select one predefined property from the list presented in this box. When you define a new member variable, the ClassWizard updates your applications source code in several places. For example, adding the m_nShape member variable through the dialog shown in Figure 1.17 is reflected by the following change in the CMCTLPro age header file:
// Dialog Data //{{AFX_DATA(CMCTLPropPage) enum { IDD = IDD_PROPPAGE_MCTL }; int m_nShape; //}}AFX_DATA

The ClassWizard also changed the classs implementation file. Initialization for the new variable has been added to the classs constructor:
CMCTLPropPage::CMCTLPropPage() : COlePropertyPage(IDD, IDS_MCTL_PPG_CAPTION) { //{{AFX_DATA_INIT(CMCTLPropPage) m_nShape = -1; //}}AFX_DATA_INIT }

Another function that is modified by ClassWizard is the classs DoDataExchange member function. In this function, information is exchanged between the control object itself and the variable representing it or its value. In the case of CMCTLPropPage, this function is modified as follows:

Using the Development Environment CHAPTER 1


/////////////////////////////////////////////////////////////////// // CMCTLPropPage::DoDataExchange - Moves data between page and properties void CMCTLPropPage::DoDataExchange(CDataExchange* pDX) { //{{AFX_DATA_MAP(CMCTLPropPage) DDP_CBIndex(pDX, IDC_CBSHAPE, m_nShape, _T(Shape) ); DDX_CBIndex(pDX, IDC_CBSHAPE, m_nShape); //}}AFX_DATA_MAP DDP_PostProcessing(pDX); }

25

1
USING THE DEVELOPMENT ENVIRONMENT

Not only is there a call to the Dialog Data Exchange (DDX) function, facilitating the exchange of data between the control object and the member variable, a Property Page (DDP) function is also called; this function exchanges data between the member variable and the ActiveX control object. In the case of classes representing records in databases, the ClassWizard can make modifications to other functions as well (for example, the DoFieldExchange member function of CRecordSet-derived classes).

Automation
The Automation tab of ClassWizard enables you to add or modify Automation (formerly called OLE Automation) properties and methods (see Figure 1.18). A property is basically a member variable that is exposed to the outside world through the Automation interface; a method is a member function exposed in a similar fashion. In addition to classes that have Automation enabled, classes representing ActiveX controls also have exposed properties and methods.

FIGURE 1.18.
Defining Automation properties and methods.

26

Introduction to the Development System PART I

ActiveX Events
ActiveX events are events generated by ActiveX controls. Events represent a mechanism through which the control can communicate with its container. ActiveX events for an ActiveX control class can be added or modified through the ActiveX Events tab in ClassWizard (see Figure 1.19).

FIGURE 1.19.
ActiveX events.

Class Info
The last page in the ClassWizard dialog is the Class Info page (see Figure 1.20). In this page, various properties of classes are displayed, and certain advanced options can be modified.

FIGURE 1.20.
Class Info options.

Using the Development Environment CHAPTER 1

27

Creating a New Class


By clicking the Add Class button you can add a new class to your application. ClassWizard enables you to add a class using an existing implementation, create a class using a type library file, or create a new class from scratch. Adding a class from an existing implementation really means importing class data to ClassWizard; the class is presumably already part of your project. This function is most useful if you imported a new class to your project by copying its header and implementation files and added them to your project manually. It can also be used to re-create your applications class information file (CLW file) if it has been damaged. This is the file where ClassWizard keeps all class-related information. Adding a class from a type library means creating a new class that wraps the interface described in the type library. For example, you can use this feature to add a class that represents an ActiveX control or Automation object. You can also add a new class from scratch. If you select the New option from the Add Class buttons pop-up menu, you are presented with the New Class dialog (see Figure 1.21).

1
USING THE DEVELOPMENT ENVIRONMENT

FIGURE 1.21.
Adding a new class.

Editing Code
Another common ClassWizard function is the Edit Code function, available in the Message Maps and Automation pages. Clicking the Edit Code button dismisses the ClassWizard dialog, opens the source file corresponding to the function that was most recently selected in ClassWizard, and positions the cursor over the implementation of the selected function. This is a great convenience feature of ClassWizard.

28

Introduction to the Development System PART I

Using the Component Gallery


Complementing the capabilities of AppWizard is the Visual C++ Component Gallery: a repository of a variety of standard and user components. What is a component? A component can be a class complete with its header file, implementation file, and resources; it can also be a smart component supplied by Microsoft or a third-party vendor. The Component Gallery provides a facility to store and manage these components.

Inserting a Component into a Project


You invoke the Component Gallery through the Components and Controls command in the Add to Project submenu of Visual Studios Project menu. It appears in the form of the Components and Controls Gallery dialog box (see Figure 1.22).

FIGURE 1.22.
The Components and Controls Gallery.

To add a specific component to your project, open the folder that contains the component, select the component, and click the Insert button. If you want more information about the component before adding it to your project, click the More Info button. Many standard components display configuration dialogs when you select them for insertion. Components may also examine your applications source code to determine whether the selected component is compatible with your application.

Creating Your Own Components


As you work with Visual C++, you will develop many components that are potential candidates for reuse. For example, you can create a splash dialog, a CDocItem-derived class of document objects, or a customized About dialog with your companys animated logoall of which are reusable components.

Using the Development Environment CHAPTER 1

29

Adding these components to the Component Gallery is easy. All you must do is open your project in ClassView, right-click the class that you want to turn into a reusable component, and select the Add to Gallery command from the pop-up menu.

1
USING THE DEVELOPMENT ENVIRONMENT

Using the Visual C++ Debugger


In order for the symbolic debugger to function, you must compile an application with debugging information. If your application is an MFC application that was originally created through AppWizard, chances are that you do not have to do anything; the AppWizard already created a debug configuration for your project and made it the default configuration. However, if you must create a debug configuration yourself, you can do so in the Project Settings dialog. You must set the appropriate compiler and linker options that will make debugging possible. To set the compiler options, invoke the Project Settings dialog through the Settings command in the Build menu, and select the C/C++ tab. Select the General category, and then select the configuration you want to use as the debug configuration in the left side of the Project Settings window. To enable debugging, you must alter two settings: In the Debug Info field, select Program Database; and in the Optimizations field, select Disable (Debug), as shown in Figure 1.23. If you want to utilize the Visual Studios source browser features, you may also set the Generate Browse Info check box; for AppWizard-generated debug configurations, this check box is turned off by default to save compiler time.

FIGURE 1.23.
Setting up the compiler for debugging.

If you are using the compiler from the command line or from within a custom make file, you might need to set these debugging options manually. To turn off optimization, use the /Od compiler option; to turn on the generation of debugging information, specify the /Zi option.

30

Introduction to the Development System PART I

Another setting that is relevant for debugging specifies that your project be linked with the debug version of the C Runtime Library. This is specified by selecting the Code Generation category and picking the desired debug library in the Use Run-Time Library field (see Figure 1.24). The equivalent compiler command-line option is /MDd (debug DLL), /MLd (debug singlethreaded library), or /MTd (debug multithreaded library).

FIGURE 1.24.
Specifying a debug runtime library.

In addition to compiler settings, linker settings must also be modified. This can also be done from the Project Settings dialog. Select the Link tab and the General category. To turn on the generation of debugging information, set the Generate Debug Info check box (see Figure 1.25).

FIGURE 1.25.
Setting up the linker for debugging.

The command-line equivalent for this option is /debug on the linker command line.

Using the Development Environment CHAPTER 1

31

Summary
The compiler and other tools of any modern development environment can be used from the command line or invoked from within the graphical environment. The command-line method is sometimes preferred for its simplicity; the graphical environment can be used to deal with large, complex projects consisting of multiple components. The most frequently used command-line tools are the compiler proper (which translates source files into object files) and the linker (which combines object files into executable images). Other command-line tools include, for example, the librarian and the make utility. Integrated development environments, in addition to greatly simplifying the rebuilding of large projects, offer a variety of built-in features, including a source code editor, a resource editor, an integrated debugger, project templates, components, and more. In Visual C++, a starting point for your application can be created using AppWizards. In particular, you can use the MFC AppWizard to create a skeleton application based on the Microsoft Foundation Classes. Such an application can then be manipulated using the ClassWizard; this tool can link code segments with user interface elements and automate the process of adding event handlers, member variables, Automation support, and more. Visual C++ projects can be compiled with support for debugging. Visual C++ also offers a versatile profiler, which can be used to locate any performance bottlenecks in your applications.

1
USING THE DEVELOPMENT ENVIRONMENT

IN THIS PART
s Operating System Overview s The Message Loop 49 s Windows, Dialog Boxes, and Controls 63 s Resource Files 97 3

PART

II

s Drawing and Device Contexts 109 s Threads and Processes 139 s Memory Management s File Management 177 193 157

s The Windows Clipboard s The Registry 205 s Exception Handling 219

Under the Hood: Windows and the Win32 API

Operating System Overview CHAPTER 2

35

Operating System Overview


IN THIS CHAPTER
2
OPERATING SYSTEM OVERVIEW 36

s Windows and Messages

s Messages and Multitasking 40 s Windows Function Calls 41 s Platform Differences 48

36

Under the Hood: Windows and the Win32 API PART II

The 32-bit edition of Visual C++ can be used to develop programs for two Win32 platforms: Windows NT and Windows 95/98. Despite the obvious differences between these platforms, they share most essential features. In particular, most simple applications are expected to be compatible with both Windows NT and Windows 95/98 with little or no modification. For this reason, I usually discuss operating system or compiler features without regard to the target operating system; if significant platform differences exist, however, I mention those.

Windows and Messages


Windows is often referred to as a message-passing operating system. At the very heart of the system is the mechanism that translates just about every event (a keypress, a mouse movement, a timer countdown) into a message; typical applications are built around a message loop that retrieves these messages and dispatches them to the appropriate message-handler functions. Messages, although sent to applications, are not addressed to them; they are, instead, addressed to the other fundamental components of the operating system, windows. A window is much more than merely a rectangular area of the computers screen; it also represents an abstract entity through which the user and the application interact with each other.

Applications, Threads, and Windows


What is the relationship between applications and windows? A typical Win32 application consists of one or more threads, which are basically parallel paths of execution. Think of threads as multitasking within a single application; for example, one thread in a word processing application may be processing user input, while another is busy sending a document to the printer. A window is always owned by a thread; a thread may own one or more windows, or none at all. Finally, windows themselves are in a hierarchical relationship; some are top-level windows, others are subordinated to their parent windows. Figure 2.1 illustrates this hierarchy. There are many types of windows in Windowsno pun intended! The most obvious, of course, is the large rectangular area that we typically associate with an application. Also obvious is that a dialog box is a window in its own right; it can be moved around, sometimes sized, maximized, or minimized just like the main window of an application. What is less obvious is that many elements displayed within main windows or dialogs are also windows themselves. Every button, edit box, scrollbar, list box, icon, even the screen background itself is treated as a window by the operating system. A very revealing exercise, if you have not done this before, is spending some time with the Spy++ application that comes with Visual C++. Use its Find Window command from the Spy menu and drag the Finder tool around the screen to find out how even an apparently simple application window can have many window components. Figure 2.2 shows a typical screen under Windows with each of the multitude of windows within it marked by a thick black border.

Operating System Overview CHAPTER 2

37

FIGURE 2.1.
Processes, threads, and windows.

Process 1

Thread 1A

Window

Window

Thread 1B

2
OPERATING SYSTEM OVERVIEW
Window

Process 2

Thread 2A

Thread 2B

Thread 2C

Window

FIGURE 2.2.
The multitude of windows during a typical session.

38

Under the Hood: Windows and the Win32 API PART II

Window Classes
The basic behavior of a window is defined by its window class. The window class carries information about the windows initial appearance; the default icon, cursor, and menu resource associated with the window; and perhaps most importantly, the address of a function called the window procedure. When an application processes messages, it usually does so by calling the Windows function DispatchMessage for each message received; DispatchMessage, in turn, calls the appropriate window procedure by checking the class of the window the message is for. It is the window procedure that actually processes messages sent to that window. There are many standard window classes provided by Windows itself. These system global classes implement the functionality of common controls, for one thing. Any application can use these classes for its windows; for example, any application can implement edit controls by using the Edit window class. Applications can also define their own window classes through the RegisterClass function. This function enables programmers to implement window behavior that is not part of any of the system-supplied global classes. For example, this is how a typical application implements the functionality of its own main window and registers the main windows icon and menu resource. Windows also allows subclassing and superclassing an existing class. Subclassing substitutes the window procedure for a window class with another. Subclassing is accomplished by changing the window procedure address through the SetWindowLong (instance subclassing) or SetClassLong (global subclassing) function. The difference? In the first case, only the behavior of a specific window will change; in the second case, the behavior of all windows of the specified class will be affected. Superclassing creates a new class based on an existing class, retaining its window procedure. To superclass a window class, an application retrieves class information using the GetClassInfo function, modifies the WNDCLASS structure thus received, and uses the modified structure in a call to RegisterClass. Through GetClassInfo, the application also obtains the address of the original window procedure, which it should retain; messages that the new window procedure does not process should be passed to this function. Although the terminology is reminiscent of object-oriented terminology, the concept of a window class should not be confused with C++ concepts (or, in particular, concepts of the MFC library). The concept of window classes predates the use of object-oriented languages in Windows by several years.

Message Types
Messages come in many flavors, representing events at many different levels. Again, the Spy++ tool can help you appreciate the complex message set every single window must process. Use the Spy++ tool to select some simple element, such as a dialog box, to snoop on; then watch

Operating System Overview CHAPTER 2

39

the seemingly endless cascade of messages streaming by in the Spy++ window as you move the mouse over a button in the dialog and click it. For example, clicking the OK button in the Word for Windows About dialog produces 12 different messages such as WM_LBUTTONDOWN, WM_PAINT (indicating that the dialog or parts of it need to be redrawn), WM_WINDOWPOSCHANGING, WM_ACTIVATE, or WM_DESTROY (when the dialog is finally dismissed). In this case just as everywhere else, messages representing every single occurrence, every single action are sent to the window for processing. Fortunately, an application does not have to be aware of the meaning of every single message. Instead of processing all possible messages, an application is free to pick and choose; messages that remain unprocessed are passed to the operating systems default message-handler function. Windows messages consist of several parts. Perhaps it is best to review the MSG structure, shown in Listing 2.1, which is used to represent messages.

2
OPERATING SYSTEM OVERVIEW

Listing 2.1. The MSG structure.


typedef struct tagMSG { HWND hwnd; UINT message; WPARAM wParam; LPARAM lParam; DWORD time; POINT pt; } MSG;

The first element of this structure, hwnd, uniquely identifies the window to which this message has been posted. Every window in Windows has such an identifier. The next element identifies the message itself. This element may have hundreds of different values, indicating one of the many hundreds (literally!) of different messages that Windows applications may receive. Messages can be organized into several groups depending on their function. Message identifiers are usually referred to symbolically (such as WM_PAINT and WM_TIMER) rather than by numeric value; these symbolic values are defined in the standard Windows header files. (You need only include windows.h; it, in turn, contains #include directives for the rest.) By far the most populous group of Windows messages is the group of window management messages. The symbolic identifiers for these messages all begin with WM_. This group is so large, it only makes sense to further subdivide it into categories. These categories include DDE (dynamic data exchange) messages, clipboard messages, mouse messages, keyboard messages, nonclient area messages (messages that relate to the title, border, and menu areas of a window, typically those areas managed by the operating system, not the application), MDI (multipledocument interface) messages, and many other types. These categories are somewhat inexact, not always strictly defined; they simply serve as a tool of convenience for programmers trying to form a mental picture of the large set of window management messages. Nor is the set of WM_ messages fixed; it is constantly growing as new operating system capabilities are added.

40

Under the Hood: Windows and the Win32 API PART II

Other message groups are related to specific window types. There are messages defined for edit controls, buttons, list boxes, combo boxes, scrollbars, list and tree views, and so on. These messages, with few exceptions, are typically processed by the window procedure of the controls window class and are rarely of interest to the application programmer. Applications can also define their own messages. Unique message identifiers can be obtained through a call to the function RegisterWindowMessage. Using private message types enables parts of an application to communicate with each other; separate applications can also exchange information this way.

Messages and Multitasking


In Windows 3.1, the message loop had another important role in the interaction between the application and the operating system: it enabled the application to yield control. Windows 95/ 98 and Windows NT applications no longer need to yield control this way.

Message Queues
The preemptive multitasking employed in Windows NT and Windows 95/98 requires a more complex mechanism than the single message queue used in 16-bit Windows. In these preemptive operating systems, the orderly cooperation of competing tasks or threads is not guaranteed. Two or more threads can quite possibly attempt to access the message queue at the same time; furthermore, as task switching is no longer dependent on the next available message in the queue, there are no guarantees that a task would retrieve only the messages addressed to it. This is just one of a number of reasons why the single message queue of 16-bit Windows has been separated into individual message queues for each and every thread in the system.

Processes and Threads


The subject of threads came up briefly during the discussion of the relationship of processes and threads versus windows, earlier in this chapter. In a non-multithreaded operating system, such as many flavors of UNIX, the smallest unit of execution is a task or process. The task-scheduling mechanism of the operating system switches between these tasks; multitasking is accomplished between two or more processes. If an application needs to perform multiple functions simultaneously, it splits itself into several tasks (for example, by using the UNIX fork system call). This approach has some severe drawbacks: tasks are a limited resource (most operating systems cannot handle more than a few hundred simultaneously executing tasks), spawning a new task consumes a prodigious amount of time and system resources, and the new task loses access to its parents address space. In contrast, in a multithreaded system the smallest unit of execution is a thread, not a process. A task or process may consist of one or more threads (usually one designated as the main thread). Setting up a new thread requires little in terms of system resources; threads of the same process

Operating System Overview CHAPTER 2

41

have access to the same address space; switching between threads of the same process requires very little system overhead. In fact, I cannot think of any drawbacks a multithreaded operating system has when contrasted with a single-threaded one.

Threads and Messages


Earlier I indicated that ownership of windows is assigned to individual threads. Correspondingly, each thread has a private message queue, in which the operating system deposits messages addressed to windows the thread owns. Does this mean that a thread must own at least one window and contain a message loop? Fortunately, no; otherwise, the use of threads in typical programming situations would become cumbersome indeed. Threads can exist that own no windows and have no message processing loop whatsoever. Consider, for example, a sophisticated mathematical application in which a complex calculation needs to be performed on every element of a two-dimensional array (a matrix). The easiest way to do this is to implement a loop in which the calculation is performed repeatedly. Under 16-bit Windows, this approach was strictly forbidden; during the execution of the loop, no other applications could control the processor, and the computer effectively froze. In Win32, however, it is perfectly legitimate to set up a separate thread in which such a calculation is performed while the applications main thread continues processing any messages the application may receive. The only effect on the system is a performance hitsomething not entirely unexpected when a complex, processing-intensive calculation is being performed. The thread doing the calculations has no windows, no message queue, no message loop; it does one thing only, and that is the calculation itself. In MFC, these threads acquire a name of their own; they are called worker threads, in contrast to the more sophisticated, message queue processing user-interface threads.

2
OPERATING SYSTEM OVERVIEW

Windows Function Calls


Although the existence of a message loop is perhaps the most distinguishing characteristic of Windows applications, it is by far not the only mechanism through which an application and Windows interact. Windows, like other operating systems, offers a humongous number of system calls to perform a wide variety of tasks, including process control, window management, file handling, memory management, graphics services, communications, and many other functions. The core set of Windows system calls can be organized into three major categories. Kernel services include system calls for process and thread control, resource management, and file and memory management. User services include system calls for the management of user-interface elements, such as windows, controls, dialogs, or messages. Graphics Device Interface (GDI) services provide device-independent graphics output functionality.

42

Under the Hood: Windows and the Win32 API PART II

The Windows system also includes many miscellaneous Application Programming Interfaces (APIs). Separate APIs exist for a multitude of tasks; examples include MAPI (Messaging API), TAPI (Telephony API), or ODBC (Open Database Connectivity). The degree to which these APIs have been integrated into the core system varies. For example, the Component Object Model (COM, the foundation for ActiveX and OLE, object linking and embedding), although implemented in the form of a series of system dynamic link libraries, or DLLs, is nevertheless considered part of the core Windows functionality. Other APIs, such as WinSock, are considered extras or add-ons. This distinction between what is core and what isnt is fairly arbitrary. Indeed, from the perspective of an application there is little difference between a core API function that is part of the Kernel module and a function that is implemented in a DLL. Nothing illustrates this better than the conventions used to invoke API functions from the Visual Basic programming language. All API functions are declared identically, as functions in an external DLL. The only difference is the module name: Kernel in the case of a kernel system call, and the name of the DLL in the case of a call to a DLL function.

Kernel Services
Kernel services typically fall into the categories of file management, memory management, process and thread control, and resource management. Although far from being an exhaustive list, these categories accurately describe most commonly used Kernel module functions. The preferred method of file management differs from what is typically used in C/C++ programs. Instead of accessing files through the standard C library functions for stream or lowlevel I/O, or through the C++ iostream class, applications should utilize the Win32 concept of a file object and the rich set of functions associated with those. File objects enable accessing files in ways that are not possible using the C/C++ libraries; examples include overlapped I/O and memory mapped files used for intertask communication. In contrast, the memory management requirements of most applications are completely satisfied through the C malloc family of functions or the C++ new operator; in a Win32 application, these calls automatically translate into the appropriate Win32 memory management system calls. For applications with more elaborate memory management requirements, sophisticated functions exist for managing virtual memory; for example, these functions can be used to manipulate address spaces that are several hundred megabytes in size by allocating but not committing memory. The most important facet of process and thread management concerns synchronization. This problem is new to the Windows environment because it was not encountered in 16-bit Windows. Under the cooperative multitasking regime of Windows 3.1, applications give up control only at well-defined points during their execution; the execution of competing tasks is synchronous. In contrast, in the preemptive multitasking environment, processes and threads cannot deduce knowledge about the execution status of competing threads. To ensure that

Operating System Overview CHAPTER 2

43

competing threads that are mutually dependent execute in an orderly fashion, and to avoid deadlock situations where two or more threads are suspended indefinitely, waiting for each other, a sophisticated synchronization mechanism is required. In Win32, this is accomplished through a variety of synchronization objects that threads can use to inform other threads about their status, protect sensitive areas of code from reentrant execution, or obtain information about other threads or the status of other objects. Speaking of objects, in Win32, many kernel resources (not to be confused with user-interface resources) are represented as kernel objects. Examples include files, threads, processes, and synchronization objects. Objects are typically referred to through handles; some functions exist for the generic manipulation of objects, while others manipulate objects of a specific type. Under Windows NT, objects also have security-related properties. For example, a thread cannot manipulate a file object unless it has appropriate permissions that match the file objects security properties. The Kernel module also provides functions to manage user-interface resources. These resources include icons, cursors, dialog templates, string resources, version resources, accelerator tables, bitmaps, and other user-defined resource types. Kernel system calls are not aware of the purpose of a resource; however, they provide functionality to allocate memory for resources, load resources from a disk file (typically, the applications executable file), and purge resources from memory. Some areas of Kernel module functionality are specific to Windows NT. For example, the NT Kernel module provides a variety of functions through which the security attribute of kernel objects can be examined and manipulated. Another NT-specific area is tape backup functionality. Calls are available for erasing and formatting a tape and for reading and writing tape contents. Accessing the contents of initialization files (INI files) is also accomplished through Kernel module calls such as WriteProfileString or GetPrivateProfileString. Use of these functions is not recommended, however; instead, new applications should use the Windows Registry for storing initialization information. The Kernel module also provides the functionality required for 32-bit text-only programs called console applications. At first sight, these programs appear as plain old DOS programs; in reality, these are full-featured 32-bit applications that run from the command line and do not use the Windows graphical interface. Nevertheless, these applications can still access a rich set of Win32 system calls; for example, a console application can use virtual memory functions or it can be a multithreaded program. There are many other areas of Kernel module functionality, ranging from the simple (such as operations on large integers) to the complex (such as the use of named pipes).

2
OPERATING SYSTEM OVERVIEW

44

Under the Hood: Windows and the Win32 API PART II

User Services
The User module, as its name implies, provides system calls that manage elements and aspects of the user interface. These include functions that handle windows, dialogs, menus, text and graphics cursors, controls, the clipboard, and many other areas. In fact, it is through User module functions that awareness of these high-level components of the user interface becomes possible. The Kernel module provides memory allocation, thread management, and other services required for windows to function; the GDI module provides graphics primitives; but it is the User module that integrates these two areas and provides the concept of a window, for example. Window management calls include functions to manage a windows size, position, appearance, and window procedure, as well as functions to enable or disable a window and to obtain information about windows. These functions are also used to manage controls, such as buttons, scrollbars, or edit boxes. The User module also contains functions to manage multipledocument interface (MDI) child windows. Menu-related calls in the User module provide functionality to create, display, and manipulate menus, menu bars, and pop-up menus. Through a family of User module functions, applications can manage the shape and appearance of the graphics cursor (the mouse cursor) and text cursor (the caret). Management of the Windows clipboard is also accomplished through User module functions. The Windows clipboard is basically a simple mechanism through which applications can exchange data. An application can place data in the clipboard in a variety of public or private clipboard formats; other applications can examine the clipboard and retrieve data in any of the available formats they can interpret. Most applications provide a set of Edit menu commands (Cut, Copy, Paste) for the explicit manipulation of clipboard contents. The User module also provides functions for the management of messages and thread message queues. Applications can use these calls to check the contents of their message queues, retrieve and process messages, and create new messages. New messages can be either sent or posted to any window. A message that has been posted is simply entered into the message queue of the thread that owns the destination window. In contrast, sending a message directly invokes the window procedure of the destination window; the SendMessage function does not return until the destination window has processed the message. Not only does this mechanism bypass the message queue, it also makes it possible for the sending application to obtain a return value before continuing.

GDI Services
Graphics Device Interface functions are typically used to perform primitive deviceindependent graphics operations on device contexts. A device context is essentially an interface to a specific graphics device. It can be used to obtain information about the device and also perform graphics output to the device.

Operating System Overview CHAPTER 2

45

The information that can be obtained through a device context describes the device in detail. The technology of the device (for example, vector or raster), its type, name, resolution, color capability, font capability, and so on, can all be obtained through appropriate device context calls. Graphics output is performed through a device context by passing the handle of the device context to the appropriate GDI output function. Through the device context, a generic, device-independent graphics call is translated into a set of instructions that realize the output on the specific device. For example, when an application calls the GDI function Ellipse, the device context determines which device driver will actually execute the call; the device driver, in turn, may further refer the call to a hardware accelerator, if the video subsystem has such an accelerator capability. GDI device contexts can describe a wide variety of devices. Typical device contexts include display device contexts (for output that goes directly to the computers screen), memory device contexts (for output into a bitmap stored in memory), or printer device contexts (for output that eventually gets translated into printer control codes and sent to the printer). A very special kind of a device context is the metafile device context that enables applications to make a permanent record of GDI output calls. Such a record is device-independent and can be played back on any device later. More than a mere convenience feature, metafiles play a crucial role in the device-independent representation of embedded OLE objects, the very mechanism that makes OLE objects portable and enables container applications to display or print them even in the absence of the server application. Drawing into a device context usually takes place through logical coordinates. Logical coordinates describe objects using device-independent real-world measurements; for example, a rectangle can be described as two inches wide and one inch high. The GDI provides the necessary functionality for the mapping of logical coordinates to physical coordinates. Of the large number of GDI functions, perhaps the ones used most frequently are those that draw various objects; examples include the Rectangle, Ellipse, Polygon, or TextOut functions. (These are just a few representative cases; the actual number of these functions is very large.) Other frequently used drawing functions are the bit blit functions that are used to quickly and efficiently copy bitmaps. (Well, maybe not that quickly and efficiently; for applications, such as games, which really require blazing speed, there is a faster, albeit less safe, set of bitmap manipulation functions in the DirectX SDK.) Other functions manage device contexts. Device contexts for various devices can be created and destroyed, their state can be saved and reloaded, or information about them can be obtained through these functions. GDI functions can also be used to manipulate palettes. This is mostly useful for applications that strive to achieve color fidelity on devices that offer a limited number of simultaneous colors256 colors, for example.

2
OPERATING SYSTEM OVERVIEW

46

Under the Hood: Windows and the Win32 API PART II

Yet another GDI feature is the ability to create and manage GDI objects. Brushes, pens, fonts, bitmaps, or palettes can be created and selected into device contexts to determine the appearance of shapes that are drawn subsequently. Other functions exist to manage two types of metafiles (the old-style Windows metafiles and the new, enhanced metafiles). Metafiles can be created, saved, reloaded, and replayed into any device context. The GDI module also provides the capability to manage regions and clipping. Clipping is of utmost importance in the Windows environment because it enables applications to draw to a display surface without regard to the boundaries of the surface (a client window, for example), or the possibility that parts of the surface are obscured by other objects on the screen.

Other APIs
Windows is much more than the capabilities implemented in the three core modules. Many other modules, many other APIs existeach implementing another specific area of functionality. Here are some of the more commonly used APIs, many of which are discussed in substantially more detail later: s Common control functions are used to manipulate common controls, including common controls introduced with Windows 95. s Common dialogs include system-supplied dialogs for opening a file for reading or writing, selecting a color from a color palette, selecting a font from the set of fonts installed on your system, and specifying a search or search-and-replace operation. These dialogs can be used as is, or their functionality can be modified through new dialog templates and window procedures. s MAPI, or the messaging applications programming interface, gives applications access to messaging functions through mail delivery systems like the Internet or Microsoft Exchange. s DirectX is the highly efficient interface for graphics, sound, and multimedia. s The COM API is a very rich collection of system calls implementing the core of ActiveX. s TAPI is the Telephony API. Applications can use TAPI for a device-independent method of accessing telephony-based resources (modems, fax-modems, or voicemessaging hardware). There are several areas of network-related functionality; examples include WinSock, the Windows Sockets library; WinInet, the Windows Internet API; RAS, the Remote Access Service; and RPC, the Remote Procedure Call library.

Operating System Overview CHAPTER 2

47

Error Reporting
Many Windows functions use a common mechanism for error return. When an error occurs, these functions set a thread-specific error value that can be retrieved by calling the GetLastError function. The 32-bit values returned by this function are defined in the header file winerror.h or in library-specific header files. Functions in your application can also set this error value by calling SetLastError. Application-specific error codes should have bit 29 of the value set; error codes with this bit set are reserved by the operating system for application-specific use.

Using Standard C/C++ Library Functions


Win32 applications can also use the standard set of C/C++ library functions, although some limitations apply. First and foremost, a Windows application does not normally have access to the traditional stdin, stdout, or stderr streams, the corresponding DOS file handles (0, 1, and 2), or C++ iostream objects (cin and cout). Only text-based console applications can use these standard file handles. (However, Windows applications, too, may have a standard input or standard output open if they are launched with their I/O redirected.) As I mentioned already, Windows applications should use the Win32 file management functions for file handling. This is not to say that the standard C stream and low-level I/O functions or the C++ iostream library are no longer available; it is simply that these libraries do not have all the capabilities available through the Win32 API. For example, the C/C++ library functions are not aware of a file objects security properties; nor can they be used for asynchronous, overlapped input and output operations. Applications should also refrain from using the MS-DOSstyle C library process control functions (the exec family of functions) in favor of CreateProcess. Most other C/C++ libraries can be used without restrictions. In particular, although the Win32 API offers a richer function set for memory manipulation, most applications do not require any services more sophisticated than those offered by malloc or the C++ new operator. The C/C++ math, buffer, string manipulation, character and byte classification, and data conversion routines, to name a few, can also safely be used. Win32 applications should not attempt to access MS-DOS Interrupt 21 or IBM PC BIOS functions. The APIs that were available for this purpose in 16-bit Windows have been removed. Applications that require low-level access to system hardware are probably best developed using the appropriate DDK (device driver development kit).

2
OPERATING SYSTEM OVERVIEW

48

Under the Hood: Windows and the Win32 API PART II

Platform Differences
Although the Win32 API is intended to serve as a platform-independent API, many platform differences exist due to limitations in the underlying operating system. Although both the Windows NT and Windows 95/98 platforms evolve in leaps and bounds, generally Windows NT offers a more complete implementation of the Win32 API and features. Windows NT offers Unicode support, advanced security features, server-side functionality and system-level support for tape backup. Beginning with version 4.0, NT has become almost as responsive and as fast with graphics as Windows 95/98. The strength of Windows 95/98 is compatibility with old software and its capability to function well with older, lower-end hardware. As a development platform, both Windows NT and Windows 95/98 can work well with Visual C++. In particular, a well-configured Windows 95/98 system is not far behind Windows NT in terms of stability.

Summary
Visual C++ is a development system that targets the Win32 environment: specifically, Windows 95/98 and Windows NT. At the core of every Windows application is its message loop. The Windows operating system delivers information about a variety of events in the form of messages to cooperating applications, which in turn process those messages by dispatching them to the appropriate window procedure. A window is a rectangular area in the screen; it is also an abstract entity that receives and processes messages. Windows are owned by threads, which are simultaneous paths of execution within an application. Threads, in turn, are owned by processes or applications. Applications also interact with Windows by calling one of the many operating system functions implemented either in the core of Windows or as a variety of add-ons. The core can roughly be divided into three categories: the Kernel module provides memory, file, and process management; the User module manages user-interface elements (specifically, windows) and handles messages; the GDI module provides graphics services. Other modules implement specific areas of functionality, such as COM, MAPI, networking, common controls and dialogs, and multimedia. With some caveats, Visual C++ applications can also use standard C/C++ library functions. Windows NT offers the more complete implementation of the Win32 API and functionality. Windows 95/98 offer a very rich subset while at the same time offering improved compatibility with older, 16-bit (DOS and Windows) applications and lower-end hardware.

The Message Loop CHAPTER 3

49

The Message Loop


IN THIS CHAPTER
s The Real Hello, World Program 50 s A Simple Message Loop: Sent and Posted Messages 51 s Window Procedures 53 s Comparison with generic.c 56 s Multiple Message Loops and Window Procedures 57

3
THE MESSAGE LOOP

50

Under the Hood: Windows and the Win32 API PART II

It has often been said sarcastically about Windows programming that something must be wrong with an operating system that requires hundreds of lines of code for the simplest program of all, one which displays Hello, World! and does nothing else. But is it really true, or is it just another urban legend about the evil company Microsoft and its monstrous operating system contraption called Windows?

The Real Hello, World Program


Consider the dialog shown in Figure 3.1. Make a guess: How many lines of C code, how many resource file lines, and so on, were required to create an application that displayed just this dialog, nothing else? How many times longer is this program than the infamous original Hello, World program that appeared at the beginning of Chapter 1 of the Kernighan-Ritchie C bible?

FIGURE 3.1.
The simplest Hello, World application under Windows.

You guessed it right. The number of lines (not counting the blank line inserted for cosmetic purposes only) is five in both cases. The Windows version of the Hello, World program is shown in Listing 3.1.

Listing 3.1. Source of the simplest Hello, World program, hello.c.


#include <windows.h> int WINAPI WinMain(HINSTANCE d1, HINSTANCE d2, LPSTR d3, int d4) { MessageBox(NULL, Hello, World!, , MB_OK); }

Compiling this program is no more difficult than compiling the original Hello, World program from the command line. (Let this also serve as a little preview on using the Visual C++ compiler from the command line.) In order for the compiler to work from the command line on Windows 95/98, it is necessary to first run the batch file VCVARS32.BAT, which Visual C++ creates in its binary files directory Program Files\DevStudio\VC\bin during installation. It might also be necessary to enlarge the environment space allocation for DOS boxes to avoid any Out of Environment Space errors this batch file may cause.) This is not a problem under Windows NT if you allowed the Visual C++ installer to register environment variables during installation. With the environment variables in place, all you have to do is type cl hello.c user32.lib and voil! The program hello.exe is ready to be executedwhich you can do by simply typing hello at the command line; both Windows 95/98 and Windows NT can launch Windows applications this way.

The Message Loop CHAPTER 3

51

For all its simplicity, the behavior of this application, if it can be thought to deserve that title, is surprisingly complex. Unlike its plain C counterpart, this application not only displays the message, it interacts with the user in a complex fashion. After initially displaying its message, the program stays alive on the screenmeaning it can be moved around with the mouse. The mouse cursor can be clicked on the OK button to dismiss the program; alternatively, by clicking the mouse over the OK button and keeping the mouse button depressed, you can watch the OK button change its appearance as you move the mouse cursor over it. The window also has a simple menu that can be invoked by pressing Alt+Space or, under Windows 95/98, clicking its title area with the right mouse button; the single Move command can be used to change the windows position using the clipboard. Finally, the application can also be dismissed by using the Enter or Escape keys. Remarkably rich behavior from a five-line piece of code, dont you think? But where does all this complexity come from? The secret lies in the magic words message loop and window procedure ; unfortunately, a five-liner is not exactly a very revealing piece of programming excellence when it comes to understanding Windows application behavior. We have to move on to something more complex.

A Simple Message Loop: Sent and Posted Messages


The problem with our first Hello, World program is its simplicity. The MessageBox call at the center of this application encompasses (and hides!) a lot of functionality. In order to better understand what is taking place, we have to make this functionality more visible; in other words, we have to create a window that we manage ourselves, instead of letting the MessageBox function do this for us. The new version of hello.c is shown in Figure 3.2. This time, the Hello, World! text appears as part of a button that occupies the entire client area. (It also appears in the windows title bar.) I employed a few unorthodox shortcuts to keep the application as simple as possible; this allows us to focus on the issues at hand instead of getting bogged down with irrelevant details.

3
THE MESSAGE LOOP

FIGURE 3.2.
Version of Hello, World with a simple message loop.

A typical Windows program, during initialization, registers a window class first, and then creates its main window using the newly registered class. For now, I wanted to avoid having to register a new class and all that comes with it (such as writing a window procedure); instead, I

52

Under the Hood: Windows and the Win32 API PART II

decided to use one of the existing window classes, the BUTTON class. The functionality of this class does not allow me to identically reproduce the behavior of the previous version of hello.c, but that is not the purpose anyway; our goal is to demonstrate the function of a very simple message loop. This new version of the application is shown in Listing 3.2.

Listing 3.2. Source of Hello, World with a simple message loop.


#include <windows.h> int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE d2, LPSTR d3, int d4) { MSG msg; HWND hwnd; hwnd = CreateWindow(BUTTON, Hello, World!, WS_VISIBLE | BS_CENTER, 100, 100, 100, 80, NULL, NULL, hInstance, NULL); while (GetMessage(&msg, NULL, 0, 0)) { if (msg.message == WM_LBUTTONUP) { DestroyWindow(hwnd); PostQuitMessage(0); } DispatchMessage(&msg); } return msg.wParam; }

While brevity is not the point of this exercise, I cannot help but point out that the application is still far from the legendary hundreds of lines that displaying Hello, World! is supposed to take; even with my publishers formatting requirements, it still fits conveniently on a traditional text screen of 25 80-column lines. This example reveals the infamous message loop. After creating its window, the program enters into a while loop where it makes repeated calls to the GetMessage Windows function. Whenever the application receives a message, GetMessage returns; its return value is FALSE only if the message received was a WM_QUIT message. (This and other symbolic constants are defined in the header files, such as windows.h.) The special case of a WM_LBUTTONUP message is handled; the call to DestroyWindow causes the applications window to be destroyed, while the call to PostQuitMessage ensures that the GetMessage function receives a WM_QUIT message, which causes the loop to terminate. Messages other than WM_LBUTTONUP are dispatched through the DispatchMessage function. Dispatched through the DispatchMessage functionbut dispatched to whom? A good question. These messages are in fact dispatched to the default window procedure of the BUTTON class. As in the case when we called the MessageBox function, this window procedure remains hidden from us, as it is implemented as part of the operating system.

The Message Loop CHAPTER 3

53

The call to the DestroyWindow function in the block that handles WM_LBUTTONUP messages in fact creates another message, a WM_DESTROY message. To terminate the application, instead of calling PostQuitMessage from the handler of WM_LBUTTONUP, a seemingly more elegant solution would be to include the call to PostQuitMessage in another case block that responds to these WM_DESTROY messages. The problem is that WM_DESTROY messages are typically not posted but sent to the application. There is a subtle, but crucial, difference. When a message is posted, the application retrieves it through a GetMessage or PeekMessage call at the time of its own choosing. (PeekMessage is used when the application wants to perform some tasks when it has no messages to process.) In contrast to posting, sending a message to an application implies a direct, immediate call to the window procedure, bypassing any message loops. So in this case, the WM_DESTROY message that is generated in response to the call to DestroyWindow is never seen by the GetMessage loop; instead, it is passed directly to the window procedure of the BUTTON window class. This example still failed to show us the innards of the window procedure. Therefore, it is time to move on to yet another, more sophisticated version of our Hello, World! program. But do not despairwe are still far from hundreds of lines!

Window Procedures
The new version of hello.c, shown in Listing 3.3, registers its own window class. Done in part for cosmetic reasons (so we can do away with the ugly kludge of using the BUTTON class for our purposes), the most important reason for this is so that we can install our own window procedure.

3
THE MESSAGE LOOP

Listing 3.3. Source of Hello, World with a new window class.


#include <windows.h> void DrawHello(HWND hwnd) { HDC hDC; PAINTSTRUCT paintStruct; RECT clientRect; hDC = BeginPaint(hwnd, &paintStruct); if (hDC != NULL) { GetClientRect(hwnd, &clientRect); DPtoLP(hDC, (LPPOINT)&clientRect, 2); DrawText(hDC, Hello, World!, -1, &clientRect, DT_CENTER | DT_VCENTER | DT_SINGLELINE); EndPaint(hwnd, &paintStruct); } }

continues

54

Under the Hood: Windows and the Win32 API PART II

Listing 3.3. continued


LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_PAINT: DrawHello(hwnd); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); return msg.wParam; }

I can almost sense your outrage: What? Sixty-four lines? Worse yet, to compile this program successfully you actually have to specify the gdi32.lib library on the command line? (Compile with cl hello.c user32.lib gdi32.lib.) Rest assured, this is as far as we go; our text version of Hello, World will not get any more complex at this time. And at 64 lines, this is a full-featured Windows application (see Figure 3.3); it has a system menu; it can be moved, resized, minimized, and maximized; it knows when

The Message Loop CHAPTER 3

55

to redraw itself; and it responds to the Close menu item or the Alt+F4 keystroke. Not bad for a program that, when listed, still fits easily on a printed page!

FIGURE 3.3.
Version of Hello, World with its own window class.

So how does this work? As before, execution starts with the WinMain function. The first thing the application does is check whether it has any copies already running. If so, there is no need to re-register its window class. Otherwise, the window class is registered, its properties and behavior determined through the WndClass structure. In particular, it is through the WndClass structure that the address of the window procedure, WndProc, is given. Next, an actual window is created through the CreateWindow system call. After the window is displayed, WinMain enters the message loop; the message loop exits when GetMessage returns FALSE upon receiving a WM_QUIT message. Finally, through WndProc, the purpose and structure of the mysterious window procedure are revealed. A typical window procedure is nothing but a giant switch statement. Depending on the message received, different functions are called that perform the necessary action. The code in Listing 3.3 processes exactly two messages: WM_PAINT and WM_DESTROY.
WM_PAINT messages indicate that parts or all of the applications window must be redrawn. Sophisticated applications would normally not redraw parts of the application window for which redrawing has not been requested; in our case, we simply dont care, we just redisplay the Hello, World! text whenever a WM_PAINT message is received.

3
THE MESSAGE LOOP

messages are received in response to user actions that cause the applications window to be destroyed. Our response to this is a call to PostQuitMessage; by doing this, we ensure that the GetMessage function in WinMain receives a WM_QUIT message, causing the main message loop to terminate.
WM_DESTROY

56

Under the Hood: Windows and the Win32 API PART II

What happens to messages that are not processed by our window procedure? They are instead passed to the default window procedure, DefWindowProc. This function determines the behavior of the applications window and many of its nonclient area components (such as its title bar) through the default handling of messages that it provides. A companion to DefWindowProc is DefDlgProc. This default window procedure is specifically designed for windows that are dialog windows. This function provides handling for dialogspecific messages and also provides default management of the dialogs controls for cases when the dialog loses or gains focus. If you have access to the Windows 3.1 SDK (perhaps through a subscription to the Microsoft Developer Network, level 2), it may be an educational exercise to look at the DEFPROC sample; this sample is nothing else but the source code of the two default window procedures, DefWindowProc and DefDlgProc.

Comparison with generic.c


So if it is this easy to write a Hello, World program with just a few lines of code, what is the explanation for the common myth that even a simple program like this takes several hundred lines of code in Windows? The answer to this curious question can be found in how Microsoft presented its Windows Software Development Kit, or SDK, back in the good old days. The tutorial centerpiece of the old SDK was the Generic Windows Application, a program that did nothing other than display a window with standard decorations and provide an About, a Help, and an Exit function. This program successfully served as the skeleton for many well-written Windows applications. For all its simplicity, the generic.c source code was still almost 500 lines long, with another nearly 200 lines in its resource file, generic.rc. No wonder programmers in the old days found Windows programming forbidding. Nevertheless, even in those days it was not necessary to reproduce these hundreds of lines of code from scratch every time you embarked upon a new Windows project. On the contrary, generic.c was used as a starting point, and it provided a default implementation for all the basic mechanics of a Windows application. It also influenced both the visual appearance and code style of Windows applications; for example, although not strictly necessary, most Windows applications ended up having separate InitApplication and InitInstance functions. The task of the Visual C++ programmer is much easier nowadays, although really not that different. Instead of starting from a static skeleton, generic.c, MFC programmers start with an application skeleton dynamically created by the Visual C++ AppWizard. But the fact that

The Message Loop CHAPTER 3

57

you start off from a skeleton application that implements a framework for the basic mechanics of your application has not changed. In my opinion, the availability of a well-written skeleton application had a significant positive influence on Windows programming.

Multiple Message Loops and Window Procedures


There was a single message group in all the examples that we have built so farnamely, the three versions of hello.c. (Well, in the case of the first one, the presence of the message loop was implicit in the MessageBox function call.) Can there be applications with more than one message loop? And if so, why would you want to write an application that way? The answer to the first question is a sound yes. Applications can have as many message loops as they want. Consider the simplest of these situations, when an application that has its own message loop makes a call to the MessageBox function; would this call not imply that, temporarily, the implicit message loop in MessageBox takes over the processing of messages while the message is displayed? This scenario also suggests an answer to the second question. You would implement a second (or third, or fourth) message loop when, during a particular stage of execution of your program, messages must be processed in a fashion that is markedly different from normal processing. Consider, for example, the case of drawing with mouse capture. An application can provide a freehand drawing capability by looking for mouse events and capturing the mouse when the left button is pressed within its client area. While the mouse is captured, the application is informed of every mouse movement through a separate message; thus, the application can draw a freehand line by adding a new segment whenever the user moves the mouse. The mouse is released when the user releases the left mouse button. Our fourth and final version of Hello, World, shown in Figure 3.4, is exactly such an application. This 94-line enormity, which must be compiled with the command line cl hello.c user32.lib gdi32.lib, although far less elegant than the Visual C++ Scribble tutorial, actually allows you to draw the text Hello, World! using the mouse cursor. Even a cursory glance at the applications source code, shown in Listing 3.4, reveals the existence of two while loops with calls to the GetMessage function. The main message loop found in WinMain is not different from before; the new stuff is in the DrawHello function.

3
THE MESSAGE LOOP

58

Under the Hood: Windows and the Win32 API PART II

FIGURE 3.4.
A graphics version of Hello, World.

Listing 3.4. Source of the graphics version of Hello, World.


#include <windows.h> void AddSegmentAtMessagePos(HDC hDC, HWND hwnd, BOOL bDraw) { DWORD dwPos; POINTS points; POINT point; dwPos = GetMessagePos(); points = MAKEPOINTS(dwPos); point.x = points.x; point.y = points.y; ScreenToClient(hwnd, &point); DPtoLP(hDC, &point, 1); if (bDraw) LineTo(hDC, point.x, point.y); else MoveToEx(hDC, point.x, point.y, NULL); } void DrawHello(HWND hwnd) { HDC hDC; MSG msg; if (GetCapture() != NULL) return; hDC = GetDC(hwnd); if (hDC != NULL) { SetCapture(hwnd); AddSegmentAtMessagePos(hDC, hwnd, FALSE); while(GetMessage(&msg, NULL, 0, 0)) {

The Message Loop CHAPTER 3


if (GetCapture() != hwnd) break; switch (msg.message) { case WM_MOUSEMOVE: AddSegmentAtMessagePos(hDC, hwnd, TRUE); break; case WM_LBUTTONUP: goto ExitLoop; default: DispatchMessage(&msg); } } ExitLoop: ReleaseCapture(); ReleaseDC(hwnd, hDC); } } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_LBUTTONDOWN: DrawHello(hwnd); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);

59

3
THE MESSAGE LOOP

continues

60

Under the Hood: Windows and the Win32 API PART II

Listing 3.4. continued


ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); return msg.wParam; }

Previously, DrawHello merely put out a text string with the words Hello, World! in it. The new version is much more complex. Its work begins with checking whether any other application is already capturing the mouse or not, and acquiring a device context handle for the main window. Next, it captures the mouse via the SetCapture function, thus instructing Windows to send the application WM_MOUSEMOVE messages. The DrawHello function also makes a call to the helper function AddSegmentAtMessagePos which, when called with the Boolean value FALSE as its third parameter, simply moves the drawing position to the position of the most recent message. For this, it makes use of the GetMessagePos function, which retrieves the position of the mouse cursor at the time the most recent message was created. AddSegmentAtMessagePos also makes use of coordinate transform functions to translate the screen coordinates of the mouse into logical coordinates in the window. After the call to AddSegmentAtMessagePos, the DrawHello function enters its new message loop. While the mouse is captured, we expect special behavior from our application; most notably, it is expected to follow mouse movements on the screen by drawing additional freehand segments. This is again accomplished with calls to AddSegmentAtMessagePos with the third parameter set to TRUE, whenever a WM_MOUSEMOVE message is received. This message loop terminates when the mouse button is released, or when the application loses mouse capture for whatever reason. At that time, DrawHello returns, and the primary message loop resumes processing subsequent messages. Was it really necessary to implement this application with two message loops? Could we not have handled WM_MOUSEMOVE messages in our window procedure instead, dispatched there through the main message loop? It is certainly possible; however, organizing code the way demonstrated here makes it much more maintainable and helps avoid extremely large and complex window procedures.

Summary
Every Windows application is built around a message loop. A message loop makes repeated calls to the GetMessage or PeekMessage functions and retrieves messages, which it then dispatches to window procedures through DispatchMessage.

The Message Loop CHAPTER 3

61

Window procedures are defined for window classes at the time the window class is registered through RegisterClass. A typical window procedure contains a switch statement with case blocks for all messages the application is interested in; other messages are dispatched to the default window procedure DefWindowProc (or DefDlgProc in the case of dialog windows). Messages can be posted or sent to an application. Posted messages are deposited in a message queue, from which GetMessage or PeekMessage retrieves them. In contrast, sending a message implies an immediate call to the window procedure, bypassing the message queue and the message loop. An application can have several message loops depending on its requirements. Although it is never required to have more than one message loop, this approach can aid in the development of better organized, more maintainable code.

3
THE MESSAGE LOOP

62

Under the Hood: Windows and the Win32 API PART II

Windows, Dialog Boxes, and Controls CHAPTER 4

63

Windows, Dialog Boxes, and Controls


IN THIS CHAPTER

s The Window Hierarchy

64

s Window Management 67 s Painting Window Contents 70 s Window Management Messages 72 s Window Classes 74 s Dialog Boxes 81 s Common Dialogs 84 s Controls 91

4
WINDOWS, DIALOG BOXES, AND CONTROLS

64

Under the Hood: Windows and the Win32 API PART II

A window in Windows can be defined as a rectangular area on the screen. However, this definition, in all its simplicity, hides the volumes of functionality behind the abstract idea of a window as the primary unit through which a user and a Windows application interact. A window is not only an area through which an application can present its output; it is also a target of events, a target of messages within the Windows environment. Although the window concept in Windows predates the use of object-oriented languages on the PC by several years, the terminology is more than appropriate here: The properties of a window determine its appearance, while its methods determine how it responds to user input and other system events. A window is identified by a window handle. This handle (usually a variable of type HWND) uniquely identifies each window in the system. The list includes the obvious application windows and dialog boxes as well as the less obvious ones such as the desktop, certain icons, or buttons. Userinterface events are packaged into Windows messages with the appropriate window handle attached and then sent, or queued, to the application (or thread, to be more precise) that owns that window. Needless to say, Windows offers a lot of functionality covering the creation and management of windows.

The Window Hierarchy


Windows maintains its windows (I wish there were a way to talk about windows within Windows without turning every second sentence into an unintentional joke!) in a hierarchical organization. Each window has a parent and zero or more siblings. At the root of all windows is the desktop window, created by Windows at startup time. The parent window for top-level windows is the desktop window; the parent window for child windows is either a top-level window or another child window higher up in the hierarchy. Figure 4.1 demonstrates this hierarchy by dissecting a typical Windows screen. Actually, the situation under Windows NT is somewhat more complex. Unlike its simpler cousins, Windows NT has the capability to maintain multiple desktops simultaneously. In fact, Windows NT normally maintains three desktops: one for the Winlogon screen, one for user applications, and one for the screen saver. The visual window hierarchy normally reflects the logical hierarchy. That is, windows at the same hierarchy level are normally displayed in the Z-order, which is essentially the order in which siblings appear. However, this order can be changed for top-level windows. Top-level windows with the extended window style WM_EX_TOPMOST appear on top of any non-topmost top-level windows.

Windows, Dialog Boxes, and Controls CHAPTER 4

65

FIGURE 4.1.
The window hierarchy.

Desktop Window

Parent Parent

Sibling Application Window (overlapped) Owner Parent

Dialog Box (popup) Parent Button Button

Client Window (child)

Another relationship exists between top-level windows. A top-level window may have an owner, which is another top-level window. An owned window always appears on top of its owner and disappears if its owner is minimized. A typical case of a top-level window owned by another occurs when an application displays a dialog box. The dialog box is not a child window (it is not confined to the client area of the applications main window), but it remains owned by the application window. Several functions allow applications to traverse the window hierarchy and find a specific window. Heres a review of a few of the more frequently used functions:
GetDesktopWindow

4
WINDOWS, DIALOG BOXES, AND CONTROLS

EnumWindows

Enables an application to retrieve the handle of the current desktop window. Enumerates all top-level windows. A user-defined callback function, the address of which is supplied in the call to EnumWindows, is called once for every toplevel window. EnumWindows does not enumerate toplevel windows that are created after the function has been called, even if it has not yet completed the enumeration when the new window is created.

66

Under the Hood: Windows and the Win32 API PART II


EnumChildWindows

EnumThreadWindows

FindWindow

GetParent GetWindow

Enumerates all child windows of a given window, identified by a handle that is supplied in the call to EnumChildWindows. The enumeration is accomplished by a user-defined callback function, the address of which is also supplied in the call to EnumChildWindows. This function also enumerates descendant windows; that is, child windows that are themselves children (or descendants) of child windows of the window specified in the call to EnumChildWindows. Child windows that are destroyed before they are enumerated, or child windows that are created after the enumeration process started, will not be enumerated. Enumerates all windows owned by a specific thread by calling a user-supplied callback function once for every such window. The handle to the thread and the address of the callback function are supplied by the application in the call to EnumThreadWindows. The enumeration includes top-level windows, child windows, and descendants of child windows. Windows that are created after the enumeration process began are not enumerated by EnumThreadWindows. Can be used to find a top-level window by its window class name or window title. Identifies the parent window of the specified window. Offers the most flexible way for manipulating the window hierarchy. Depending on its second parameter, uCmd, this function can be used to retrieve the handle to a windows parent, owner, sibling, or child windows.

Windows, Dialog Boxes, and Controls CHAPTER 4

67

Window Management
Typically, an application creates a window in two steps. First, the window class is registered; next, the window itself is created through the CreateWindow function. The window class determines the overall behavior of the new window type, including most notably the address of the new window procedure. Through CreateWindow, the application controls minor aspects of the new window, such as its size, position, and appearance.

The RegisterClass Function and the WNDCLASS Structure


A new window class is registered when an application calls the following function:
ATOM RegisterClass(CONST WNDCLASS *lpwc);

The single parameter of this function, lpwc, points to a structure of type WNDCLASS describing the new window type. The return value is a Windows atom, a 16-bit value identifying a unique character string in a table maintained by Windows. The WNDCLASS structure is defined as follows:
typedef struct _WNDCLASS { UINT style; WNDPROC lpfnWndProc; int cbClsExtra; int cbWndExtra; HANDLE hInstance; HICON hIcon; HCURSOR hCursor; HBRUSH hbrBackground; LPCTSTR lpszMenuName; LPCTSTR lpszClassName; } WNDCLASS;

The meaning of some of these parameters is fairly straightforward. For example, hIcon is a handle to the icon used to represent minimized windows of this class; hCursor is a handle to the standard mouse cursor that is used when the mouse enters the window rectangle; hbrBackground is a handle to the GDI brush that is used to draw the windows background. The string pointed to by lpszMenuName identifies the menu resource (by name or, through the MAKEINTRESOURCE macro, by an integer identifier) that is used as the standard menu for this class; lpszClassName is the name of the window class. The parameters cbClsExtra and cbWndExtra can be used to allocate extra memory for the window class or for individual windows. Applications can use this extra memory to store application-specific information pertaining to the window class or individual windows. I left the explanation of the first two parameters for last, and for good reason. Most of what makes a window such a unique and complex entity is controlled through the window class style and the window procedure.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

68

Under the Hood: Windows and the Win32 API PART II

The parameter lpfnWndProc specifies the address of the window procedure function. This function is responsible for handling any messages the window receives. It can either handle those messages itself or invoke the default window procedure, DefWindowProc. The messages can be anything: window sizing and moving, mouse events, keyboard events, commands, repaint requests, timer and other hardware-related events, and so on. A typical window procedure contains a large switch statement block. Inside, case blocks exist for every message the application is interested in. Messages that the application does not handle are passed to DefWindowProc through the default block. The skeleton of such a window procedure is shown in Listing 4.1.

Listing 4.1. Window procedure skeleton.


LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_DESTROY: PostQuitMessage(0); break; // Other case blocks come here default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; }

Certain global characteristics of the window class are controlled through the class style parameter, style. This parameter may be set to a combination of values (using the bitwise OR operator, |). For example, CS_BYTEALIGNCLIENT specifies that the windows client area is always to be positioned on a byte boundary in the screen displays bitmap to enhance graphics performance (a very useful thing to remember when writing performance-intensive applications intended to run on lower-end graphics hardware). The value CS_DBLCLKS specifies that Windows should generate double-click mouse messages when the user double-clicks the mouse within the window. The pair of values CS_HREDRAW and CS_VREDRAW specifies that the window be redrawn in its entirety every time its horizontal or vertical size changes. Or the value CS_SAVEBITS specifies that Windows should allocate what UNIX and X programmers often refer to as backing store; a copy of the window bitmap in memory, so that it can automatically redraw the window when parts of it become unobscured. (This should be used with caution; the large amounts of memory required for this may cause a significant performance hit.)

Windows, Dialog Boxes, and Controls CHAPTER 4

69

NOTE
In 16-bit Windows, it was possible to register an application global class through the style CS_GLOBALCLASS. An application global class was accessible from all other applications and DLLs. This is not true in Win32. In order for an application global class to work as intended, it must be registered from a DLL that is loaded by every application. Such a DLL can be defined through the Registry.

Creating a Window Through CreateWindow


Registering a new window class is the first step in window creation. Next, applications must actually create a window through the CreateWindow function:
HWND CreateWindow( LPCTSTR lpClassName, LPCTSTR lpWindowName, DWORD dwStyle, int x, int y, int nWidth, int nHeight, HWND hWndParent, HMENU hMenu, HANDLE hInstance, LPVOID lpParam );

The first parameter, lpClassName, defines the name of the class that this window inherits its behavior from. The class must either be registered through RegisterClass or be one of the predefined control classes. The predefined classes include the BUTTON, COMBOBOX, EDIT, LISTBOX, SCROLLBAR, and STATIC classes. There are also some window classes that are mostly used internally by Windows and are referenced only through integer identifiers; these include classes for menus, the desktop window, and icon titles, to name but a few. The dwStyle parameter specifies the windows style. This parameter shall not be confused with the class style, passed to RegisterClass through the WNDCLASS structure when the new window class is registered. Although the class style determined some of the permanent properties of windows belonging to that class, the window style passed to CreateWindow is used to initialize the more transient properties of the window. For example, dwStyle can be used to determine the windows initial appearance (minimized, maximized, visible, or hidden). As is the case with the class style, the window style is also typically a combination of values (combined with the bitwise OR operator). In addition to the generic style values that are common to all types of windows, some values exist that are specific to the predefined window classes. For example, the BS_PUSHBUTTON style can be used for windows of the BUTTON class that are to send WM_COMMAND messages to their parents when clicked. Some dwStyle values are important enough to deserve a closer look.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

70

Under the Hood: Windows and the Win32 API PART II

The WS_POPUP and WS_OVERLAPPED styles specify top-level windows. The basic difference is that a WS_OVERLAPPED window always has a caption, while a WS_POPUP window does not need to have one. Overlapped windows are typically used as the main window of applications, while popup windows are used for dialog boxes. When a top-level window is created, the calling application sets its owner window through the hwndParent parameter. The parent window of a top-level window is the desktop window. Child windows are created with the WS_CHILD style. The major difference between a child window and a top-level window is that a child window is confined to the client area of its parent. Windows defines some combinations of styles that are most useful when creating typical windows. The WS_OVERLAPPEDWINDOW style setting combines the WS_OVERLAPPED style with the WS_CAPTION, WS_SYSMENU, WS_THICKFRAME, WS_MINIMIZEBOX, and WS_MAXIMIZEBOX styles to create a typical top-level application window. The WS_POPUPWINDOW style setting combines WS_POPUP with the WS_BORDER and WS_SYSMENU styles to create a typical dialog box.

Extended Styles and the CreateWindowEx Function


The CreateWindowEx function, while otherwise identical to the CreateWindow function, enables you to specify a combination of extended window styles. Extended window styles provide finer control over certain aspects of a windows appearance or the way it functions. For example, through the WS_EX_TOPMOST style applications can make a window a topmost window; that is, a top-level window that is not obscured by other top-level windows. A window created with the WS_EX_TRANSPARENT style does not obscure other windows and only receives a WM_PAINT message when all windows under it have been updated. The WS_EX_TOOLWINDOW style can be used to create a tool window. A tool window is a window with a smaller than usual title bar and other properties that make it useful as a floating toolbar window. Yet another set of extended styles specifies the windows behavior with respect to the selected shell language. For example, the WS_EX_RIGHT, WS_EX_RTLREADING, and WS_EX_LEFTSCROLLBAR extended styles can be used in conjunction with a right-to-left shell language selection such as Hebrew or Arabic.

Painting Window Contents


Painting in a window is performed through the normal set of GDI drawing functions. Applications usually obtain a handle to the display device context through a function such as GetDC, and then call GDI functions such as LineTo, Rectangle, or TextOut. But even more typically, window painting occurs in response to a specific message, WM_PAINT.

Windows, Dialog Boxes, and Controls CHAPTER 4

71

The WM_PAINT Message


The WM_PAINT message is sent to a window when parts of it require redrawing by the application and no other message is pending in the message queue of the thread that owns the window. Applications typically respond to this with a set of drawing instructions enclosed between calls to the BeginPaint and EndPaint functions. The BeginPaint function retrieves a set of parameters that are stored in a PAINTSTRUCT structure:
typedef struct tagPAINTSTRUCT { HDC hdc; BOOL fErase; RECT rcPaint; BOOL fRestore; BOOL fIncUpdate; BYTE rgbReserved[32]; } PAINTSTRUCT; BeginPaint

also takes care of erasing the background, if necessary, by sending the application a WM_ERASEBKGND message.

NOTE
The BeginPaint function should be called only in response to a WM_PAINT message. Each call to BeginPaint must be accompanied by a subsequent call to the EndPaint function.

Applications can use the hDC member of the structure to draw into the client area of the window. The rcPaint member represents the smallest rectangle that encloses all areas of the window that require updating. By limiting their activities to this rectangular region, applications can speed up the painting process.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

Repainting a Window by Invalidating Its Contents


The functions InvalidateRect and InvalidateRgn can be used to invalidate all or parts of a window. Windows sends a WM_PAINT message to a window if its update region, that is, the union of all update regions specified in prior calls to InvalidateRect and InvalidateRgn, is not empty and the thread that owns the window has no more messages in its message queue. This behavior suggests a very efficient mechanism for applications that need to update parts of their window. Instead of updating the window immediately, they can schedule the update by invalidating the appropriate region. When they process WM_PAINT messages, they can examine the update region (the rcPaint member of the PAINTSTRUCT structure) and update only those elements in the window that fall into this region. Alternatively (or in addition to this), applications can maintain private variables in which they store hints; that is, information that assists

72

Under the Hood: Windows and the Win32 API PART II

the window updating procedure in determining the most efficient way to update the window. Such hints are used throughout the Microsoft Foundation Classes.

Window Management Messages


A typical window responds to many other messages in addition to WM_PAINT messages. Heres a review of some of the more frequently processed messages:
WM_CREATE

WM_DESTROY

WM_CLOSE

WM_QUIT

WM_QUERYENDSESSION

The first message that the window procedure of a newly created window receives. This message is sent before the window is made visible, and before the CreateWindow or CreateWindowEx function returns. In response to this message, applications can perform initialization functions that are necessary before the window is made visible. Sent to the window procedure of a window that has already been removed from the screen and is about to be destroyed. Sent to a window indicating that the window should be closed. The default implementation in DefWindowProc calls DestroyWindow when this message is received. Applications can, for example, display a confirmation dialog and call DestroyWindow only if the user confirms closing the window. Usually the last message an applications main window receives. Receiving this message causes GetMessage to return zero, which terminates the message loop of most applications. This message indicates a request to terminate the application. It is generated in response to a call to PostQuitMessage. Notifies the application that the Windows session is about to be ended. An application may return FALSE in response to this message to prevent the shutdown of Windows. After processing the WM_QUERYENDSESSION message, Windows sends all applications a WM_ENDSESSION message with the results of the WM_QUERYENDSESSION processing.

Windows, Dialog Boxes, and Controls CHAPTER 4


WM_ENDSESSION

73

WM_ACTIVATE

WM_SHOWWINDOW

WM_ENABLE

WM_MOVE WM_SIZE WM_SETFOCUS

WM_KILLFOCUS

WM_GETTEXT

WM_SETTEXT

Sent to applications after the WM_QUERYENDSESSION message has been processed. It indicates whether Windows is about to shut down or whether the shutdown has been aborted. If an imminent shutdown is indicated, the Windows session may end at any time after the WM_ENDSESSION message has been processed by all applications. It is important, therefore, that applications perform all tasks pertaining to safe termination. Indicates when a top-level window is about to be activated or deactivated. The message is first sent to the window that is about to be deactivated, then to the window that is about to be activated. Indicates when a window is about to be hidden or shown. A window can be hidden as a result of a call to the ShowWindow function, or as a result of another window being maximized. Sent to a window when it is enabled or disabled. A window can be enabled or disabled through a call to the EnableWindow function. A window that is disabled cannot receive mouse or keyboard input. Indicates that the windows position has been changed. Indicates that the windows size has been changed. Indicates that the window has gained keyboard focus. An application may, for example, display the caret in response to this message. Indicates that the window is about to lose keyboard focus. If the application displays a caret, the caret should be destroyed in response to this message. Sent to a window requesting that the window text be copied to a buffer. For most windows, the window text is the window title. For controls like buttons, edit controls, static controls, or combo boxes, the window text is the text displayed in the control. This message is usually handled by the DefWindowProc function. Requests that the window text be set to the contents of a buffer. The DefWindowProc function sets the window text and displays it in response to this message.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

74

Under the Hood: Windows and the Win32 API PART II

Several messages concern the nonclient area of a window; that is, its title bar, border, menu, and other areas that are typically not updated by the application program. An application can intercept these messages to create a window frame with a customized appearance or behavior:
WM_NCPAINT

WM_NCCREATE

WM_NCDESTROY

WM_NCACTIVATE

Indicates that the nonclient area of a window (the window frame) needs to be repainted. The DefWindowProc function handles this message by repainting the window frame. Before the WM_CREATE message is sent to a window, it also receives a WM_NCCREATE message. Applications may intercept this message to perform initializations specific to the nonclient area of the window. Indicates that a windows nonclient area is about to be destroyed. This message is sent to a window after the WM_DESTROY message. Sent to a window to indicate that its nonclient area has been activated or deactivated. The DefWindowProc function changes the color of the window title bar to indicate an active or inactive state in response to this message.

Window Classes
Every window is associated with a window class. A window class is either a class provided by Windows or a user-defined window class registered through the RegisterClass function.

The Window Procedure


The purpose of a window class is to define the characteristics and behavior of a set of related windows. Perhaps the most notable, but by far not the only property of a window class, is the window procedure. I have already demonstrated a simple skeleton for a window procedure in Listing 4.1. The window procedure is called every time a message is sent to the window through the S e n d M e s s a g e function, and every time a posted message is dispatched through the DispatchMessage function. The role of the window procedure is to process messages sent or posted to that window. In doing so, it can rely on the default window procedure (DefWindowProc, or in the case of dialog boxes, DefDlgProc) for the processing of unwanted messages. It is through the window procedure that the behavior of a window is implemented. By responding to various messages, the window procedure determines how the window reacts to mouse and cursor events and how its appearance changes in reaction to those events. For example, in

Windows, Dialog Boxes, and Controls CHAPTER 4

75

the case of a button, the window procedure may respond to WM_LBUTTONDOWN messages by repainting the window, indicating that the button is pressed. Or in the case of an edit control, the window procedure may respond to a WM_SETFOCUS message by displaying the caret. Windows supplies two default window procedures:
DefWindowProc

and

DefDlgProc.

The

DefWindowProc function implements the default behavior for typical top-level windows. It pro-

cesses nonclient area messages and manages the window frame. It also implements some other aspects of top-level window behavior, such as responding to keyboard events; for example, responding to the Alt key by highlighting the first item in the windows menu bar. The DefDlgProc function is for the use of dialog boxes. In addition to the default top-level window behavior, it also manages the focus within a dialog box. It implements the behavior of dialogs whereby the focus jumps from one dialog control to the next when the user presses the Tab key. In addition to the default window procedures, Windows also supplies a set of window classes. These implement the behavior of dialog box controls, such as buttons, edit fields, list and combo boxes, and static text fields. The name for these classes is system global class, which is a leftover from the days of 16-bit Windows. In Win32 these classes are no longer global. That is, a change that affects a system global class will only affect windows of that class within the same application and have no effect on windows in another application because Win32 applications run in separate address spaces, and thus they are shielded from one another. Whether it is a Windows-supplied class or a class defined by the application, an application can use an existing window class from which to derive a new class and implement new or modified behavior. The mechanisms for accomplishing this are called subclassing and superclassing.

WARNING
An application should not attempt to subclass or superclass a window that belongs to another process.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

Subclassing
Subclassing means substituting the window procedure for a window class with another. This is accomplished by calling the SetWindowLong or SetClassLong function. Calling SetWindowLong with the GWL_WNDPROC index value substitutes the window procedure for a specific window. In contrast, calling SetClassLong with the GCL_WNDPROC index value substitutes the window procedure for all windows of that class that are created after the call to SetClassLong.

76

Under the Hood: Windows and the Win32 API PART II

Consider the simple example shown in Listing 4.2. (You can compile this code from the command line by typing cl subclass.c user32.lib.) This example displays the Hello, World! message. In a somewhat unorthodox fashion, it uses the BUTTON system class for this purpose. However, it subclasses the BUTTON class by providing a replacement window procedure. This replacement procedure implements special behavior when a WM_LBUTTONUP message is received: it destroys the window, effectively ending the application. To ensure proper termination, the WM_DESTROY message also receives special handling: a WM_QUIT message is posted through a call to PostQuitMessage.

Listing 4.2. Subclassing the BUTTON class.


#include <windows.h> WNDPROC OldWndProc; LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_LBUTTONUP: DestroyWindow(hwnd); break; case WM_DESTROY: PostQuitMessage(0); break; default: return CallWindowProc(OldWndProc, hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE d2, LPSTR d3, int d4) { MSG msg; HWND hwnd; hwnd = CreateWindow(BUTTON, Hello, World!, WS_VISIBLE | BS_CENTER, 100, 100, 100, 80, NULL, NULL, hInstance, NULL); OldWndProc = (WNDPROC)SetWindowLong(hwnd, GWL_WNDPROC, (LONG)WndProc); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); return msg.wParam; }

Take a closer look at the mechanism used in the new window procedure, WndProc, to reference the old window procedure for the default processing of messages. The old procedure is called through the Win32 function CallWindowProc. In 16-bit Windows, it was possible to call the address obtained by the call to SetWindowLong directly; this was always the address of the old

Windows, Dialog Boxes, and Controls CHAPTER 4

77

window procedure. In Win32, this is not necessarily so; the value may instead be a handle to the window procedure. This example performs the subclassing through SetWindowLong, meaning that it only affected the single button window for which SetWindowLong was called. If I had called SetClassLong instead, I would have altered the behavior of all buttons created subsequently. Consider the sample program in Listing 4.3 (to compile this program from the command line, type cl subclass.c user32.lib).

Listing 4.3. Subclassing the BUTTON class.


#include <windows.h> WNDPROC OldWndProc; LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_LBUTTONDOWN: MessageBeep(0xFFFFFFFF); default: return CallWindowProc(OldWndProc, hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE d2, LPSTR d3, int d4) { HWND hwnd; hwnd = CreateWindow(BUTTON, , 0, 0, 0, 0, 0, NULL, NULL, hInstance, NULL); OldWndProc = (WNDPROC)SetClassLong(hwnd, GCL_WNDPROC, (LONG)WndProc); DestroyWindow(hwnd); MessageBox(NULL, Hello, World!, , MB_OK); }

4
WINDOWS, DIALOG BOXES, AND CONTROLS

This example creates a button control but never makes it visible; the sole purpose of this controls existence is so that through its handle, the class behavior can be modified. Immediately after the call to SetClassLong, the button control is actually destroyed. But the effects of SetClassLong linger on! The subsequently displayed message box contains an OK button; and the behavior of this button (namely that when it is clicked by the left mouse button, the PC speaker emits a short beep) reflects the new window procedure. Similarly, if the program displayed other dialogs or message boxes, indeed anything that had button controls in it, all the newly created buttons would exhibit the modified behavior.

78

Under the Hood: Windows and the Win32 API PART II

Global Subclassing
In 16-bit Windows, a subclassing mechanism similar to that presented in the previous section was often used to change the system-wide behavior of certain types of windows such as dialog controls. (This is how the 3D control library CTL3D.DLL was implemented.) Subclassing the window class affected all newly created windows of that class, regardless of the application that created them. Unfortunately, in Win32 this is no longer the case; only windows of the same application are affected by such a change. So how can developers influence the global behavior of certain types of windows? The answer is, you have to use a DLL and ensure that it is loaded into every applications address space. Under Windows NT, this can be accomplished easily by creating a setting in the Registry. The following Registry value needs to be modified:
\HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Windows\ APPINIT_DLLS

DLLs that are listed under this Registry key are loaded into the address space of every newly created process. If you want to add several DLLs, separate the pathnames by spaces. Listing 4.4 shows a DLL that subclasses the BUTTON class just like the example shown in Listing 4.3. If you add the full pathname of this DLL to the above-mentioned Registry key, every time a button control is clicked, a short beep will be heard.

Listing 4.4. Subclassing in a DLL.


#include <windows.h> WNDPROC OldWndProc; LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_LBUTTONDOWN: MessageBeep(0xFFFFFFFF); default: return CallWindowProc(OldWndProc, hwnd, uMsg, wParam, lParam); } return 0; } BOOL WINAPI DllMain (HANDLE hModule, DWORD dwReason, LPVOID lpReserved) { HWND hwnd; switch(dwReason) {

Windows, Dialog Boxes, and Controls CHAPTER 4


case DLL_PROCESS_ATTACH: hwnd = CreateWindow(BUTTON, , 0, 0, 0, 0, 0, NULL, NULL, hModule, NULL); OldWndProc = (WNDPROC)SetClassLong(hwnd, GCL_WNDPROC, (LONG)WndProc); DestroyWindow(hwnd); } return TRUE; }

79

To compile this DLL from the command line, use cl /LD beepbtn.c user32.lib. The command-line flag instructs the compiler to create a DLL instead of an executable file.

/LD

WARNING
Be careful to add only a fully tested DLL to the Registry. A faulty DLL may render your system unstable or may prevent it from starting altogether. If that happens, a quick-and-dirty remedy is to boot into MS-DOS and rename the DLL file to prevent it from being loaded. Obviously, if your DLL file sits on an NTFS partition, this may not be so easy to do.

Adding your DLLs pathname to the APPINIT_DLLS Registry key is perhaps the simplest, but certainly not the only technique to inject your DLLs code into another applications address space. This technique also has some drawbacks, not the least of which is the fact that it does not work under Windows 95. (You may find information to the contrary on the Microsoft Developer Library. I tried the technique and it does not work; when I asked Microsofts product support, they confirmed that this Registry setting is not supported under Windows 95.) Another drawback of this technique includes the fact that a DLL specified this way is loaded into the address space of every applicationor, to be more precise, every GUI application that links with USER32.DLL. Even the slightest bug in your DLL may seriously affect the stability of the entire system. Fortunately, there are other techniques available that enable you to inject your DLL into the address space of another process. The first such technique requires the use of a Windows hook function. By using the function, it is possible to install a hook function into another applications address space. Through this mechanism, you can add a new window function to a window class owned by another application.
SetWindowsHookEx

4
WINDOWS, DIALOG BOXES, AND CONTROLS

The second technique relies on the CreateRemoteThread function and its capability to create a thread that runs in the context of another process.

80

Under the Hood: Windows and the Win32 API PART II

Superclassing
Superclassing means creating a new class based on the behavior of an existing class. An application that wishes to superclass an existing class can use the GetClassInfo function to obtain a WNDCLASS structure describing that class. After this structure has been suitably modified, it can be used in a call to the RegisterClass function that registers the new class for use. The example shown in Listing 4.5 demonstrates the technique of superclassing. In this example, a new window class, BEEPBUTTON, is created, its behavior based on the default BUTTON class. This new class is then used to display a simple message. To compile this program from the command line, type cl supercls.c user32.lib.

Listing 4.5. Superclassing the BUTTON class.


#include <windows.h> WNDPROC OldWndProc; LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_LBUTTONDOWN: MessageBeep(0xFFFFFFFF); default: return CallWindowProc(OldWndProc, hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE d2, LPSTR d3, int d4) { MSG msg; HWND hwnd; WNDCLASS wndClass; GetClassInfo(hInstance, BUTTON, &wndClass); wndClass.hInstance = hInstance; wndClass.lpszClassName = BEEPBUTTON; OldWndProc = wndClass.lpfnWndProc; wndClass.lpfnWndProc = WndProc; RegisterClass(&wndClass); hwnd = CreateWindow(BEEPBUTTON, Hello, World!, WS_VISIBLE | BS_CENTER, 100, 100, 100, 80, NULL, NULL, hInstance, NULL); while (GetMessage(&msg, NULL, 0, 0)) { if (msg.message == WM_LBUTTONUP) { DestroyWindow(hwnd); PostQuitMessage(0); }

Windows, Dialog Boxes, and Controls CHAPTER 4


DispatchMessage(&msg); } return msg.wParam; }

81

You have looked at the difference between the two techniques, subclassing and superclassing, in terms of their implementation. But what is the difference between them in terms of their utility? In other words, when would you use subclassing, and when would you use superclassing? The difference is simple. Subclassing modifies the behavior of an existing class; superclassing creates a new class based on the behavior of an existing class. In other words, if you use subclassing, you implicitly alter the behavior of every feature in your application that relies on the class that you subclass. In contrast, superclassing only affects windows that are based explicitly on the new class; windows based on the original class are not affected.

Dialog Boxes
In addition to its main application window with its title and menu bar and applicationdefined contents, an application most commonly uses dialogs to exchange information with the user. Typically, the applications main window exists throughout the life of the application, while its dialogs are more transient in nature, popping up only for the duration of a brief exchange of data; but this is not the key difference between a main window and a dialog. Indeed, there are applications that use a dialog box as their main window; in other applications, a dialog may remain visible for most of the applications lifetime. A dialog box usually contains a set of dialog controls, themselves child windows, through which the user and the application exchange data. There are several Win32 functions that assist in constructing, displaying, and managing the contents of a dialog box. Application developers usually need not be concerned about painting a dialogs controls or handling user-interface events; instead, they can focus on the actual exchange of data between the dialogs controls and the application. Dialog boxes represent a versatile capability in Windows. To facilitate their efficient use, Windows provides two types of dialog boxes: modeless and modal.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

Modal Dialogs
When an application displays a modal dialog box, the window that owns the dialog box is disabled, effectively suspending the application. The user must complete interaction with the modal dialog before the application can continue. A modal dialog is usually created and activated through the DialogBox function. This function creates the dialog window from a dialog template resource and displays the dialog as a modal dialog. The application that calls the DialogBox function supplies the address of a callback function; DialogBox does not return until the dialog box is dismissed through a call to EndDialog made from this callback function (possibly in response to a user-interface event, such as a click on the OK button).

82

Under the Hood: Windows and the Win32 API PART II

Although it is possible to create a modal dialog with no owner, it is not usually recommended. If such a dialog box is used, several issues must be taken into account. As the applications main window is not disabled, steps must be taken to ensure that messages sent or posted to it continue to be processed. Windows does not destroy or hide an ownerless dialog when other windows of the application are destroyed.

Modeless Dialogs
In contrast to modal dialogs, presenting a modeless dialog does not suspend execution of the application by disabling the owner window of the dialog box. However, modeless dialogs remain on top of their owner window even when the owner window gains focus. Modeless dialogs represent an effective way of continuously displaying relevant information to the user. A modeless dialog is typically created through the CreateDialog function. As there is no equivalent of the DialogBox function for modeless dialogs, applications are responsible for retrieving and dispatching messages for the modeless dialog. Most applications do this in their main message loop; however, to ensure that the dialog responds to keyboard events as expected and allows the user to move between controls using keyboard shortcuts, the application must call the IsDialogMessage function. A modeless dialog does not return a value to its owner. However, the modeless dialog and its owner can communicate using SendMessage calls. The dialog box procedure for a modeless dialog must not call the EndDialog function. The dialog is normally destroyed by a call to DestroyWindow. This function can be called in response to a user-interface event from the dialog box procedure. Applications are responsible for destroying all modeless dialog boxes before terminating.

Message Boxes
Message boxes are special dialogs that display a user-defined message, a title, and a combination of predefined buttons and icons. Their intended use is to display brief informational messages to the user and present a limited set of choices. For example, message boxes can be used to notify the user of an error condition and request instructions whether to retry or cancel the operation. A message box is created and displayed through the MessageBox function. The application that calls this function specifies the text string that is to be displayed and a set of flags indicating the type and appearance of the message box. In addition to the default application modal behavior of a message box, an application can specify two other modes of behavior: task modal and system modal. Use a task modal message box if you wish to disable interaction with all top-level windows of the application, not just the owner window of the message box. A system modal message box should be used in extreme cases,

Windows, Dialog Boxes, and Controls CHAPTER 4

83

warning the user of a potential disaster that requires immediate attention. System modal message boxes disable interaction with all other applications until the user deals with the message box.

NOTE
System modal message boxes should be used very carefully. Few things are more annoying than a misbehaving application that displays a system modal message box repeatedly in a loop (perhaps due to a programming error), effectively rendering the entire system useless.

Dialog Templates
Although it is possible to create a dialog in memory, most applications rely on a dialog template resource to determine the type and appearance of controls within a dialog. Dialog templates are typically created as part of the applications resource file. They can be created manually as a set of instructions in the resource file, or they can be created through a visual resource file editor, such as the resource editor of the Developer Studio. The dialog template defines the style, position, and size of the dialog and lists all controls within it. The style, position, and appearance of controls are also defined as part of the dialog template. The various dialog box functions draw the entire dialog based on the dialog template, except for controls that are marked as owner-drawn.

The Dialog Box Procedure


Dialog box procedure is just another name for the window procedure of a dialog box. There is no fundamental difference between a dialog box procedure and a window procedure, except perhaps the fact that a dialog procedure relies on DefDlgProc, rather than DefWindowProc, for default processing of messages. A typical dialog box procedure responds to WM_INITDIALOG and WM_COMMAND messages but little else. In response to WM_INITDIALOG, the dialog box procedure initializes the controls in the dialog. Windows does not send a WM_CREATE message to a dialog box procedure; instead, the WM_INITDIALOG message is sent, but only after all the controls within the dialog have been created, just before the dialog is displayed. This allows the dialog box procedure to properly initialize controls before they are seen by the user. Most controls send WM_COMMAND messages to their owner window (that is, the dialog box itself). To carry out the function represented by a control, the dialog box procedure responds to WM_COMMAND messages by identifying the control and performing the appropriate action.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

84

Under the Hood: Windows and the Win32 API PART II

Common Dialogs
Win32 implements a series of commonly used dialogs, freeing the programmer from the need to implement these for every application. These common dialogs are well known to every Windows user. They include dialogs for opening and saving files, selecting a color or a font, printing and setting up the printer, selecting a page size, and searching and replacing text. Common dialogs can be used in two ways. Applications can utilize the common dialog as is by calling one of the common dialog functions that is part of the Win32 API. Alternatively, applications can customize common dialogs by implementing a special hook function and supplying a custom dialog template.

NOTE
The appearance of all common dialog boxes has changed substantially in Windows 95. If you are porting a 16-bit application that supplies its own dialog templates, you must take this fact into account in order to present a visual appearance that is consistent with the rest of the operating system.

When a common dialog function encounters an error, the CommDlgExtendedError function can often be used to obtain additional information about the cause and nature of the problem.

The Open and Save As Dialogs


The File Open and File Save As dialogs are perhaps the most often seen common dialogs. The purpose of these dialogs is to allow the user to browse the file system and select a file to be opened for reading or writing. The File Open dialog is displayed when the application calls the GetOpenFileName function. The functions single parameter is a pointer to an OPENFILENAME structure. Members of this structure provide initialization values for the dialog box, and, optionally, the address of a hook function and the name of a custom dialog template, which are used for customizing the dialog. When the dialog is dismissed, applications can obtain the users selection from this structure. A typical File Open dialog is shown in Figure 4.2. The File Save As dialog is displayed in response to a call to GetSaveFileName. This function also takes a pointer to an OPENFILENAME structure as its single parameter. An example of the File Save As dialog is shown in Figure 4.3. Applications for which you want to use the new, Windows 95-style look of the common file dialogs (and take advantage of the new Explorer-related functionality) must specify the style OFN_EXPLORER in the Flags member of the OPENFILENAME structure.

Windows, Dialog Boxes, and Controls CHAPTER 4

85

FIGURE 4.2.
The File Open dialog (Explorer-style).

FIGURE 4.3.
The Save As dialog (Explorer-style).

These newer versions of the common file dialogs have another new feature. When a file dialog is customized, it is no longer necessary to reproduce the entire dialog template before adding your modifications. Instead, it is possible to create a dialog template containing only the controls you wish to add to the dialog and an optional special field, labeled with the ID stc32, indicating where the standard components of the dialog should be placed.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

The Choose Color Dialog


The Choose Color dialog box is used when the user is requested to select a color. The dialog can be used to select a color from the system palette or to specify a custom color. The Choose Color dialog, shown in Figure 4.4, is presented in response to a call to the ChooseColor function. Applications can control the initialization values of this dialog through the pointer to a CHOOSECOLOR structure, passed to the ChooseColor function as its single parameter. Through this structure, applications can also customize the dialogs behavior by supplying a hook function and the name of a custom dialog template. When the dialog is dismissed, the new color selection is available as the rgbResult member of the CHOOSECOLOR structure.

86

Under the Hood: Windows and the Win32 API PART II

FIGURE 4.4.
The Choose Color dialog.

The Font Selection Dialog


Another of the more frequently seen common dialogs is the Font Selection dialog. Through this dialog, the user can select a typeface, a font style, font size, special effects, text color, and a script. The Font Selection dialog is shown in Figure 4.5.

FIGURE 4.5.
The Font Selection dialog.

The Font Selection dialog is initialized through the CHOOSEFONT structure. This structure can also be used to specify a custom hook function and the name of a custom dialog template. The lpLogFont member of this structure points to a LOGFONT structure that can be used to initialize the dialog and receives information about the newly selected font when the dialog is dismissed. This structure can be used in a call to the GDI function CreateFontIndirect to actually create the font for use.

Windows, Dialog Boxes, and Controls CHAPTER 4

87

Dialogs for Printing and Print Setup


In the Windows 3.1 version of the Print dialog box, the user selects printing parameters and starts the printing process. The user selects and configures the printer from a separate dialog, the Print Setup dialog. Under Windows 95, Windows NT 4.0, or later versions these dialogs look and behave differently. The Print dialog (Figure 4.6) combines the functionality of the Windows 3.1 Print and Print Setup dialogs. Selection of the paper source and paper type, previously a function of the Print Setup dialog, is now available as part of a new dialog, the Page Setup dialog (Figure 4.7).

FIGURE 4.6.
The Print dialog.

To use the Print dialog, applications must first prepare the contents of a PRINTDLG structure, then call the PrintDlg function with a pointer to this structure as the functions only parameter.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

FIGURE 4.7.
The Page Setup dialog.

88

Under the Hood: Windows and the Win32 API PART II

The Page Setup dialog is displayed when applications call the PageSetupDlg function. The functions only parameter is a pointer to a PAGESETUPDLG structure. Through this structure, applications can control the fields of the dialog and possibly specify customization. When the dialog is dismissed, the users selections are available in this structure.

Text Find and Replace Dialogs


The Find and Find and Replace dialogs present an interface where the user can enter a text search string and, optionally, a replacement string. These dialogs differ fundamentally from the other common dialogs in that they are modeless dialogs; the other common dialogs all operate as modal dialogs. Therefore, the application that creates them is responsible for providing the message loop and dispatching dialog messages through the IsDialogMessage function. The Find dialog, shown in Figure 4.8, is displayed in response to a call to the FindText function. The function returns a dialog handle that can be used in the applications message loop in a call to IsDialogMessage. The dialog is initialized through a FINDREPLACE structure, which also receives any values the user may enter in the dialog.

FIGURE 4.8.
The Find dialog.

The dialog communicates with its owner window through a series of messages. Before calling FindText, applications should register the message string FINDMSGSTRING through a call to the RegisterWindowMessage function. The Find dialog will send this message to the application whenever the user enters a new search value. The Replace dialog (Figure 4.9) is a close cousin to the Find dialog and is initialized through an identical FINDREPLACE structure. This dialog is displayed in response to a call to the ReplaceText function.

FIGURE 4.9.
The Replace dialog.

When the application receives a message from a Find or Replace dialog, it can check the Flags member of the FINDREPLACE structure to determine what action was requested by the user.

Windows, Dialog Boxes, and Controls CHAPTER 4

89

NOTE
The Find and Replace dialogs are not destroyed when the FindText or ReplaceText functions return. For this reason, an application would normally allocate the FINDREPLACE structure in global memory. If memory allocated for the FINDREPLACE structure is deallocated before the Find or Replace dialogs are destroyed, the application will fail.

Common Dialogs Example


The sample program shown in Listing 4.6 creates and displays each of the common dialogs in sequence. This example has little practical value; it simply demonstrates, with a minimum amount of code, how these dialogs can be created and displayed. This sample can be compiled from the command line with cl commdlgs.c comdlg32.lib user32.lib.

Listing 4.6. Common dialogs.


#include <windows.h> LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; OPENFILENAME ofn; CHOOSECOLOR cc; CHOOSEFONT cf; PRINTDLG pd; PAGESETUPDLG psd; FINDREPLACE fr; COLORREF crCustColors[16]; LOGFONT lf; char szFindWhat[80]; char szReplaceWith[80]; HWND hdlgFt, hdlgFr;

4
WINDOWS, DIALOG BOXES, AND CONTROLS

continues

90

Under the Hood: Windows and the Win32 API PART II

Listing 4.6. continued


if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_APPWORKSPACE + 1); wndClass.lpszClassName = COMMDLGS; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(COMMDLGS, Common Dialogs Demonstration, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); memset(&ofn, 0, sizeof(ofn)); ofn.lStructSize = sizeof(OPENFILENAME); GetOpenFileName(&ofn); memset(&ofn, 0, sizeof(ofn)); ofn.lStructSize = sizeof(OPENFILENAME); GetSaveFileName(&ofn); memset(&cc, 0, sizeof(cc)); memset(crCustColors, 0, sizeof(crCustColors)); cc.lStructSize = sizeof(cc); cc.lpCustColors = crCustColors; ChooseColor(&cc); memset(&cf, 0, sizeof(cf)); memset(&lf, 0, sizeof(lf)); cf.lStructSize = sizeof(cf); cf.lpLogFont = &lf; cf.Flags = CF_SCREENFONTS | CF_EFFECTS; ChooseFont(&cf); memset(&pd, 0, sizeof(pd)); pd.lStructSize = sizeof(pd); PrintDlg(&pd); memset(&psd, 0, sizeof(psd)); psd.lStructSize = sizeof(psd); PageSetupDlg(&psd); memset(&fr, 0, sizeof(fr)); memset(szFindWhat, 0, sizeof(szFindWhat)); memset(szReplaceWith, 0, sizeof(szReplaceWith)); fr.lStructSize = sizeof(fr); fr.hwndOwner = hwnd; fr.lpstrFindWhat = szFindWhat; fr.lpstrReplaceWith = szReplaceWith; fr.wFindWhatLen = sizeof(szFindWhat); fr.wReplaceWithLen = sizeof(szReplaceWith); hdlgFt = FindText(&fr); hdlgFr = ReplaceText(&fr); while (GetMessage(&msg, NULL, 0, 0)) if(!IsDialogMessage(hdlgFt, &msg)) if(!IsDialogMessage(hdlgFr, &msg))

Windows, Dialog Boxes, and Controls CHAPTER 4


DispatchMessage(&msg); return msg.wParam; }

91

OLE Common Dialogs


As part of the OLE 2 implementation, the system provides common dialogs for the following set of functions: Insert Object, Paste Special, Change Source, Edit Links, Update Links, Object Properties, Convert, and Change Icon. Most applications do not invoke these dialogs directly, but use the Microsoft Foundation Classes (and, in particular, the wrapper classes for these dialogs) to implement OLE functionality.

Controls
A control is a special window that typically enables the user to perform a simple function and sends messages to this effect to its owner window. For example, a pushbutton control has one simple function, namely that the user can click on it; when that happens, the pushbutton sends a WM_COMMAND message to the window (typically a dialog) that owns it. Windows offers several built-in control classes for the most commonly used controls. A dialog with a sample collection of these controls is shown in Figure 4.10.

FIGURE 4.10.
A collection of standard controls.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

Many of the standard controls were introduced with Windows 95 and consequently they are often referred to as Windows 95 common controls. This name is slightly misleading as the new control classes are now also available in Windows NT 3.51 and later, and also in Win32s 1.3. Figure 4.11 shows another dialog with a collection of Windows 95 common controls. Applications can also create their own controls. These can be derived from the standard control classes, or they can be built independently. The control class and the control style (which defines variations of behavior within a button class) are usually both defined in an applications resource file. Alternatively, applications that create controls programmatically select the button class and specify the button style as parameters to the CreateWindow function.

92

Under the Hood: Windows and the Win32 API PART II

FIGURE 4.11.
Some Windows 95 common controls.

A brief description of the major control classes follows. Static controls The simplest of all control types. The sole purpose of their existence is to display a piece of text, such as a label for another control. Static controls do not respond to user-interface events and do not send messages to their owner window. Respond to simple mouse clicks. A pushbutton is a button that posts a WM_COMMAND message to its owner window when it is clicked. A check box indicates one of two states, selected and not selected. A variant of the check box, the three-state check box, adds a third, disabled state to the other two. A radio button is a control that is typically used in groups, indicating a set of mutually exclusive choices. Rectangular areas where the user can enter unformatted text. The text can be a few characterssuch as the name of a fileor an entire text file; for example, the client area of the Windows Notepad application is one large edit control. Applications typically communicate with the edit control through a series of messages that are used to set or retrieve text from the control. Contain a collection of values arranged in rows. Users can use the mouse cursor to select the desired value from the list. If the list box contains more values than can be displayed at once, a vertical scrollbar is also displayed as part of the list box.

Buttons

Edit controls

List boxes

Windows, Dialog Boxes, and Controls CHAPTER 4

93

Combo boxes

Scrollbars

Animation controls Header controls Hotkey controls

List controls

Progress bars

Rich-text edit controls

Slider controls

Spin buttons Status bars

Combine the functionality of a list box and an edit control. Users can enter a value in the edit control part of the combo box. Alternatively, they can click the down arrow next to the edit control to display the list box part, where a value can be selected. Consist of a rectangular area with two arrows at the end and a sliding pointer. A scrollbar can be vertical or horizontal. Scrollbars are typically used to indicate the position of the visible portion within a larger area. Provide little animated displays for use during a lengthy operation, for instance. Provide column headers, such as those used in list controls. Accept a keystroke combination from the user, which the application can use to set up a hotkey through the WM_SETHOTKEY message. Expand the functionality of a list box by providing a means to display a list of items in one of several formats. In a typical list control, items have an icon and some text; the control can display these items in a variety of formats as icons or as list items arranged in rows. Indicate the progress of a lengthy process. Progress bars do not accept user input; they are used for informational purposes only. Expand the functionality of the Windows 3.1 edit control by allowing the editing of Microsoft RTF (rich text format) files. Rich-text controls encapsulate the capability of a reasonably sophisticated word processor. Provide a function similar to the sliding volume control on many stereo systems. The user can position the sliding tab with the mouse to set a specific position in the slider control. Slider controls are ideal in multimedia applications as volume or picture controls, or controls through which the user can set the position during playback of a multimedia data source. Used to increment or decrement the value of an associated control, usually an edit control. Provide a standard status bar at the bottom of a window (typically, the applications main window).

4
WINDOWS, DIALOG BOXES, AND CONTROLS

94

Under the Hood: Windows and the Win32 API PART II

Tab controls

Toolbar controls Tooltip controls Tree controls

Help in implementing multipage dialogs, also known as tabbed dialogs or property sheets. A tab control provides a user interface in which the user can select the dialog page (property page) by clicking a little tab. The tab gives the visual appearance of several sheets organized on top of each other, and clicking on the tab gives the visual impression of bringing the selected sheet to front. Provide a dockable bar containing toolbar buttons and dialog controls. Provide a floating window with explanatory text. Present a list of items in a hierarchical organization. Tree controls are ideal for displaying hierarchical lists, such as a list of directories on disk. Tree controls provide an efficient mechanism for displaying a large number of items by providing the ability to expand and collapse higher-level items.

Summary
A window is a rectangular area on the screen through which applications and the user communicate. Applications draw into the window to display information for the user. Applications receive messages on user-interface events through a handle to the window. Windows are arranged hierarchically. At top is the desktop window. Top-level windows are those whose parent is the desktop windowor those that have no parent window. Child windows are those whose parent is a top-level window or another child window. Windows sharing the same parent are siblings; the order in which sibling windows are displayed is called the Zorder. A special category of windows contains top-level windows that have the topmost attribute; these windows always precede non-topmost windows in the Z-order, even when a nontopmost window is the active window. A top-level window may have an owner window that is different from its parent window. Typical windows that users normally interact with include overlapped windows (normal application windows); pop-up windows (dialog boxes); and controls. Window messages are handled in the window procedure. A window procedure and other window attributes are associated with the window class from which windows are derived. In addition to the capability of defining their own window classes, applications can also superclass and subclass existing window classes. Subclassing means modifying the behavior of an existing window class; superclassing means creating a new window class based on the behavior of an existing class.

Windows, Dialog Boxes, and Controls CHAPTER 4

95

Part of the Win32 API is a set of functions that assists in creating, displaying, and managing dialogs. Windows distinguishes between modal dialogs and modeless dialogs. A modal dialog disables its owner window while it is displayed and does not return control to the application until the user dismisses the dialog. In contrast, modeless dialogs are displayed without disabling their owner window. Applications must provide message loop functionality and dispatch dialog messages through the IsDialogMessage function for modeless dialogs. Windows also provides a set of common dialogs for common tasks. These include dialogs for opening and saving a file, printer and page setup, color and font selection, and text find and replace functions. In addition, a set of common dialogs is available to implement OLE-related functionality. Controls include buttons, static text, edit boxes, list boxes, combo boxes, and scrollbars. Applications can also implement new control types. In addition, Windows 95 defines a set of new common controls: list views, tree views, tab controls, hotkey controls, sliders, progress bars, spin buttons, and rich-text edit controls. Controls are usually defined through dialog box templates in the applications resource file. Controls communicate with the application by sending messages (such as WM_COMMAND messages) to their owner window, that is, the dialog box.

4
WINDOWS, DIALOG BOXES, AND CONTROLS

96

Under the Hood: Windows and the Win32 API PART II

Resource Files CHAPTER 5

97

Resource Files
IN THIS CHAPTER
s Resource File Components 98 s Compiling and Using Resource Scripts 107

5
RESOURCE FILES

98

Under the Hood: Windows and the Win32 API PART II

Resource files represent one of the most prominent features of Windows programming. It is through resource files that most applications define the visible elements of their user interface: menus, dialogs, text strings, bitmaps, and other types of resources. Resource files are created in a form readable by humans and compiled with the Resource Compiler. The compiled result is usually linked with the rest of the application to form a single binary image that contains executable code and resource information. Strictly speaking, use of resource files is not mandatory. However, it would be foolish to write needlessly complex code to implement an applications user-interface elements when resource files provide a convenient way to do the same. Furthermore, the use of resource files makes it much easier to replace language-dependent user-interface elements when working on an application that will be used in multiple language environments. In the good old days programmers used a text editor to edit a resource file. These days, handediting of resource files is rare. Instead, graphics resource editors are used, such as the integrated resource editor that is part of the Developer Studio. Nevertheless, the ability to make sense of resource file contents, the ability to hand-edit resource files when necessary, can still prove to be a useful skill. The syntax and the structure of resource files are simple, and understanding what is under the hood helps put the features of sophisticated editors in perspective. This chapter reviews use of resource files and the resource compiler.

NOTE
Often, the term resource file is used to apply to files generated by the resource compiler; in other words, files with the .res extensions. However, in this text I use the term exclusively to refer to files containing resource scripts; that is, files with the .rc extension.

Resource File Components


A resource file, or resource script, may contain any number of resource script statements and preprocessor directives. For example, consider the sample resource file shown in Listing 5.1.

Listing 5.1. Resource script example.


#include <windows.h> DlgBox DIALOG 20, 20, 90, 64 STYLE DS_MODALFRAME | WS_CAPTION | WS_SYSMENU CAPTION Sample Dialog BEGIN DEFPUSHBUTTON Sample Button, IDOK, 19, 44, 52, 14, WS_GROUP CTEXT Sample Message, -1, 0, 8, 90, 8 END

Resource Files CHAPTER 5

99

This simple script defines a dialog with a button and a static text field (Figure 5.1). The dialog will be referred to by name (DlgBox) from the application using this resource. Although it is not obvious at first sight, some constants (for example, IDOK, DS_MODALFRAME, or WS_GROUP) are defined in the file windows.h and are not part of the resource script language. The C/C++ style #include directive is used to specify files that contain macro definitions.

FIGURE 5.1.
Sample dialog box.

Generally, a resource file statement consists of an identifier that can either be a text string (DlgBox in the preceding example) or a numeric value, followed by the statement itself. The statement can be single line (in which case the statement is followed by one or more parameters) or multiline (in which case the statement is followed by a block of script instructions).

Resource File Preprocessing


Preprocessor syntax and semantics are similar to the syntax and semantics of the C/C++ preprocessor. The resource compiler preprocessor, or RC preprocessor for short, understands the following directives: s s s s Macro definitions: #define and #undef Conditional compilation: #if, #ifdef, #ifndef, #else, #elif, #endif Header files: #include Compile-time error: #error

The meaning of these directives is familiar to even the novice C programmer. The preprocessor also understands (and removes) C and C++ style comments; that is, comments enclosed between /* and */ and comment lines that begin with //. The preprocessor also understands the stringizing operator # and the token-pasting operator ##. It is possible to use the same header file from both C source and resource script files. During resource compilation, the resource compiler defines the symbol RC_INVOKED; this symbol can be used in the header file to protect against compiler errors. For example, if the header file contains a class declaration, you may wish to protect it as follows:
#ifndef RC_INVOKED class myclass { ... }; #endif // RC_INVOKED

5
RESOURCE FILES

100

Under the Hood: Windows and the Win32 API PART II

Another useful symbol to remember is the symbol APSTUDIO_INVOKED. This symbol is defined when a resource file is loaded by the integrated resource editor in Developer Studio. (In the days of Visual C++ 1.0, resource editing was performed by a separate program, Application Studio. Hence, the name of this constant.) Resource scripts can also contain constant expressions involving the following operators: addition (+), subtraction (-), multiplication (*), division (/), unary not (~), binary and (&), and binary or (|).

Single-Line Statements
Single-line statements define bitmaps, cursors, fonts, icons, message tables, and the language of resource statements. The BITMAP statement specifies a bitmap file (edited separately with a bitmap editor) that is to define a bitmap resource. For example,
MyBitmap BITMAP mybitmap.bmp

The CURSOR statement specifies a file that defines the shape of the mouse cursor. The cursor file is a binary file that is edited by a separate editor. Here is an example for this statement:
MyCursor CURSOR mycursor.cur

The FONT statement specifies a font resource. An example is


MyFont FONT myfont.fnt

The ICON statement specifies a binary icon file (edited by an icon editor) that defines an icon resource. For example,
MyIcon ICON myicon.ico

The LANGUAGE statement sets the language of all subsequent resources up to the end of the resource script or until another LANGUAGE statement is encountered. The LANGUAGE statement can also be used to set the language of a specific resource in a multiline resource statement if it appears just before the BEGIN keyword in such a statement. The LANGUAGE statement is followed by a language and a sublanguage identifier, both of which must be constants defined in the file winnls.h. The MESSAGETABLE statement identifies a message table, which is used primarily in Windows NT event logging. A message table is created by the message compiler, mc.exe. The BITMAP, CURSOR, FONT, and ICON statements also accept an attribute parameter. The attribute parameter specifies load and memory characteristics of the resource. In 32-bit Windows, only one attribute is used: The DISCARDABLE attribute specifies that a resource can be removed from memory if it is no longer needed. For example,
TempBmp BITMAP DISCARDABLE c:\\bitmaps\\tempbmp.bmp

Resource Files CHAPTER 5

101

Multiline Resource Statements


Multiline resource statements define dialogs, string tables, accelerator tables, menus, and version information resources. Multiline statements begin with an identifier, the statement, and optional parameters, followed by a block of instructions enclosed between a BEGIN-END keyword pair:
identifier STATEMENT [optional-parameters] [optional instructions] BEGIN [instructions] END

Optional instructions may include the CHARACTERISTICS instruction (specifies a single 32-bit value used only by resource file management tools), the LANGUAGE instruction, and the VERSION instruction (specifies a 32-bit version number used by resource management tools). Other optional instructions are multiline statement-specific; for example, the CAPTION instruction defines the title of a dialog box. Because of their relative complexity, the following sections look at multiline statements individually.

Accelerators
Accelerators are keystrokes that represent shortcuts to specific tasks. For example, when you use Ctrl+C to copy an item to the clipboard, you are using an accelerator. Enclosed between the BEGIN-END keyword pair, an accelerator statement contains an arbitrary number of keyboard events followed by the identifier of the accelerator. Consider the following example:
MyAcc ACCELERATOR BEGIN C, ID_EDIT_COPY, VIRTKEY, CONTROL, NOINVERT V, ID_EDIT_PASTE, VIRTKEY, CONTROL, NOINVERT X, ID_EDIT_CUT, VIRTKEY, CONTROL, NOINVERT END

This example assumes that the symbolic constants ID_EDIT_COPY, ID_EDIT_PASTE, and ID_EDIT_CUT are defined elsewhere (in a header file) and refer to numeric identifiers. Accelerators are interpreted when an application calls the TranslateAccelerator function after retrieving a message through GetMessage (or PeekMessage). TranslateAccelerator translates WM_KEYDOWN (or WM_SYSKEYDOWN) messages into WM_COMMAND (or WM_SYSCOMMAND) messages. The identifiers that follow accelerator keys in an accelerator statement will become the command identifiers in the WM_COMMAND messages.

5
RESOURCE FILES

102

Under the Hood: Windows and the Win32 API PART II

Dialogs
Together with menu statements, dialog statements define what are perhaps the most visible parts of a typical application. A dialog statement defines the layout and appearance of a dialog box. A sample dialog statement is shown in Listing 5.1. The statement consists of several lines. The first line contains an identifier, the DIALOG keyword, and four numeric parameters that define the position of the dialogs upper-left corner and the size of the dialog box. All size and position information in a dialog statement is measured in dialog units. Dialog units are derived from the size of the font specified for the dialog. Dialog base units represent the average height and width of a character in the selected font. Four horizontal dialog units amount to one horizontal dialog base unit; eight vertical dialog units amount to one vertical dialog base unit. For dialogs that use the system font, applications can obtain the size of dialog base units, in pixels, by calling GetDialogBaseUnits. For dialogs using other fonts, it may be necessary to explicitly calculate the average size of characters to obtain the dialog base units. Once the dialog base units are known, applications can convert between dialog units and pixels using the following formulas:
pixelX = (dialogunitX pixelY = (dialogunitY dialogunitX = (pixelX dialogunitY = (pixelY * * * * baseunitX) / 4 baseunitY) / 8 4) / baseunitX 8) / baseunitY

Following the line containing the DIALOG keyword, a dialog statement may contain several optional instructions. These may include commonly used optional instructions or dialogspecific optional instructions. The CAPTION optional instruction is followed by a string specifying the title of the dialog box. The default is a dialog with no title. The STYLE optional instruction specifies the style of the dialog. Style values are usually predefined in the header file windows.h. Several values can be combined using the logical or (|) operator. The default style for dialogs that contain no STYLE instruction is WS_POPUP | WS_BORDER | WS_SYSMENU. The EXSTYLE optional instruction is similar to STYLE and specifies extended styles.
CLASS

The CLASS optional instruction can be used to specify a special window class for a dialog. The instruction should be used sparingly, as it redefines the behavior of the dialog.

The FONT statement specifies the font to be used in the dialog. The default is the system font. The MENU statement identifies the menu resource that defines the menu of the dialog box. In the absence of this statement, the dialog box will be created without a menu bar.

Resource Files CHAPTER 5

103

Enclosed between the BEGIN-END keyword pair is a series of control statements specifying the controls within the dialog. There are several types of control statements. Each control statement contains the control type, control text, a control identifier (text or integer), control position, and control style and extended style parameters:
CONTROL-STATEMENT control-text, identifier, x, y, width, height [, style [, extended-style]]

An edit control is defined by the EDITTEXT control statement. A static control is defined by the LTEXT, CTEXT, RTEXT, or ICON control statement. The first three of these control statements define a left-aligned, centered, or right-aligned static control, respectively. The last specifies a static control with the SS_ICON style. A button control is defined by one of the following keywords:
USERBUTTON. AUTO3STATE, AUTOCHECKBOX, AUTORADIOBUTTON, CHECKBOX, DEFPUSHBUTTON, GROUPBOX, PUSHBOX, PUSHBUTTON, RADIOBUTTON, STATE3,

The COMBOBOX control statement defines a combo box control. The LISTBOX control statement can be used to specify a list box. The SCROLLBAR control statement defineswhat else?a scrollbar. The CONTROL control statement can be used to define a generic control. The syntax of this statement is somewhat different from the syntax used for other control statements:
CONTROL control-text, identifier, class-name, x, y, width, height [, extendedstyle]

The class-name parameter specifies the window class for the control. This can be one of the Windows control classes. Thus, the CONTROL statement can be used as an alternative syntax for all the other control statements. A variant of the DIALOG statement is the DIALOGEX statement. It extends the syntax of the DIALOG statement in the following ways: s It allows you to specify help identifiers for the dialog itself as well as any controls within it. s It allows font weight and italic settings in the FONT instruction. s It allows control-specific data to be added to control statements (enclosed by the BEGIN-END keyword pair). s It allows the use of the keywords BEDIT, HEDIT, and IEDIT for pen controls.

5
RESOURCE FILES

104

Under the Hood: Windows and the Win32 API PART II

Menus
Menu statements in the resource script can be used to specify menu bars or pop-up menus. Menu statements contain one or more menu definition statements enclosed between a BEGINEND keyword pair. Consider the following example:
MyMenu MENU BEGIN POPUP &File BEGIN MENUITEM &New\tCtrl+N, ID_FILE_NEW MENUITEM SEPARATOR MENUITEM E&xit, ID_FILE_EXIT END POPUP &Edit BEGIN MENUITEM Cu&t\tCtrl+X, ID_EDIT_CUT MENUITEM &Copy\tCtrl+C, ID_EDIT_COPY MENUITEM &Paste\tCtrl+V, ID_EDIT_PASTE END POPUP &Help BEGIN MENUITEM &About, ID_HELP_ABOUT END END

The identifiers (for example, ID_FILE_NEW) in this example are assumed to be defined elsewhere and refer to numeric values. A menu definition statement can specify a menu item or a submenu. To define a menu item, use the MENUITEM keyword. This is followed either by the text of the menu item and the menu identifier or by the SEPARATOR keyword; the latter specifies a separator that is a vertical line for menu bars or a horizontal line for pop-up (sub) menus. A menu items identifier may be followed by a list of options. Options are separated by commas or spaces and include the following keywords: CHECKED, GRAYED, HELP, INACTIVE, MENUBARBREAK, MENUBREAK. To specify a submenu, use the POPUP keyword. The POPUP keyword is followed by the title of the submenu and a set of menu items enclosed between a BEGIN-END keyword pair. A submenu may contain nested submenus.

String Tables
A string table resource defines an arbitrary number of text strings. These text strings can be referred to from within the application by symbolic identifiers. The primary advantage of using a string table resource is that it makes it possible for all language-dependent components of an application to be conveniently located within a resource file and thus renders the task of localization substantially easier.

Resource Files CHAPTER 5

105

A string table is defined by the STRINGTABLE keyword, followed by optional instructions and one or more string definitions enclosed between a BEGIN-END keyword pair. A string definition consists of a string identifier and a string value. For example,
STRINGTABLE BEGIN IDS_HELLO Hello IDS_GOODBYE Goodbye END

where IDS_HELLO and IDS_GOODBYE are symbolic identifiers defined elsewhere. Using a string from a string table resource is straightforward. An application can load a string resource using the LoadString function. For Microsoft Foundation Class applications, using string table resources is even easier. Many MFC functions that accept a string parameter can also accept a numeric parameter representing a string resource in the applications resource file. For example, an MFC application can display a message box using AfxMessageBox as follows:
AfxMessageBox(IDS_ERROR);

Here, IDS_ERROR is a symbolic reference to a numeric identifier. The


CString

class also offers specific support for string resources. The member function

CString::LoadString can be used to initialize a CString object to a value obtained from a string

table in the applications resource file.

Toolbars
Toolbar resources are used in MFC applications. A toolbar is defined in a resource file by a TOOLBAR statement. This statement lists the identifiers of the buttons in the toolbar. The size of toolbar buttons is also specified, as in the following example:
IDR_MAINFRAME TOOLBAR DISCARDABLE BEGIN BUTTON ID_FILE_NEW BUTTON ID_FILE_OPEN BUTTON ID_FILE_SAVE SEPARATOR BUTTON ID_EDIT_CUT BUTTON ID_EDIT_COPY BUTTON ID_EDIT_PASTE SEPARATOR BUTTON ID_FILE_PRINT BUTTON ID_APP_ABOUT END 16, 15

5
RESOURCE FILES

For every toolbar resource, there should be a corresponding bitmap resource that contains the button bitmaps. The button bitmaps should be arranged horizontally, in the same order as the

106

Under the Hood: Windows and the Win32 API PART II

button identifiers appear in the toolbar resource (except for separators). The bitmap resource should have the same identifier as the toolbar resource:
IDR_MAINFRAME BITMAP MOVEABLE PURE res\\Toolbar.bmp

Toolbar resources are accessed through the MFC function Ctoolbar::LoadToolBar.

Version Information
The version information resource can be used to identify the version of a binary file (typically, an executable or library file). Version information is used by File Installation Library functions. The version information resource statement contains several version statements that identify the version number of the file and the product, provide additional version information, and specify the language and the target operating system. Version information can be accessed using the GetFileVersionInfo, GetFileVersionInfoSize, and VerQueryValue functions.

User-Defined and Raw Data Resources


The resource file syntax also allows for the creation of user-defined resource types. A userdefined resource statement can be a single-line or multiline statement. Single-line statements are used to identify user-defined resources that are stored in separate files. The syntax of a singleline statement is as follows:
name type [load-memory-options] filename

Multiline user-defined resource statements can be used to embed the definition of a userdefined resource within the resource file. The syntax of a multiline user-defined resource statement is this:
name type [load-memory-options] BEGIN raw-data END

The raw-data block may contain integers in decimal, hexadecimal, or octal notation, or strings enclosed in double quotes. Raw data strings must be explicitly null-terminated. Individual data items are separated by commas. Raw data can also be specified in the form of an RCDATA resource. The syntax of an RCDATA statement and the syntax of a multiline user-defined resource statement are nearly identical, except that in the case of raw data statement, the keyword RCDATA is used in place of type, and the statement may also contain the optional instructions CHARACTERISTICS, LANGUAGE, and VERSION.

Resource Files CHAPTER 5

107

NOTE
Strings in all resource statements except raw data and user-defined resource statements are interpreted as null-terminated Unicode strings. If you wish to explicitly specify a Unicode string in an RCDATA or user-defined resource statement, use the L prefix.

Compiling and Using Resource Scripts


Before a resource script can be used by an application, it must be compiled. Although not strictly necessary, in the case of most applications the compiled resource file is also linked with other application components and becomes part of the applications executable file image (.exe file).

Running the Resource Compiler


A resource file can be compiled from the command line using the resource compiler, rc.exe. For most resource files, this command can be used with no parameters. For example, to compile the resource file shown in Listing 5.1, you would type rc dlg.rc. In 16-bit versions of Windows, the resource compiler was also used to add resources to the executable file. In Win32, the 32-bit linker is used for this purpose.

Running the Linker


The Visual C++ linker, link.exe , can be used to link together a resource file and other components of an applications executable. In the simplest case, you specify the name of the compiled resource file on the C/C++ compiler command line just as you would specify any other object file or library file. For example,
cl dlg.cpp dlg.res user32.lib

In order for the linker to work properly with compiled resource files, the cvtres.exe tool must be present in the same directory as the linker executable, link.exe. This tool converts compiled resource files into the Common Object File Format (COFF), which is understood by the linker.

Resource DLLs
Resources do not need to be linked with your applications executable file. They can also be linked into a separate dynamic-link library. This has the advantage that changing the applications resource file does not require recompiling the entire application, only replacing the DLL. You can also ship your application with multiple DLLs representing resources in various languages. In MFC applications, this approach is made even easier by the

5
RESOURCE FILES

108

Under the Hood: Windows and the Win32 API PART II


AfxSetResourceHandle call; calling this function with a handle to a resource DLL ensures that subsequently, MFC loads all resources from that DLL instead of your applications executable file.

Summary
Resource files define the visual appearance of an application. Resources include dialogs, menus, bitmaps, icons, cursors, text strings, and several other types. Although resources are typically created with graphics editors, it is possible (and sometimes necessary) to edit the resource file directly. Resource files (files with the .rc extension) are human-readable ASCII text files. Resource files contain preprocessor directives, single-line resource statements, and multiline resource statements. Preprocessor directives are similar in syntax and semantics to their C/C++ counterparts. Single-line resource file statements are used to identify resources stored in separate binary files (edited by specialized editors): bitmaps, icons, cursors, fonts, and generic resources. Single-line statements are also used to specify message table resources (used for Windows NT event logging) and to specify the language of subsequent resource statements. Multiline statements are used to define menus, dialogs, string tables, accelerator tables, version information resources, and generic resources. Resource files are compiled using the Resource Compiler, rc.exe. The resulting compiled resource file (usually a file with the .res extension) is suitable for linking with other components of the application. A compiled resource file can also be specified on the cl command line.

Drawing and Device Contexts CHAPTER 6

109

Drawing and Device Contexts


IN THIS CHAPTER

6
DRAWING AND DEVICE CONTEXTS

s The GDI, Device Drivers, and Output Devices 110 s Device Contexts 111 s Coordinates 113

s Drawing Objects 122 s Clipping 129 s Drawing Functions 132 s Notes About Printing 137

110

Under the Hood: Windows and the Win32 API PART II

To say that drawing on the screen, the printer, or another output device is one of the most important aspects of a Windows application is to state the obvious. Throughout their lifetimes, Windows applications continually draw and redraw the contents of their windows in response to user actions or other events. Needless to say, applications draw to hardware devices using a series of device-independent system functions. Otherwise Windows applications, similar to their MS-DOS counterparts, would be plagued with device incompatibilities and would require device drivers for various video cards, printers, or other graphics hardware. Indeed, device independence is one of the major advantages offered by a graphical operating system like Windows.

The GDI, Device Drivers, and Output Devices


Applications draw to an output device by calling graphics device interface (GDI) functions. The GDI library containing these functions, gdi.dll, makes calls, in turn, to device-specific function libraries, or device drivers. The device drivers perform operations on the actual physical hardware. Device drivers are supplied either as part of Windows or, for less commonly used hardware, as third-party add-ons. The interrelationship between graphical applications, the GDI, device driver software, and hardware devices is schematically illustrated in Figure 6.1.

FIGURE 6.1.
Interaction between applications, the GDI, device drivers, and output devices.
Application Application Application

GDI

Device Driver

Device Driver

Device Driver

Device

Device

Device

Drawing and Device Contexts CHAPTER 6

111

Most drawing functions take a handle to a device context as one of their parameters. In addition to identifying the device on which the drawing should take place, the device context also specifies a number of other characteristics, including s Mapping logical coordinates to actual physical coordinates on the device s Use of drawing objects such as fonts, pens, or brushes to carry out the requested operation s Clipping of drawing functions to visible areas

6
DRAWING AND DEVICE CONTEXTS

Device Contexts
A device context thoroughly specifies the characteristics of a hardware device. Drawing system functions use this information to translate device-independent drawing calls into a series of device-specific operations carried out with the help of low-level driver code. Before a device context can be used, it must be created. The most generic function for creating a device context is the CreateDC function. When calling this function, applications specify the device for which the device context is created, the driver software, the physical port to which the device is attached, and device-specific initialization data. When drawing to the screen, applications need not create a device context using CreateDC. Instead, applications can retrieve a handle to a device context representing the client area of a window through the GetDC function or the entire window (including nonclient areas) through GetWindowDC. A typical GDI drawing function is the Rectangle function. An application may make the following call to draw a rectangle:
Rectangle(hDC, 0, 0, 200, 100);

This call draws a rectangle on the device identified by the handle hDC, with its upper-left corner at logical coordinates [0,0], and lower-right corner at [200,100]. Needless to say, a lot takes place behind the scenes before the actual rectangle is formed on the screen. How does the GDI know the physical coordinates corresponding to these logical coordinates? How does it know the color of the rectangle and its interior? The styles used for the rectangles contours or for filling its interior? The answer is, all this information is available as part of the device context. Coordinate transformations are defined by the mapping mode and any world transformation that may be in effect. The appearance and color of objects drawn are a function of GDI objects that have been selected into the device context. I review all of this shortly.

Device Context Types


In the case of the display, Windows distinguishes between common and private device contexts. Common device contexts represent a shared resource across applications. Private device contexts are created for windows with a window class carrying the CS_OWNDC style. Private device contexts are deleted when the window to which they belong is destroyed.

112

Under the Hood: Windows and the Win32 API PART II

Memory and Metafile Device Contexts


Device contexts typically represent physical devices, such as the display, the printer, plotters, or FAX modems. However, there are also some special device contexts in Windows. One of them I already mentioned. A memory device context is a device context that represents a bitmap. By utilizing this device context, applications can draw into bitmaps. In addition to the obvious use in creating bitmaps (such as in a bitmap editor like the Windows 95 Paint application), memory device contexts have another practical use in graphicsintensive applications. By drawing into a memory device context and transferring the contents only when the drawing is complete, applications can reduce unwanted screen flicker. Through a clever use of multiple memory device contexts, applications can create smooth animation effects. Several functions, which are reviewed in this chapter, assist in efficiently transferring bitmap data from one device context to another. A memory device context is created by a call to the CreateCompatibleDC function. This function creates a memory device context that is compatible with a specified physical device. Another type of a device context is a metafile device context. A metafile is essentially a deviceindependent record of GDI operations. Win32 recognizes two metafile types: standard and enhanced metafiles. Standard metafiles are compatible with Windows 3.1, but they do not implement complete device independence; for this reason, the use of enhanced metafiles for new applications is recommended. A metafile device context is created by calling the CreateMetaFile function or, in the case of enhanced metafiles, the CreateEnhMetaFile function. When an application is finished drawing into the metafile device context, it closes the metafile using CloseMetaFile (CloseEnhMetaFile). This call returns a metafile handle that can then be used in calls to P l a y M e t a F i l e (PlayEnhMetaFile) or the various metafile manipulation functions. A metafile handle can also be obtained by a call to GetMetaFile (GetEnhMetaFile) for metafiles that have been saved to disk previously. Relatively few applications manipulate metafiles directly. However, most applications use metafiles implicitly through OLE. The device-independent metafile format is used by OLE to graphically represent embedded or linked objects. Applications that display embedded objects thus do not need to call the OLE server application (which may not even be installed on the system) every time an OLE object needs to be rendered; instead, they just play back the recorded metafile.

Information Contexts
Information contexts are used to retrieve information about a specific device. An information context is created by a call to the CreateIC function. Creating an information context requires far less overhead than creating a device context and is therefore the preferred method for retrieving information about a device. An information context must be deleted after use by calling DeleteDC.

Drawing and Device Contexts CHAPTER 6

113

Coordinates
Applications typically specify the position and size of output objects in the form of logical coordinates. Before an object appears at a physical location on the screen or printer, a series of calculations takes place to obtain actual physical positions on the device.

6
DRAWING AND DEVICE CONTEXTS

Logical and Device Coordinates


The transformation from logical to physical coordinates, although simple in concept, can sometimes trick even the experienced Windows programmer. The mapping from logical to physical coordinates is accomplished by specifying the characteristics of the window and the viewport. The window, in this context, represents the logical coordinate space; the viewport represents the physical coordinate space of the device. For both the window and the viewport, two pairs of values must be supplied. One pair is the horizontal and vertical coordinates of the origin; the other pair is the horizontal and vertical extent. Figure 6.2 illustrates how the logical coordinates of a set of rectangles are mapped to devicespecific physical coordinates. From this illustration, it should be clear that the absolute size of the logical and physical extents should be of no consequence; what matters is their relative sizes that is, the number of logical units mapped to a physical unit or vice versa.

FIGURE 6.2.
The logical and the physical coordinate system.
Window Extent

Viewport Origin

Viewport Extent

Window Origin

On most devices, the origin of the physical coordinate system is in the upper-left corner and the vertical coordinate grows downward. In contrast, in most logical coordinate systems, the origin is in the lower-left corner and the vertical coordinate grows upward. The origin and the extent of the logical and physical coordinate systems can be set using the following four functions: SetViewportExtEx, SetViewportOrgEx, SetWindowExtEx, SetWindowOrgEx. (Use of the old functions SetViewportExt, SetViewportOrg, SetWindowExt, and SetWindowOrg is not supported in Win32.)

114

Under the Hood: Windows and the Win32 API PART II

For reference, here is how the GDI converts from logical to physical coordinates and vice versa:
Dx Dy Lx Ly = = = = (Lx (Ly (Dx (Dy xWO) yWO) xVO) yVO) * * * * xVE/xWE yVE/yWE xWE/xVE yWE/yVE + + + + xVO yVO xWO yWO

The meaning of these symbols should be fairly obvious; for example, Dx is the horizontal device coordinate, and yWE is the vertical window extent. Figure 6.3 identifies these symbols graphically.

FIGURE 6.3.
Mapping logical to physical coordinates.
[xWE, yWE]

[xVO, yVO]

[Dx, Dy]

[Lx, Ly] [xVE, yVE]

[xWO, yWO]

WARNING
Although Windows 95/98 and Windows NT all use 32-bit coordinate values in GDI function calls, only Windows NT represents coordinates internally as 32-bit values. In the case of Windows 95/98, 16-bit values are used; the upper 16 bits are simply ignored.

To facilitate easy changes from one mapping to another, Windows offers a few helper functions. These include OffsetViewportOrg , OffsetWindowOrg , ScaleViewportExt , and ScaleWindowExt. Note that an application can change the horizontal or vertical orientation of the window or viewport by specifying a negative extent value. To calculate explicitly a set of physical coordinates from logical coordinates, or vice versa, applications can use the LPtoDP and DPtoLP functions.

Constrained Mapping Modes


What has been said about mapping modes so far is true for the so-called unconstrained mapping mode.

Drawing and Device Contexts CHAPTER 6

115

The GDI supports several mapping modes; the unconstrained mapping mode MM_ANISOTROPIC is but one. Other mapping modes include the following: The origin of the logical coordinate system is the upper-left corner, and vertical coordinates are growing downward. In other words, MM_TEXT is equivalent to no mapping at all. A logical unit equals one pixel. MM_LOENGLISH. The origin is in the lower-left corner, and vertical coordinates grow upward. A logical unit is equal to one hundredth of an inch (0.01"). MM_HIENGLISH. The origin is in the lower-left corner, and vertical coordinates grow upward. A logical unit is equal to one thousandth of an inch (0.001"). MM_LOMETRIC. The origin is in the lower-left corner, and vertical coordinates grow upward. A logical unit is equal to one tenth of a millimeter (0.1 mm), or one hundredth of a centimeter (0.01 cm). MM_HIMETRIC. The origin is in the lower-left corner, and vertical coordinates grow upward. A logical unit is equal to one hundredth of a millimeter (0.01 mm), or one thousandth of a centimeter (0.001 cm). MM_TWIPS. The origin is in the lower-left corner, and vertical coordinates grow upward. A logical unit is one twentieth of a point (1/1440"). MM_ISOTROPIC. The only restriction is that horizontal and vertical logical units are of equal length. Applications can freely specify the origin of the logical and physical coordinate systems, as well as their horizontal extents. The vertical extents are computed from the horizontal by the GDI. In the six constrained mapping modes, applications are free to change the viewport and window origin, but attempts to change the viewport or window extent (through SetViewportExtEx or SetWindowExtEx) are ignored.
MM_TEXT.

6
DRAWING AND DEVICE CONTEXTS

World Coordinate Transforms


Flexible as the coordinate mapping capabilities in Windows are, Windows NT further extends these capabilities with the concept of World Coordinate Transforms. This capability makes it possible for applications to specify an arbitrary linear transformation as the mapping from the logical to the physical coordinate space. To understand how world transformations work, it is necessary to delve into coordinate geometry. Linear transformations fall into the following categories: translation, scaling, rotation, shear, and reflection. Translation (see Figure 6.4) means that constants are added to both the horizontal and vertical coordinates of an object:

116

Under the Hood: Windows and the Win32 API PART II

x1 = x + Dx

y1 = y + Dy
FIGURE 6.4.
Translation.

Scaling (see Figure 6.5) means stretching or compressing the horizontal or vertical extent of an object:

x1 = xSx
y1 = ySy
FIGURE 6.5.
Scaling.

During a rotation (see Figure 6.6), points of an object are rotated around the origin. If the angle of the rotation, , is known, the rotation can be expressed as follows:

Drawing and Device Contexts CHAPTER 6

117

x1 = x cos - y sin
y1 = x sin + y cos
FIGURE 6.6.
Rotation.

6
DRAWING AND DEVICE CONTEXTS

Shearing (see Figure 6.7) is a transformation that turns rectangles into parallelograms. Shearing adds a displacement to a points horizontal coordinate that is proportional to the vertical coordinate, and vice versa. Shearing can be expressed by the following formulae:

x1 = x + Sxy
y1 = y + Syx
FIGURE 6.7.
Shearing.

118

Under the Hood: Windows and the Win32 API PART II

A reflection mirrors an object with respect to either the horizontal or the vertical axis. Figure 6.8 shows a reflection with respect to the horizontal axis. This reflection can be expressed with the following formula:

y1 = -y
FIGURE 6.8.
Reflection with respect to the horizontal axis.

A reflection with respect to the vertical axis can in turn be expressed as follows:

x1 = -x
All these transformations can also be expressed in matrix form using 33 matrices. The matrix form of a translation is this:

[ x1 y1 1 ] = [ x y 1 ]

1 0

0 1

0 0

Dx Dy 1
The matrix form of scaling:

x1

y1 1

]=[xy1]

Sx 0 0 0 Sy 0 0 0 1

Drawing and Device Contexts CHAPTER 6

119

The matrix form of a rotation, expressed using trigonometric functions of the rotation angle:

6
DRAWING AND DEVICE CONTEXTS

cos -sin 0 [

x1

y1 1

]=[xy1]

sin cos 0 0 0 1

The matrix form of a shearing:

x1

y1 1

]=[xy1]

1 Sx Sy 1 0 0

0 0 1

A reflection with respect to the horizontal axis is expressed in matrix form as follows:

1 [ x1 y1 1 ] = [ x y 1 ] 0 0

0 -1 0

0 0 1

Finally, a reflection with respect to the vertical axis takes the following matrix form:

x1

y1 1

]=[xy1]

-1 0 0

0 1 0

0 0 1

Linear transformations can be combined. The result of two linear transformations is a third linear transformation. In matrix formulation, the resulting transformation can be expressed as the product of the matrices representing the original transformation.

NOTE
Linear transformations are not commutative. In other words, the order in which they are performed is important.

120

Under the Hood: Windows and the Win32 API PART II

Although any linear transformation can be expressed in the form of a series of the five basic transformations mentioned here, a generic linear transformation may not be a simple translation, scaling, rotation, shearing, or reflection. A generic linear transformation can be expressed as follows:

M11 [

M12 M22 Dy

0 0 1

x1

y1 1

]=[xy1]

M21 Dx

This is exactly the type of matrix an application must supply to the SetWorldTransform function. The second parameter of this function is a pointer to an XFORM structure, which is defined as follows:
typedef struct _XFORM { FLOAT eM11; FLOAT eM12; FLOAT eM21; FLOAT eM22; FLOAT eDx; FLOAT eDy; } XFORM;

Before you start worrying about matrix multiplication, I should tell you about the CombineTransform function. What this function really does is a multiplication of two transformation matrices expressed in the form of XFORM structures. Once a world transformation has been set for a device context, it will transform logical coordinates from world space to page space. Page space coordinates are further subject to the transformation specified by the mapping mode, as discussed in the section immediately preceding this one. Although applications can use the DPtoLP function to obtain the world coordinates for a given set of physical coordinates, it is sometimes useful to explicitly obtain the transformation matrix corresponding to the inverse transform. In order to obtain the inverse matrix, one should first calculate the determinant of the transformation matrix:

D=

M11 M21 Dx

M12 M22 Dy

0 0 1

M11 M21

M12 M22

= M11 M22 - M12 M21

Drawing and Device Contexts CHAPTER 6

121

If this value is zero, the inverse matrix does not exist. This happens when the world transformation is pathological, and maps many points in world space to the same point in page space, for example, when it maps world space onto a line in page space. In this case, a point in page space no longer corresponds to a unique point in world space and thus the inverse transformation is not possible. After the determinant has been obtained, the inverse matrix can be calculated easily:
M22 Dy M21 Dx M21 Dx 0 1 0 1 M22 Dy M12 Dy M11 Dx M11 Dx M22/ D = 0 1 0 1 M12 Dy M12 M22 M11 M21 M11 M12 0 0 0 0 M21 M22

6
DRAWING AND DEVICE CONTEXTS

A-1 = 1 D

- M12/ D 0
0 1

- M21/ D M11/ D ( M21 Dy - M22 Dx )/ D ( M12 Dx - M11 Dy )/ D

Accordingly, here is a short function (Listing 6.1) that creates the inverse transform of a world transform. If the inverse transform does not exist, the function returns the identity transform. The functions return value is set to FALSE in this case to indicate an error. In keeping with the tradition of other XFORM-related functions, InvertTransform also accepts the same pointer for both the input and the output XFORM structure.

Listing 6.1. Inverting a world transformation.


BOOL InvertTransform(LPXFORM lpxformResult, CONST XFORM *lpxform) { XFORM xformTmp; FLOAT D; D = lpxform->eM6*lpxform->eM22 - lpxform->eM12*lpxform->eM21; if (D == 0.0) { lpxformResult->eM11 = 1.0; lpxformResult->eM12 = 0.0; lpxformResult->eM21 = 0.0; lpxformResult->eM22 = 1.0; lpxformResult->eDx = 0.0; lpxformResult->eDy = 0.0; return FALSE; } xformTmp.eM11 = lpxform->eM22 / D; xformTmp.eM12 = -lpxform->eM12 / D;

continues

122

Under the Hood: Windows and the Win32 API PART II

Listing 6.1. continued


xformTmp.eM21 = -lpxform->eM21 / D; xformTmp.eM22 = lpxform->eM11 / D; xformTmp.eDx = (lpxform->eM21*lpxform->eDy lpxform->eM22*lpxform->eDx) / D; xformTmp.eDy = (lpxform->eM12*lpxform->eDx lpxform->eM11*lpxform->eDy) / D; *lpxformResult = xformTmp; return TRUE; }

On a final note, the SetWorldTransform function will fail unless the graphics mode for the device context has first been set to GM_ADVANCED using the SetGraphicsMode function. In order to reset the graphics mode to GM_COMPATIBLE, applications must first reset the world transformation matrix to the identity matrix.

Drawing Objects
Coordinate transformations define where a drawing is placed on the output device. What the drawing looks like is defined by the use of GDI objects. GDI offers a variety of drawing objects: pens, brushes, fonts, palettes, and bitmaps. Applications that use such objects must perform the following steps: 1. 2. 3. 4. 5. Create the GDI object. Select the GDI object into the device context. Call GDI output functions. Select the object out of the device context. Destroy the object.

These steps are demonstrated by the following code fragment, which uses a pen object to draw a rectangle into a device context identified by the handle hDC:
HPEN hPen = CreatePen(PS_SOLID, 1, RGB(0, 0, 0)); HPEN hOldPen = (HPEN)SelectObject(hDC, hPen); Rectangle(hDC, 0, 0, 100, 100); SelectObject(hOldPen); DeleteObject(hPen);

GDI objects are created using any one of a variety of functions that are introduced in the following sections. Once created, a GDI object is referred to by a handle and can be selected into the device context using the SelectObject function. (Palettes are selected using SelectPalette.) This function also returns a handle to the previously selected pen, brush, font, or bitmap; when drawing is completed, this can be used to restore the device context to its previous state. Unused objects are destroyed using the DeleteObject function.

Drawing and Device Contexts CHAPTER 6

123

It is not always necessary to create a GDI object from scratch. Applications can also retrieve predefined system objects using the GetStockObject function. GetStockObject can be used to retrieve a handle to a variety of pens, brushes, fonts, and the system palette. Although it is not necessary to call DeleteObject for a stock object, it is not harmful either.

6
DRAWING AND DEVICE CONTEXTS

Pens
Pens are used to draw lines, curves, and the contours of other shapes. A pen is created using the CreatePen function. When calling CreatePen, applications specify the pens width, style, and color. Pen color is specified as an RGB value; however, if there is a matching entry in the logical palette, Windows will usually substitute the nearest palette color. The exception is the case when the width of the pen is greater than one and the style is PS_INSIDEFRAME; in this case, Windows uses a dithered color. Dashed and dotted pen styles are not supported for pens with a width greater than one. However, in the case of Windows NT, such pens can be created using the ExtCreatePen function. This function is also available under Windows 95/98, but its utility is limited.
ExtCreatePen

also gives greater control over the shapes of joins and end caps.

Another function that can be used to create a pen is the CreatePenIndirect function. This function takes a pointer to a LOGPEN structure as its parameter. The LOGPEN structure defines the pens width, color, and style. Drawing with a pen is affected by the foreground mix mode. This mode is set using the SetROP2 function. There are several settings that define various logical operations between the pen color and the pixel color. The current mixing mode can be retrieved using the GetROP2 function.

Brushes
Brushes are used to fill the interior of drawing shapes. The use of a brush defines the interior color and pattern. A brush is created by a call to the CreateBrushIndirect function. This function accepts a pointer to a LOGBRUSH structure, which specifies the brush style, color, and pattern. A brush pattern can be based on a bitmap. If the brush style is set to the values BS_DIBPATTERN or BS_DIBPATTERNPT, the lbStyle member of the LOGBRUSH structure specifies a handle to a bitmap.

NOTE
Windows 95/98 only support 88 bitmaps in pattern brushes. If a larger bitmap is specified, only a portion of the bitmap is used.

124

Under the Hood: Windows and the Win32 API PART II

Alternatively, a brush can be hatched; in this case, the lbStyle member of the LOGBRUSH structure specifies the hatch pattern. The lbColor member specifies the foreground color of a hatched brush. However, the background color and mode are controlled by the SetBkColor and SetBkMode functions, respectively. A specific problem related to pattern and hatch brushes is the problem of brush origin. In order to provide a smooth appearance, it is necessary to align the origin of a brush bitmap or hatch brush pattern when portions of a shape are drawn at different times. Under Windows 95/98, this is accomplished by calling UnrealizeObject every time before a brush is selected into a device context. This is not necessary under Windows NT, which tracks brush origins. Applications can explicitly specify the brush origin through SetBrushOrgEx. The brush origin is a pair of coordinates that specify the displacement of the brush pattern relative to the upperleft corner of the windows client area. There are several additional functions assisting in the creation and use of brushes. Solid brushes, pattern brushes, and hatch brushes can be created by calling C r e a t e S o l i d B r u s h , CreatePatternBrush, and CreateHatchBrush, respectively. Brushes based on device-independent bitmaps can be created with CreateDIBPatternBrushPt. Drawing the interior of an object is also affected by the foreground mix mode setting as specified by a call to the SetROP2 function.

Fonts
Before an application can output any text, it must select a logical font for text output. Logical fonts are created by calling the CreateFont function. Users who are accustomed to applications that enable them to explicitly select a font by name, attributes, and size may find using CreateFont confusing at first. Although it is still possible to select a font by name, CreateFont offers a selection of a large number of additional parameters. However, one has to realize that this method of creating a logical font is yet another feature through which Windows implements complete device-independence. Instead of making applications dependent on the presence of a specific font (which may not be available on all output devices, or may not be available on different computers), fonts are selected on the basis of their characteristics. When an application requests a font through CreateFont, Windows supplies, from the set of available fonts, the font that best matches the requested characteristics. Nevertheless, it is possible to specify the name and size of a typeface to CreateFont. If this is done, Windows will attempt to select the desired font if it is available on the system. Applications can also use CreateFontIndirect to obtain a logical font. This function takes a pointer to a LOGFONT structure as its parameter. This function is especially useful in conjunction with the Font Selection Common Dialog, which returns the users choice in the form of a LOGFONT structure.

Drawing and Device Contexts CHAPTER 6

125

The EnumFontFamilies function can be used to enumerate all font families, or the fonts in a font family. Many other font-related functions assist the application programmer. For example, functions such as GetCharABCWidths help determine the width of characters. The function GetTabbedExtent or GetTextExtentPoint32 calculate the width and height of a text string. Applications can also install and remove fonts using the CreateScalableFontResource, and RemoveFontResource functions.
AddFontResource,

6
DRAWING AND DEVICE CONTEXTS

Palettes
Palettes would not be necessary if all output devices were capable of displaying the full range of colors defined by a 24-bit RGB value. Unfortunately, most lower-cost display devices offer a compromise between color depth and screen resolution. Many of todays PCs operate with a screen resolution of 800600, 1024768, or 12801024 using 256 colors. Whether a given device supports palettes can be determined by calling the GetDeviceCaps function and checking for the RC_PALETTE flag in the RASTERCAPS value. For these devices, a color palette defines the colors that are currently available for use by applications. The system palette specifies all colors that can be currently displayed by the device. However, applications cannot directly modify the system palette, although they can view its contents through the GetSystemPaletteEntries function. The system palette contains a number (usually 220) of static colors that cannot be modified by palette changes. However, applications can set the number of static colors using the SetSystemPaletteUse function. The default palette has typically 20 color entries, although this may vary from device to device. If an application requests a color that is not in the palette, Windows approximates the color by selecting the closest match from the palette or, in the case of solid brushes, by using dithering. However, this may not be sufficient for color-sensitive applications. What applications can do is specify a logical palette to replace the default palette. A logical palette may contain several colors (up to the number of colors defined by the SIZEPALETTE value, returned by GetDeviceCaps). A logical palette is created by a call to CreatePalette, and its colors can later be modified by calling SetPaletteEntries. A palette is selected into a device context using the SelectPalette function. A palette that is no longer needed can be deleted by calling DeleteObject. Before use, a palette needs to be realized using the RealizePalette function. In the case of the display device, depending on whether the palette is a foreground palette or a background palette, Windows realizes the palette differently. A palette can be selected as the foreground palette if the window for which it is selected is either the active window or a descendant of it. There can be only one foreground palette in the system at any given time. The critical difference is that a foreground palette can overwrite all nonstatic colors in the system palette. This is accomplished by marking all nonstatic entries unused before a foreground palette is realized.

126

Under the Hood: Windows and the Win32 API PART II

When a palette is realized, Windows fills the unused entries in the system palette with entries from the logical palette. If there are no more unused entries, Windows maps the remaining colors in the logical palette using the closest matching color in the physical palette or using dithering. Windows always realizes the foreground palette first, followed by the remaining background palettes on a first come, first served basis. It is important to realize that any changes to the system palette are global in nature; that is, they affect the entire display surface, not just the applications window. Changes in the system palette may cause applications to redraw their window contents. Because of this, there is an advantage to specifying a palette as a background palette; this avoids palette changes when the window for which the palette has been realized gains or loses focus. Windows defines some palette-related messages. A top-level window receives a WM_PALETTECHANGED message when Windows changes the system palette. Before a top-level window becomes the active window, it receives a WM_QUERYNEWPALETTE message, allowing the application to realize its palette. The application can do this by calling SelectPalette, UnrealizeObject, and RealizePalette. An interesting feature of palettes is palette animation. This technique uses periodic changes in the logical palette to create the impression of animation. Applications can use the AnimatePalette function for this purpose. To ensure that a given color from a palette is selected (especially important when palette animation is concerned), applications should use the PALETTEINDEX or PALETTERGB macros. An application that implements simple palette animation is shown in Listing 6.2. This application can be compiled from the command line by typing cl animate.c gdi32.lib user32.lib. Once again, note that this application works only when your video hardware is configured for a 256-color palette-enabled mode.

Listing 6.2. Palette animation.


#include <windows.h> struct { WORD palVersion; WORD palNumEntries; PALETTEENTRY palPalEntry[12]; } palPalette = { 0x300, 12, { {0xFF, 0x00, 0x00, PC_RESERVED}, {0xC0, 0x40, 0x00, PC_RESERVED}, {0x80, 0x80, 0x00, PC_RESERVED}, {0x40, 0xC0, 0x00, PC_RESERVED}, {0x00, 0xFF, 0x00, PC_RESERVED}, {0x00, 0xC0, 0x40, PC_RESERVED},

Drawing and Device Contexts CHAPTER 6


{0x00, {0x00, {0x00, {0x40, {0x80, {0xC0, } }; void Animate(HWND hwnd, HPALETTE hPalette) { HDC hDC; PALETTEENTRY pe[12]; HPALETTE hOldPal; static int nIndex; int i; for (i = 0; i < 12; i++) pe[i] = palPalette.palPalEntry[(i + nIndex) % 12]; hDC = GetDC(hwnd); hOldPal = SelectPalette(hDC, hPalette, FALSE); RealizePalette(hDC); AnimatePalette(hPalette, 0, 12, pe); nIndex = (++nIndex) % 12; SelectPalette(hDC, hOldPal, FALSE); ReleaseDC(hwnd, hDC); } void DrawCircle(HWND hwnd, HPALETTE hPalette) { HDC hDC; PAINTSTRUCT paintStruct; RECT rect; HPALETTE hOldPal; int i; hDC = BeginPaint(hwnd, &paintStruct); if (hDC != NULL) { hOldPal = SelectPalette(hDC, hPalette, FALSE); RealizePalette(hDC); GetClientRect(hwnd, &rect); DPtoLP(hDC, (LPPOINT)&rect, 2); for (i = 0; i < 12; i++) { HBRUSH hbr; HBRUSH hbrOld; hbr = CreateSolidBrush(PALETTEINDEX(i)); hbrOld = (HBRUSH)SelectObject(hDC, hbr); Rectangle(hDC, MulDiv(i,rect.right,24), MulDiv(i, rect.bottom, 24), rect.right - MulDiv(i, rect.right, 24), rect.bottom - MulDiv(i, rect.bottom, 24) ); SelectObject(hDC, hbrOld); DeleteObject(hbr); } 0x80, 0x40, 0x00, 0x00, 0x00, 0x00, 0x80, 0xC0, 0xFF, 0xC0, 0x80, 0x40, PC_RESERVED}, PC_RESERVED}, PC_RESERVED}, PC_RESERVED}, PC_RESERVED}, PC_RESERVED}

127

6
DRAWING AND DEVICE CONTEXTS

continues

128

Under the Hood: Windows and the Win32 API PART II

Listing 6.2. continued


SelectPalette(hDC, hOldPal, FALSE); EndPaint(hwnd, &paintStruct); } } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { static HPALETTE hPalette; switch(uMsg) { case WM_CREATE: hPalette = CreatePalette((LPLOGPALETTE)&palPalette); break; case WM_PAINT: DrawCircle(hwnd, hPalette); break; case WM_TIMER: Animate(hwnd, hPalette); break; case WM_DESTROY: DeleteObject(hPalette); hPalette = NULL; PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd);

Drawing and Device Contexts CHAPTER 6


SetTimer(hwnd, 1, 100, NULL); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); KillTimer(hwnd, 1); return msg.wParam; }

129

6
DRAWING AND DEVICE CONTEXTS

This application draws a series of 12 rectangles. Each rectangle has a different color, selected from a logical palette. The application also installs a timer; whenever a WM_TIMER message is received, it makes a call to the AnimatePalette function. The resulting animation creates the illusion of moving down a tunnel.

Bitmap Objects
Bitmaps are also treated as GDI objects. Typically, applications either draw into bitmaps or transfer the contents of a bitmap to an output device. What exactly is a bitmap? In terms of its visual appearance, it is a rectangular array of pixels. Each pixel can have a different color, represented in the form of one or more bits. The actual number of bits depends on the color depth of the bitmap. For example, a bitmap with a color depth of 8 bits can represent up to 256 colors; a true color bitmap can represent up to 16,777,216 colors using 24 bits per pixel. A blank GDI bitmap object is created using the CreateBitmap function. Although suitable for creating color bitmaps, it is recommended that CreateBitmap be used for monochrome bitmaps only; for color bitmaps, use the CreateCompatibleBitmap function. Bitmap objects are device-dependent. Functions exist that enable applications to write into device-independent bitmaps (DIBs). (This is what is stored in Windows BMP files.) Applications can draw into a bitmap by selecting the bitmap into a memory device context. To load a bitmap from a resource file, use the LoadBitmap function. This function creates a bitmap object and initializes it with the bitmap from the resource file, as specified by the functions second parameter.

Clipping
The technique of clipping is of fundamental importance in a multitasking windowing environment. Thanks to this technique, applications do not accidentally write to the display outside the client area of their windows, nor is a problem caused when parts of an applications window are covered or offscreen. In addition to these uses of clipping by the system, applications are also given explicit access to many clipping functions. They can define a clipping region for a device context and limit graphical output to that region.

130

Under the Hood: Windows and the Win32 API PART II

A clipping region is typically, but not always, a rectangular region. Table 6.1 summarizes the various types of regions and the corresponding functions that can be used to create them.

Table 6.1. Clipping regions. Symbolic identifier


Elliptical Region Polygonal Region Rectangular Region Rounded Rectangular Region

Description
CreateEllipticRgn, CreateEllipticRgnIndirect CreatePolygonRgn, CreatePolyPolygonRgn CreateRectRgn, CreateRectRgnIndirect CreateRoundRectRgn

NOTE
Using a nonrectangular region for clipping can be inefficient on certain devices.

Applications can select a clipping region into a device context by calling SelectObject or SelectClipRgn. The effects of these two functions are equivalent. Another function that allows combining a new region with the existing clipping region in the fashion of the CombineRgn function is SelectClipRgnExt. Another form of clipping is accomplished by the use of clip paths. Clip paths can define complex clipping shapes that could not be defined through clipping regions. A clipping path is a path created through the use of the BeginPath and EndPath functions, and then selected as the clipping path by calling SelectClipPath. Clip paths can be used to produce interesting special effects. One example is demonstrated in Listing 6.3. This application, shown in Figure 6.9, uses a text string to create a clip path. You can compile this program by typing cl clippath.c gdi32.lib user32.lib at the command line.

FIGURE 6.9.
Using clip paths.

Drawing and Device Contexts CHAPTER 6

131

Listing 6.3. Using clip paths.


#include <windows.h> #include <math.h> void DrawHello(HWND hwnd) { PAINTSTRUCT paintStruct; RECT rect; HFONT hFont; SIZE sizeText; POINT ptText; HDC hDC; double a, d, r; hDC = BeginPaint(hwnd, &paintStruct); if (hDC != NULL) { GetClientRect(hwnd, &rect); DPtoLP(hDC, (LPPOINT)&rect, 2); hFont = CreateFont((rect.bottom - rect.top) / 2, (rect.right - rect.left) / 13, 0, 0, FW_HEAVY, FALSE, FALSE, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH | FF_DONTCARE, Arial); SelectObject(hDC, hFont); GetTextExtentPoint32(hDC, Hello, World!, 13, &sizeText); ptText.x = (rect.left + rect.right - sizeText.cx) / 2; ptText.y = (rect.top + rect.bottom - sizeText.cy) / 2; SetBkMode(hDC, TRANSPARENT); BeginPath(hDC); TextOut(hDC, ptText.x, ptText.y, Hello, World!, 13); EndPath(hDC); SelectClipPath(hDC, RGN_COPY); d = sqrt((double)sizeText.cx * sizeText.cx + sizeText.cy * sizeText.cy); for (r = 0; r <= 90; r+= 1) { a = r / 180 * 3.14159265359; MoveToEx(hDC, ptText.x, ptText.y, NULL); LineTo(hDC, ptText.x + (int)(d * cos(a)), ptText.y + (int)(d * sin(a))); } EndPaint(hwnd, &paintStruct); } } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_PAINT: DrawHello(hwnd); break;

6
DRAWING AND DEVICE CONTEXTS

continues

132

Under the Hood: Windows and the Win32 API PART II

Listing 6.3. continued


case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); return msg.wParam; }

This application draws the text Hello, World! using a large Arial fontthe actual size is calculated based on the size of the client area. This text forms the clipping path. Next, a series of lines is drawn from the upper-left corner of the text rectangle; due to clipping, only the portions that fall within characters are seen.

Drawing Functions
We have reviewed the idea of a device context as the canvas onto which GDI functions paint graphics output; we have reviewed the tools GDI performs the painting with, such as pens, brushes, or fonts. This section reviews the actual drawing operations used by the GDI.

Drawing and Device Contexts CHAPTER 6

133

The typical steps taken by an application are illustrated in Figure 6.10. They include obtaining a handle to the device context, setting up the device context for drawing, performing drawing operations, restoring the previous state of the device context, and releasing the device context. Naturally, specific applications may elect to perform these steps in a different order, leave out irrelevant steps, or to invoke other initialization or drawing functions to satisfy specific requirements.

6
DRAWING AND DEVICE CONTEXTS

FIGURE 6.10.
Typical steps of GDI output.

Obtain a handle to the device context

GetDC() or CreateDC() or BeginPaint()

Optionally, save the DC state

SaveDC()

Create and select a pen

CreatePen() SelectObject()

Create and select a brush, set background

CreateBrushIndirect() SelectObject() SetBkMode() SetBkColor()

Create and select a palette

CreatePalette() SelectPalette()

Create and select a font

CreateFont() SelectObject()

Call output functions

Rectangle() Ellipse() OutText() ...

Restore the DC, delete unused objects

SelectObject() RestoreDC() DeleteObject()

or

Release the DC if necessary

EndPaint() DeleteDC()

or

134

Under the Hood: Windows and the Win32 API PART II

Lines
The simplest drawing function in Windows creates a line. A simple line is created by a call to the MoveToEx function, followed by a call to the LineTo function. The MoveToEx function updates the current position, which is a point in the coordinate space of the device context that is used by many drawing functions. The LineTo function creates a line from that position to the position specified through its parameters. The line is drawn using the pen that is currently selected into the device context. In the case of raster devices, a line is generally drawn using a DDA (digital differential analyzer) algorithm. This algorithm determines which pixels in the drawing surface should be highlighted. Specialized applications that require the use of a nonstandard DDA algorithm can use the LineDDA function. A polylinea line consisting of several line segmentsis defined by an array of points, a pointer to which is passed the Polyline function. Polyline does not use or update the current position; in contrast, PolylineTo begins drawing from the current position, and updates the current position to reflect the last point in the polyline. The PolyPolyline function can be used to draw a series of polylines using a single function call.

Curves
The simplest function to draw a curve is the Arc function. A curve drawn by this function is actually a segment of an ellipse. The arc is drawn using the current pen. The ArcTo function is identical to the Arc function, except that it also updates the current position. Win32 applications can also draw Bzier curves. Bzier curves represent a cubic interpolation between two endpoints, as defined by two control points. An example of a Bzier curve is shown in Figure 6.11.

FIGURE 6.11.
A Bzier curve.

Drawing and Device Contexts CHAPTER 6

135

The PolyBezier function draws one or more Bzier curves. One of its parameters is a pointer to an array of points used to define these curves. The endpoint of one curve serves as the starting point of the next curve; consequently, the number of points in this array must be a multiple of three plus one (the first starting point), that is, 4, 7, 10, and so on. The PolyBezierTo function is identical to the PolyBezier function except that it also updates the current position. Win32 also provides for combinations of lines and curves. The outline of a pie chart can be drawn using the AngleArc function. More complex combinations of lines and curves can be created using the PolyDraw function.

6
DRAWING AND DEVICE CONTEXTS

Filled Shapes
In addition to lines and curves, GDI drawing functions can also be used to create filled shapes. The outline of filled shapes, similar to lines and curves, is drawn using the current pen. The interior of shapes is painted using the current brush. Perhaps the simplest GDI shape is a rectangle. A rectangle is created by calling the Rectangle function. Variants of the Rectangle function include RoundRect (draws a rectangle with rounded corners), FillRect (draws the interior of a rectangle using a specific brush), FrameRect (draws the frame of a rectangle using a specific brush), and InvertRect (inverts a rectangular area on the screen). Other shapes can be created using the following functions: Ellipse, Chord, series of polygons can be drawn using the single function call PolyPolygon.
Pie, Polygon.

Regions
I have already mentioned regions and their role in clipping. However, the GDI offers several other uses for regions. Regions (summarized in Table 6.1) can be filled (FillRgn, PaintRgn), framed (FrameRgn), or inverted (InvertRgn). Regions can be combined using the CombineRgn function. To test whether two regions are identical, use the EqualRgn function. A region can be displaced by a specified offset using OffsetRgn. The bounding rectangle of a region can be obtained by calling GetRgnBox. To determine whether a specific point or a rectangle falls within the region, call PtInRegion or RectInRegion, respectively.

Bitmaps
Bitmap objects are discussed earlier in this chapter. Windows offers a variety of functions through which these objects can be copied and manipulated.

136

Under the Hood: Windows and the Win32 API PART II

Individual pixels in a bitmap can be set using the SetPixel function. The GetPixel function retrieves the color of the specified pixel. A region in a bitmap bounded by pixels of specific colors can be filled using the ExtFloodFill function. Perhaps the simplest of functions that manipulate whole bitmaps is the BitBlt function. This function copies a bitmap from one device context to another. It is often used to copy portions of a bitmap in a memory device context to the screen or vice versa; however, it can also be used to copy a bitmap to a different location within the same device context. returns an error if the source and destination device contexts are not compatible. To ensure that a memory device context is compatible with the display, use the CreateCompatibleDC function to create the device context.
BitBlt

Although BitBlt uses logical coordinates and performs the necessary scaling when copying bitmaps, it fails if a rotation or shear transformation is in effect. In addition to copying source pixels to the destination, BitBlt can also combine source and destination pixels using a variety of pixel operations. A variant of the BitBlt function is MaskBlt. This function uses a third bitmap as a mask when performing the operation. The PatBlt function paints the destination bitmap using the currently selected brush. The StretchBlt function copies the source bitmap to the destination bitmap, stretching or compressing the bitmap as necessary to fit it into the destination rectangle. The stretching can be controlled by the SetStretchBltMode function. The PlgBlt function copies the source bitmap into a destination parallelogram. The parallelogram is defined by an array of three points representing three of its vertices; the fourth vertex is calculated using the vector equation D = B + C - A. The bitmaps discussed so far are associated by a specific device context; hence, they are devicedependent. Windows also handles device-independent bitmaps, which are stored in memory or on disk. A DIB is specified through a BITMAPINFO structure. Applications can create a DIB using the CreateDIBitmap function. The bits in a DIB can be set using SetDIBits; the DIBs color table can be modified using SetDIBColorTable. The SetDIBitsToDevice function copies a DIB to a device; the StretchDIBits function can be used to copy bits from a device to a device-independent bitmap.

Paths
You have already encountered paths in the context of clipping. Paths represent complex shapes created by a series of calls to many GDI output functions, including, for example, the Rectangle, Ellipse, TextOut, LineTo, PolyBezier, Polygon functions.

Drawing and Device Contexts CHAPTER 6

137

A path is created by calling the BeginPath function, performing the drawing operations that form part of the path, and calling EndPath. The pair of calls to BeginPath and EndPath is often referred to as a path bracket. Calling EndPath selects the path into the device context. Applications can then do any of the following: s Draw the outline or interior of the path, or both (StrokePath, FillPath, StrokeAndFillPath) s Use the path for clipping (SelectClipPath) s Convert the path into a region (PathToRegion) s Modify the path (GetPath, FlattenPath, WidenPath)

6
DRAWING AND DEVICE CONTEXTS

Text Output
The simplest GDI text output function is the TextOut function. This function outputs text at the specified coordinates using the currently selected font. The TabbedTextOut function is a variant of TextOut that also expands tab characters. The PolyTextOut function can be used to output a series of text strings using a single function call. The ExtTextOut function also accepts a rectangle that can be used for opaquing or clipping. The DrawText and DrawTextEx functions can be used to output text with special formatting in a specific rectangle. Text output is affected by formatting attributes, which are set through the SetTextColor , SetTextAlign , SetBkColor , SetBkMode , SetTextCharacterExtra , and SetTextJustification functions. Applications can obtain the size of a block of text before drawing it by calling GetTabbedTextExtent or GetTextExtentPoint32.

Notes About Printing


The GDI is also responsible for providing hard copy output on printers, plotters, and outer output devices. In the case of most applications, knowing the details of the printing process is not necessary; creating output to a hardcopy device is no different from creating output to the display, using the standard set of GDI function calls on a printer device context. Sometimes it is necessary to be aware of the physical characteristics of the output page and the limitations of the device (for example, a plotter may not support bitmap operations). However, WYSIWYG applications can most often reuse, with minimal modifications, the same code for printing that they use for display output. There are several Windows components involved in printing. The primary component is the print spooler, which manages the printing process. The print processor converts spooled print jobs into calls to the device driver. The device driver generates raw output, which is then processed by the printer device. Finally, the port monitor passes raw device commands to the physical device through a specific port or network connection.

138

Under the Hood: Windows and the Win32 API PART II

There are several Win32 functions for spooling print jobs, retrieving information about jobs and printers, and control the printing process. Windows 3.1 applications often used printer escapes to carry out specific tasks. These have been superseded by new Win32 functions. New applications should not use the Escape function to control a printer.

Summary
The Windows GDI provides a device-independent set of functions that applications can use to create graphics output on all Windows-compatible output devices. The GDI is used to create output on the display screen, on printers, plotters, FAX modems, and other specialized graphics devices. All graphics output is directed to device contexts. A device context provides a description of the output device, its characteristics and parameters, and also acts as an interface between the device-independent GDI routines and the device driver software. In a manner of speaking, the device context is the canvas on which GDI drawing operations are performed. GDI uses a collection of tools for graphics output: s s s s Pens are used to draw lines or the contours of shapes. Brushes are used to fill the interior of shapes. Fonts are used for text output. Bitmaps are rectangular arrays of pixels that can be drawn using memory device contexts and manipulated or transferred between device contexts using bitmap manipulation functions. s Palettes are logical collections of colors that the GDI matches as closely as possible by configuring the color settings of the display device. s Regions are regular or irregular shapes that can be used, for example, to define clipping. Clipping is one of the key capabilities on the GDI. Thanks to clipping, applications do not need to confine their output to the visible part of their windows. Applications can also use clipping operations explicitly to create various graphical effects. The coordinate mapping, drawing tools, and clipping define how the GDI performs its drawing operations. What is actually drawn is specified by a series of graphics functions. Applications can draw lines, curves, and filled shapes; can output text; and can manipulate bitmaps. Applications can also utilize paths for a variety of purposes. The GDI provides a series of extra functions to facilitate greater control over printing and spooling to the printer. However, unless an application needs to explicitly control the printing process, it is rarely necessary to use these capabilities. Furthermore, in the case of most WYSIWYG applications, you can reuse display output code for printing with minimal modifications.

Threads and Processes CHAPTER 7

139

Threads and Processes


IN THIS CHAPTER
7
THREADS AND PROCESSES

s Multitasking in the Win32 Environment 140 s Programming with Processes and Threads 145

140

Under the Hood: Windows and the Win32 API PART II

As is the case with any evolving environment, Windows presents an odd mix of the old and the new; the obsolete, outdated, and the modern, the state-of-the-art. Nowhere is it more evident than in its multitasking capabilities, in particular the differences in those capabilities between the various Win32 platforms. The old: The cooperative multitasking environment of 16-bit Windows. Its antics and limitations survive intact in Win32s, which, although it provides a rich implementation of the Win32 programming interface, nevertheless cannot alter the underlying operating system or eliminate its limitations. The new: The multithreaded Windows NT operating system. An operating system that was designed fresh from the ground up, Windows NT offers a very robust multitasking capability, suitable for high-reliability applications (such as large corporate servers). The odd: Windows 95/98. Here, the goal of the designers was as much to implement the new capabilities as to maintain 100 percent (well, close to 100 percent anyway) compatibility with the old 16-bit Windows environment. The result is an astonishing combination: Windows 95/98 deliver a surprisingly robust multitasking capability while at the same time doing an excellent job (sometimes better than 16-bit Windows itself ) in maintaining compatibility with legacy applications. Naturally, this does not come without a price: Windows 95/98 suffer from some strange limitations, ever more likely to turn into annoying gotchas precisely because the system does such an excellent job delivering elsewhere. With its fluid multitasking capability, it may come as a surprise to the uninitiated that Windows 95/98 are just as likely to freeze because of an ill-behaved 16-bit application as Windows 3.1. (Although admittedly, Windows 95/98 do a lot better job recovering from such events.)

Multitasking in the Win32 Environment


Because the differences are substantial, it pays to examine the multitasking capabilities of the three Win32 environments separately. But first, we turn our attention to some of the fundamental concepts essential to understanding multitasking in Windows.

Multitasking Concepts
Multitasking in general refers to an operating systems capability to load and execute several applications concurrently. A multitasking operating system is considered a robust and reliable one if it successfully shields concurrent applications from each other, making them believe that they alone own the computer and its resources. Furthermore, a well-written multitasking operating system also shields applications from each others bugs; for example, if one application fails to perform array-bounds checking and writes beyond the allocated boundaries of an array, the multitasking operating system should prevent this from overwriting the memory space of another application.

Threads and Processes CHAPTER 7

141

To a large extent, multitasking operating systems rely on system hardware to implement these capabilities. For example, without the support of a memory management unit that generates an interrupt when an attempt is made to access memory at an illegal address, the operating system would have no way of knowing that such an attempt took place short of examining every single instruction in an applications code. This would be a very inefficient, timeconsuming solutioncompletely impractical, in fact. Another important aspect of multitasking is process scheduling. As most processors are capable of executing only a single stream of instructions at any given time, multitasking would obviously not be possible without the technique of context switching. A context switch, triggered by a specific event (such as an interrupt from a timer circuit or a call by the running application to a specific function), essentially consists of saving the processor context (instruction pointer, stack pointer, register contents) of one running program and loading that of another. Other no less important aspects of multitasking involve the operating systems capability to provide contention-free access to various system resources (such as the file system, the display device), prevent deadlock situations, and provide mechanisms through which concurrently executing applications can communicate with each other and synchronize their execution. (The complexity of these issues was dramatically highlighted during the first few weeks of the Mars Pathfinder mission, when the freshly landed spacecrafts computer underwent several spurious crashes caused by a deadlock among tasks competing for control of the same resource.) The degree to which various operating systems provide multitasking support varies greatly. Traditional mainframe operating systems have provided robust support in all aspects of multitasking since decades ago. On the other hand, multitasking on desktop computers is a relatively new phenomenon, largely because these machines only recently became sufficiently powerful to execute several tasks at once efficiently. (That said, many programmers are surprised to learn that even vintage MS-DOS provides rudimentary support for multitasking; this is what allowed developers to write robust Terminate and Stay Resident, or TSR, applications.) An examination of the difference between the multitasking support in the various Win32 environments quickly reveals that the primary emphasis is on the scheduling mechanism employed. In a cooperative multitasking environment (also referred to often as nonpreemptive), the operating system relies explicitly on applications to yield control by regularly calling a specified set of operating system functions. Context switching takes place at well-defined points during the execution of a program. In a preemptive multitasking environment, the operating system can interrupt the execution of an application at any time. This usually happens when the operating system responds to a hardware event, such as an interrupt from a timer circuit. An applications flow of execution can be interrupted at any point, not only at predefined spots. This raises the complexity of the system. In particular, in preemptive multitasking environments, the possibility of reentrancy becomes a distinct issue; a program may be interrupted while it is executing an operating system function, and while it is suspended, another program may call into the same operating system function, or reenter the function before the call to it from the first program was complete.

7
THREADS AND PROCESSES

142

Under the Hood: Windows and the Win32 API PART II

Another phrase often heard in the context of multitasking and Windows is threads. Perhaps the best way to describe threads is this: While multitasking offers the capability of running several programs concurrently, threads make possible several concurrent paths of execution within the same program. The introduction of this mechanism adds a powerful capability to the application programmers repertoire. The price (you knew there was a price to pay for this, didnt you?): Problems that were previously the concern of operating system authors only, such as the problems associated with reentrancy and process synchronization, are now something application programmers must also worry about.

Cooperative Multitasking
Although Visual C++ no longer supports Windows 3.1, not even through the development of 32-bit Win32s applications, the Windows 3.1 cooperative model still survives in Windows 95/ 98 and, to a lesser extent, Windows NT. Furthermore, a thorough understanding of cooperative multitasking issues can help you write well-behaved, responsive applications. Although it is true that an ill-behaved 32-bit application no longer brings the entire operating system down, it can still exhibit odd behavior or simply freeze if the rules of cooperative multitasking are not properly observed. In Windows 3.1, applications must regularly yield control to the operating system by calling one of the following functions: GetMessage, PeekMessage (without the PM_NOYIELD flag), and Yield.
Yield returns control to the operating system, allowing it to run other tasks. This function will return when the operating system returns control to the yielding program. The other two functions, GetMessage and PeekMessage, serve a dual purpose: In addition to relinquishing control, they also check for any pending messages in the applications message queue. These functions are at the heart of any applications message loop, and they must be called often in order for the application to remain responsive.

Cooperative multitasking may be a thing of the past for 32-bit programmers, but the need to regularly check and process pending messages certainly isnt. More about this in a moment.

Preemptive Multitasking in Windows NT


I must admit that ever since the early versions of Windows NT, and despite some of the compatibility problems associated with it, switching from Windows 3.1 to Windows NT always felt like stepping out from a stuffy, overcrowded room to breathe some fresh mountain air. This sensation was due in large part to NTs robust multitasking. Gone were the miseries of frozen applications, unresponsive keyboards, unsuccessful attempts to revive a system with the most drastic of methods, hitting Control+Alt+Delete. Instead, here was an operating system that always remained responsive, always offered a way to get rid of a pesky, ill-behaved program.

Threads and Processes CHAPTER 7

143

Windows NT provides preemptive multitasking for concurrent 32-bit processes. The case of 16-bit processes is special. These processes generally appear to Windows NT as a single process (the Windows On Windows, or WOW, process), although beginning with Version 3.5, Windows NT allows 16-bit processes to run in a separate memory space, meaning that a separate WOW process is started for them. Those 16-bit applications that share a WOW process, however, must abide by the rules of cooperative multitasking to allow each other to live. In other words, if a 16-bit process freezes, it will also freeze all other 16-bit processes with which it shares a WOW process; however, it will not have any ill effect on other processes, including 16-bit processes that run as part of another WOW process. Does preemptive multitasking in Windows NT mean that you can forget everything you learned about well-behaved Windows applications and start writing noncooperative code? Absolutely not, and here is the reason why. Even though Windows NT is capable of wrestling control away from an uncooperative 32-bit application and thus allow other applications to run, it will not be able to process messages aimed at the uncooperative application. Thus, if an application fails to regularly check its message queue, it will still appear unresponsive, buggy to the user. The user will not be able to interact with the application at all. Clicking on the applications window to bring it to the front will fail, and the application will not redraw parts of its window if it is uncovered when another window is moved or closed. To avoid this, an application should make every effort to regularly check its message queue and dispatch any messages in it, even when it is otherwise busy performing some lengthy task. Although failing to do so no longer threatens the integrity of the system as a whole, it certainly serves as a recipe for a very user-unfriendly application. Fortunately, there is another aspect of Windows NT multitasking that makes such lengthy processes much easier to implement. Unlike its 16-bit predecessor, Windows NT is a multithreaded operating system. A Windows NT program can easily and inexpensively create new threads of execution. For example, if it needs to perform a lengthy calculation, that task can be delegated to a secondary thread, while the primary thread continues processing messages. A secondary thread can even perform user interface functions; for example, while an applications primary thread continues processing messages sent to its main window, the application can delegate the function of processing messages for a dialog to a secondary thread. (A special terminology is used with Microsoft Foundation Classes to distinguish threads that own windows and process messages and those that do not; they are referred to as user interface threads and worker threads, respectively.)

7
THREADS AND PROCESSES

Windows 95/98: A Mixed Bag of Multitasking Tricks


Windows 95 (and its successor, Windows 98) combines the best features of both Windows 3.1 and Windows NT and loses surprisingly little in terms of tradeoffs. (I guess you can tell from this that I like Windows 95. Indeed, I like it a lot.)

144

Under the Hood: Windows and the Win32 API PART II

On the one hand, Windows 95 delivers a Windows NT-like preemptive multitasking and multithreading capability. If anything, Windows 95 is perhaps even more responsive, thanks to code that is more optimized, more specifically tailored to the Intel family of processors than the portable code of Windows NT. (Although, I must add that with the appearance of Windows NT 4.0, NT has become a close competitor of Windows 95 in these areas.) On the other hand, Windows 95 provides a remarkable degree of compatibility with legacy DOS and 16-bit Windows applications. And all this is delivered by an operating system that is only slightly more resource-hungry than its predecessor. I witnessed this firsthand, when I successfully installed Windows 95 and Visual C++ 2.1 on my 8MB 486SX25 notebook computer back in 1995. This compatibility has been accomplished, in part, by incorporating large amounts of legacy code from Windows 3.1 into Windows 95. In other words, although Windows 95 is doubtless a 32-bit operating system, some code at its very heart is old-style 16-bit code. The obvious side effect of this is that some parameters that can have a full range of 32-bit values in Windows NT are restricted to 16 bits in Windows 95 (most notably, graphics coordinates). Another, less than obvious, side effect has a direct consequence for multitasking under Windows 95. Much of the Windows 3.1 legacy code has not been designed with reentrancy in mind. In other words, because 16-bit applications participated in cooperative multitasking, there was never a chance that one was interrupted in the middle of a system call. Hence, there was no need to design mechanisms that would make it safe to repeatedly call system functions while a previous call was suspended, unfinished. Because Windows 95 processes can be interrupted at any time, Microsoft had two choices. The first was to rewrite Windows 3.1 system calls completely. Apart from being a monumental task, this approach would result in a loss of the advantage that importing Windows 3.1 legacy code represents, namely a high degree of backward compatibility. In effect, such a rewrite would result in another operating system; and that has been done, the result being Windows NT. The other, much simpler solution is simply to protect the system while its 16-bit nonreentrant parts are executing. In particular, what the Windows 95 solution means is that while one application is executing 16-bit code, all other applications are prevented from doing so. This has a very noticeable effect in the case of 16-bit applications. You see, 16-bit applications are always running in 16-bit mode. What that means is that as long as a 16-bit application has control of the processor, no other application can execute 16-bit code. Which also means that an uncooperative 16-bit application (one that fails to yield to the operating system, thus allowing other 32-bit processes to gain control) can just as effectively freeze the operating system as it did under Windows 3.1. Fortunately, Windows 95 does a much better job of recovering. For example, it can kick out the offending process and do a good job cleaning up its mess without struggling with the stability and resource allocation problems that have plagued Windows 3.1. And now, its successor, Windows 98, further improves on this operating system's stability.

Threads and Processes CHAPTER 7

145

Programming with Processes and Threads


The Win32 API contains a rich set of functions for accessing all the multitasking and multithreading features of 32-bit Windows. In some cases, these functions supersede or replace traditional UNIX or C library (or, for that matter, MS-DOS) functions. Other functions represent new areas of functionality. Yet another set of functions (for example, the yielding functions) is familiar to Windows 3.1 programmers. The remainder of this chapter reviews some of the multitasking programming techniques.

Cooperative Multitasking: Yielding in the Message Loop


Because of the Windows 3.1 heritage described earlier in this chapter, techniques for multitasking and message processing remain closely linked in Windows programming. Listing 7.1 shows the simplest Windows message loop, containing a GetMessage call. In Windows 3.1, this loop ensures that the applications message queue is processed and that other applications are allowed to gain control of the system. In Windows 95 or Windows NT, only the former of these two roles remains.

7
THREADS AND PROCESSES

Listing 7.1. A simple message loop.


int WINAPI WinMain(...) { MSG msg; ... // Application initialization goes here ... // Entering main message loop while (GetMessage(&msg, NULL, 0, 0)) { // Message dispatching goes here ... } }

// This call yields!

Processing Messages During Lengthy Processing


As I mentioned earlier, although in the 32-bit environment it is not strictly necessary for an application to yield cooperatively, it is a very good idea to continue processing messages. Freezing other applications may no longer be a concern, but it is still the programmers responsibility to ensure that his application itself remains responsive. This issue becomes particularly relevant when the application performs a lengthy task, such as a complex calculation or printing a hard copy.

146

Under the Hood: Windows and the Win32 API PART II

The sample program shown in Listing 7.2 (resource file) and Listing 7.3 (source file) demonstrates a simple technique. This is yet another example that can be compiled from the command line with the following instructions:
rc loop.rc cl loop.c loop.res user32.lib

Alternatively, you can create a Visual C++ project and add the files LOOP.C and LOOP.RC in order to compile from within the Developer Studio.

Listing 7.2. Processing loop example resource file (LOOP.RC).


#include windows.h DlgBox DIALOG 20, 20, 90, 64 STYLE DS_MODALFRAME | WS_CAPTION | WS_SYSMENU CAPTION LOOP BEGIN DEFPUSHBUTTON CANCEL IDCANCEL, 29, 44, 32, 14, WS_GROUP CTEXT Iterating -1, 0, 8, 90, 8 CTEXT 0 1000, 0, 23, 90, 8 END

Listing 7.3. Processing loop example source file (LOOP.C).


#include <windows.h> HINSTANCE hInstance; BOOL bDoAbort; HWND hwnd; BOOL CALLBACK DlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { if (uMsg == WM_COMMAND && LOWORD(wParam) == IDCANCEL) { EnableWindow(hwnd, TRUE); DestroyWindow(hwndDlg); return (bDoAbort = TRUE); } return FALSE; } void DoIterate(HWND hwndDlg) { MSG msg; int i; char buf[18]; i = 0; while (!bDoAbort) { SetWindowText(GetDlgItem(hwndDlg, 1000), _itoa(i++, buf, 10));

Threads and Processes CHAPTER 7


if (i % 100 == 0) while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) DispatchMessage(&msg); } } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { HWND hwndDlg; switch(uMsg) { case WM_LBUTTONDOWN: hwndDlg = CreateDialog(hInstance, DlgBox, hwnd, DlgProc); ShowWindow(hwndDlg, SW_NORMAL); UpdateWindow(hwndDlg); EnableWindow(hwnd, FALSE); bDoAbort = FALSE; DoIterate(hwndDlg); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hThisInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; WNDCLASS wndClass; hInstance = hThisInstance; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = LOOP; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(LOOP, LOOP, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); return msg.wParam; }

147

7
THREADS AND PROCESSES

148

Under the Hood: Windows and the Win32 API PART II

In this program, a processing-intensive loop is started when the user clicks in the client area of the applications main window. The processing in the DoIterate function is not particularly complex; it is simply incrementing the i variable and displaying the result repeatedly until the user stops the loop. Before the iteration is started, however, the application displays a modeless dialog box. Moreover, it disables user interaction with the applications main window by calling the EnableWindow function. This has basically the same effect as using a modal dialog box with one crucial difference; we do not need to call the DialogBox function, and thus we retain control while the dialog box is displayed. Inside the actual iteration loop, the function PeekMessage is called with great frequency. This ensures that the application yields control; but more importantly, it also ensures that the dialog through which the iteration can be aborted responds to user-interface events. So what is the difference between GetMessage and PeekMessage? When should you use one versus the other? Here is how I would translate the meaning of these two function calls into plain English: When you call PeekMessage, you tell the operating system that you want to know about pending messages but you also want control back because you are busy processing. GetMessage, in turn, tells a different story: It informs the operating system that you have nothing to do until a new message arrives. And that is exactly how you ensure that other processes get the most out of the CPU and that your program does not regain control repeatedly only to waste processor time with yet another empty PeekMessage call. Motto: Just because the operating system became preemptive does not mean that your applications can cease being cooperative!

NOTE
The PeekMessage call should only be used when the application actually performs background processing. Using PeekMessage instead of GetMessage not only wastes CPU time, it also prevents Windows from performing any idle-time processing such as virtual memory optimizations or power management on battery-powered systems. Therefore, PeekMessage should never be used in a general-purpose message loop.

Using a Secondary Thread


Although the previous technique can be used in programs intended for all Windows platforms, it is somewhat cumbersome. For lengthy calculations of this kind, it is much easier to use a secondary thread in which the calculation can proceed uninterrupted, uncluttered with PeekMessage calls. Consider the example shown in Listing 7.4. This example can be compiled with the same resource file as the previous one, using identical command-line instructions.

Threads and Processes CHAPTER 7

149

Listing 7.4. Processing in a secondary thread (LOOP.C).


#include <windows.h> HINSTANCE hInstance; volatile BOOL bDoAbort; HWND hwnd; BOOL CALLBACK DlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { if (uMsg == WM_COMMAND && LOWORD(wParam) == IDCANCEL) { EnableWindow(hwnd, TRUE); DestroyWindow(hwndDlg); return (bDoAbort = TRUE); } return FALSE; } DWORD WINAPI DoIterate(LPVOID hwndDlg) { int i; char buf[18]; i = 0; while (!bDoAbort) { SetWindowText(GetDlgItem((HWND)hwndDlg, 1000), _itoa(i++, buf, 10)); } } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { HWND hwndDlg; DWORD dwThreadId; switch(uMsg) { case WM_LBUTTONDOWN: hwndDlg = CreateDialog(hInstance, DlgBox, hwnd, DlgProc); ShowWindow(hwndDlg, SW_NORMAL); UpdateWindow(hwndDlg); EnableWindow(hwnd, FALSE); bDoAbort = FALSE; CreateThread(NULL, 0, DoIterate, (LPDWORD)hwndDlg, 0, &dwThreadId); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0;

7
THREADS AND PROCESSES

continues

150

Under the Hood: Windows and the Win32 API PART II

Listing 7.4. continued


} int WINAPI WinMain(HINSTANCE hThisInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; WNDCLASS wndClass; hInstance = hThisInstance; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = LOOP; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(LOOP, LOOP, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); return msg.wParam; }

Perhaps the most significant difference between the two versions of LOOP.C is that in the second version, the iteration loop within the DoIterate function no longer calls PeekMessage and DispatchMessage. It does not have to: the DoIterate function is now called from within a secondary thread, created by a call to CreateThread in the function WndProc. Instead, the primary thread of the application continues execution after creating the secondary thread, and returns processing messages in the primary message loop in WinMain. It is this primary loop that now dispatches messages for the dialog as well. Of particular interest is the changed declaration of the global variable bDoAbort. It is through this variable that the secondary thread is notified that it should stop executing; however, the value of this variable is set in the primary thread when the user dismisses the dialog. Of course, the optimizing compiler is not aware of this fact; so it is quite likely that the following construct:
while (!bDoAbort) { ... }

Threads and Processes CHAPTER 7

151

becomes optimized in such a way that the value of bDoAbort is never reloaded from memory. Why should it be? Nothing inside the while loop modifies this variables value, so the optimizing compiler can legitimately keep this value in a register, for example, which means that any changes to the value stored in memory by another thread will not be noticed by this thread. The C keyword volatile comes to our rescue. Declaring a variable volatile essentially tells the compiler that regardless of its optimization rules, the value of such a variable should be written to memory every time it is modified; and the value of such a variable should be reloaded from memory every time it is referenced. Thus, we ensure that when the primary thread sets bDoAbort to a new value, the secondary thread will actually see this change.

7
THREADS AND PROCESSES

Thread Objects
Our second LOOP.C example contained a call to CreateThread. Calling this function is the preferred method for creating a secondary thread. The return value of this function, which in this simple example we unceremoniously discarded, is a handle to the new thread object. The thread object encapsulates the properties of a thread, including, for example, its security attributes, priority, and other information. Thread manipulation functions refer to threads through thread object handles like the one returned by CreateThread. Our secondary thread in LOOP.C used the simplest exit mechanism; when the function designated as the thread function in the call to CreateThread returns, the thread is automatically terminated. This is because exiting the thread function amounts to an implicit call to ExitThread.

NOTE
The thread object remains valid even after a thread terminates, unless all handles to it (including the one obtained through CreateThread) have been closed through a call to CloseHandle.

A threads exit code (the return value of the thread function or the value passed to ExitThread) can be obtained through the GetExitCodeThread function. A threads priority can be obtained through
SetThreadPriority. GetThreadPriority

and set through

A thread can be started in a suspended state by specifying CREATE_SUSPENDED as one of the threads creation flags in the call to CreateThread. A suspended thread can be resumed by calling ResumeThread.

Creating and Managing Processes


Closely related to the creation and management of threads is the creation and management of entire processes.

152

Under the Hood: Windows and the Win32 API PART II

MS-DOS programmers have long been using the exec family of functions for spawning new processes. Windows programmers used WinExec, while those from the UNIX world are more familiar with fork. In Win32, this functionality has been consolidated into the CreateProcess function. The CreateProcess function starts an application specified by name. It returns a handle to a process object, which can later be used to refer to the newly created process. The process object encapsulates many properties of the new process, such as its security attributes or thread information. The process can be terminated by a call to the ExitProcess function. A process also terminates if its primary thread terminates.

Synchronization Objects
Our little dance with the bDoAbort variable in the previous multithreaded example represents a simplistic solution to the problem of synchronizing two or more independently executing threads. Using a global variable declared with the volatile keyword served our purposes well, but may not be adequate in more complex situations. One such situation arises when one thread has nothing to do while waiting for another thread to complete a particular task. If using a variable accessed from both threads were the only synchronization mechanism available to us, the waiting process would have to enter a loop, repeatedly checking this variables value. If it is doing that with great frequency, the result is a lot of wasted processing capacity. This problem can be alleviated somewhat by inserting a delay between subsequent checks, for example,
while (!bStatus) Sleep(1000);

Unfortunately, in many cases this is not a satisfactory solution either; we may not be able to afford to wait tens or hundreds of milliseconds before acting, but if we perform our check more frequently, we steal valuable CPU time. The Win32 API provides a set of functions that can be used to wait until a specific object or set of objects becomes signaled. There are several types of objects to which these functions apply. Some are dedicated synchronization objects, and others are objects for other purposes that nevertheless have signaled and nonsignaled states. Synchronization objects include semaphores, events, and mutexes. Semaphore objects can be used to limit the number of concurrent accesses to a shared resource. When a semaphore object is created using the CreateSemaphore function, a maximum count is specified. Each time a thread that is waiting for the semaphore is released, the semaphores count is decreased by one. The count can be increased again using the ReleaseSemaphore function. The state of an event object can be explicitly set to signaled or nonsignaled. When an event is created using the CreateEvent function, its initial state is specified, and so is its type. A manualreset event must be reset to nonsignaled explicitly using the ResetEvent function; an auto-reset

Threads and Processes CHAPTER 7

153

event is reset to the nonsignaled state every time a waiting thread is released. The events state can be set to signaled using the SetEvent function. A mutex (mut ual exclusion) object is nonsignaled when it is owned by a thread. A thread obtains ownership of a mutex object when it specifies the objects handle in a wait function. The mutex object can be released using the ReleaseMutex function. Threads wait for a single object using the functions W a i t F o r S i n g l e O b j e c t or W a i t F o r S i n g l e O b j e c t E x ; or for multiple objects, using W a i t F o r M u l t i p l e O b j e c t s , WaitForMultipleObjectsEx, or MsgWaitForMultipleObjects. Synchronization objects can also be used for interprocess synchronization. Semaphores, events, and mutexes can be named when they are created using the appropriate creation function; another process can then open a handle to these objects using OpenSemaphore, OpenEvent, and OpenMutex. Critical section objects represent a variation of the mutex object. Critical section objects can only be used by threads of the same process, but they provide a more efficient mutual exclusion mechanism. These objects are typically used to protect critical sections of program code. A thread acquires ownership of the critical section object by calling EnterCriticalSection and releases ownership using LeaveCriticalSection. If the critical section object is owned by another thread at the time EnterCriticalSection is called, this function waits indefinitely until the critical section object is released. Another simple yet efficient synchronization mechanism is interlocked variable access. Using the functions InterlockedIncrement or InterlockedDecrement, a thread can increment or decrement a variable and check the result for zero without fear of being interrupted by another thread (which might also increment or decrement the same variable before the first thread has a chance to check its value). These functions can also be used for interprocess synchronization if the variable is in shared memory. In addition to dedicated synchronization objects, threads can also wait on certain other objects. The state of a process object becomes signaled when the process terminates; similarly, the state of a thread object becomes signaled when the thread terminates. A change notification object, created by FindFirstChangeNotification, becomes signaled when a specified change occurs in the file system. The state of a console input object becomes signaled when there is unread input waiting in the consoles input buffer.

7
THREADS AND PROCESSES

Programming with Synchronization Objects


The techniques involving multiple threads and synchronization mechanisms are available not only to programs using the graphical interface, but to other programs, such as console applications, as well. In fact, the C++ example in Listing 7.5 is exactly that, a simple console application (compile with cl mutex.cpp).

154

Under the Hood: Windows and the Win32 API PART II

Listing 7.5. C++ example for the use of a mutex object.


#include <iostream.h> #include <windows.h> void main() { HANDLE hMutex; hMutex = CreateMutex(NULL, FALSE, MYMUTEX); cout << Attempting to gain control of MYMUTEX object...; cout.flush(); WaitForSingleObject(hMutex, INFINITE); cout << \n << MYMUTEX control obtained. << \n; cout << Press ENTER to release the MYMUTEX object: ; cout.flush(); cin.get(); ReleaseMutex(hMutex); }

This unremarkable little program creates a mutex object and attempts to gain ownership of it. When only a single copy of it is being executed (in a DOS window), it does not exhibit any revolutionary behavior. To really see what this application has been designed to demonstrate, open a second DOS window. Run this example in both. You will see that while the first copy successfully gains control of the mutex object, the second copy becomes suspended while attempting to do so. It remains in this suspended state as long as the first copy maintains control of the mutex object; but after the object is released through ReleaseMutex , the second copys call to WaitForSingleObject returns and it in turn gains control of the object. In fact, there is no limit to the number of processes that can cooperate through this mechanism; you could launch as many copies of this program in separate DOS windows as you like (or as memory permits). The two instances of this program were able to refer to the same mutex object because they were both referring to the object by the same name. Using the same name identified the same global object. It is easy to see how named synchronization objects can be used in a similar fashion to synchronize threads and processes, guard access to limited resources, or provide a simple communication mechanism between processes.

Summary
Multitasking represents an operating systems ability to execute several processes concurrently. The operating system accomplishes this task by regularly performing a context switch to change from one application to another. In a cooperative multitasking system, applications must explicitly yield control to the operating system. The operating system does not have the capability to interrupt the execution of a noncooperating program.

Threads and Processes CHAPTER 7

155

In a preemptive multitasking system, the operating system can and does interrupt applications based on asynchronous events, such as an interrupt from timer hardware. Such an operating system is more complex and has to deal with issues such as reentrancy. Windows 3.1 and, consequently, Win32s are examples of a cooperative multitasking system. Windows NT and Windows 95/98 are preemptive multitasking systems, but Windows 95/98 do inherit some of the limitations of Windows 3.1 through the legacy 16-bit implementation of some of their internal functions. Both Windows 95/98 and Windows NT are also multithreaded operating systems. Threads are parallel paths of execution within the same process. Although programs in Windows 95/98 and Windows NT are no longer required to yield to the operating system, they should still process messages even while performing lengthy processing. This ensures that these applications remain responsive to user-interface events. There are several methods for synchronizing the execution of threads and processes. In particular, the Win32 API provides access to special synchronization objects, such as semaphores, mutexes, and events.

7
THREADS AND PROCESSES

156

Under the Hood: Windows and the Win32 API PART II

Memory Management CHAPTER 8

157

Memory Management

IN THIS CHAPTER
s Processes and Memory 158 s 32-Bit Programs 161 s Simple Memory Management 164 s Virtual Memory and Advanced Memory Management 165 s Threads and Memory Management 174 s Accessing Physical Memory and I/O Ports 175

8
MEMORY MANAGEMENT

158

Under the Hood: Windows and the Win32 API PART II

With the advent of 32-bit Windows, memory management has become a much prettier subject than before. The immense mess of segments, selectors, all the paraphernalia of memory management in 16-bit mode on the segmented Intel processor architecture is completely and irreversibly gone. In fact, memory management has become so greatly simplified that for most applications, malloc or new are all that are needed; were this an introductory level book, I would probably be justified to end this chapter right here and move on to a different subject. That said, Win32 memory management does have its own intricacies. However, programmers are no longer forced to learn about these to perform even the simplest tasks.

Processes and Memory


Win32 provides a sophisticated memory management scheme. The two most distinguishing characteristics of this are the capability to run applications in separate address spaces, and the capability to expand the amount of memory available for allocation through the use of swap files. Both of these capabilities are part of Win32 virtual memory management.

Separate Address Spaces


For programmers familiar with 16-bit Windows, one of the most difficult ideas to adjust to is the notion that an address no longer represents a well-defined spot in physical memory. While one process may find a data item at address 0x10000000, another process may have a piece of its code running there; yet another process may regard that address as invalid. How is this accomplished? The addresses Win32 applications use are often referred to as logical addresses. Every Win32 process has the entire range of 32-bit addresses available for its use (with some operating system specific restrictions, as you will see shortly). When a Win32 process references data at a logical address, the computers memory management hardware intervenes and translates the address into a physical address (more on this later). The same logical address may (and under most circumstances, will) translate into different physical addresses for different processes. This mechanism has several consequences. Most are beneficial, but some actually render certain programming tasks a bit harder to accomplish. The most obvious benefit of having separate logical address spaces is that processes can no longer accidentally overwrite code or data belonging to another process. Invalid pointers may still cause the death of the offending process but can no longer mangle data in the address space of other processes or the operating system. On the other hand, the fact that processes no longer share the same logical address space renders the development of cooperating processes more difficult. It is no longer possible to send the address of an object in memory to another process and expect that process to be able to make use of it. That address only makes sense in the context of the sending application; in the context of the application that receives it, it is meaningless, representing a random spot in memory.

Memory Management CHAPTER 8

159

Fortunately, the Win32 API offers a set of new mechanisms for cooperating applications to use. One of these is the capability to use shared memory. Essentially, shared memory is a block of physical memory that is mapped into the logical address space of several processes. By writing data into, or reading data from, a block of shared memory, applications can cooperate.

Address Spaces
Earlier I hinted that the use of 32-bit addresses within the logical address space of a process is not entirely unrestricted. Indeed, there are some limitations. Some address ranges are reserved for use by the operating system. Moreover, the restrictions are not the same in the different Win32 environments. Using 32-bit addresses with byte-addressable memory means a total address space of 4GB (232=4,294,967,296). Of this, Windows reserves the upper 2GB for its own use, while leaving the lower 2GB available for use by the application. (Enterprise versions of Windows NT Server can reserve only 1GB, leaving 3GB available for use by applications.) Windows 95/98 further reserve the lower 4MB of the address space. This area, often referred to as the Compatibility Arena in Microsoft documentation, exists for compatibility with 16-bit DOS and Windows applications. I mentioned that Win32 applications run in separate address spaces. This is true inasmuch as the nonreserved areas of the logical address space are concerned. However, the situation of the reserved areas is somewhat different. Under Windows 95/98, all reserved areas are shared. In other words, if one application finds a particular object at a memory location in one of the two reserved areas (lower 4MB or upper 2GB), all other applications are guaranteed to find the same object there. However, applications should not rely on this behavior; otherwise, the program will be incompatible with Windows NT (and thus not qualify for Microsofts compatibility logos, for example). Besides, as you will see shortly, there are easy ways for applications to request a shared area in memory explicitly, and that mechanism works well under both Windows NT and Windows 95/98. Windows 95/98 further divide the upper 2GB into two additional arenas. The arena between 2GB and 3GB is the shared arena that holds shared memory, memory-mapped files, and some 16-bit components. The reserved system arena between 3GB and 4GB is where all of the operating systems privileged code resides. This arena is not addressable by nonprivileged application programs.

8
MEMORY MANAGEMENT

160

Under the Hood: Windows and the Win32 API PART II

Virtual Memory
In the previous sections, I discreetly avoided one question: How exactly are logical addresses mapped to physical memory? After all, most computers do not have enough memory to hold several times the 4GB of memory that each application can address. (Even if they did, the power and heat dissipation requirements of that much memory might represent somewhat of a problem. Or the price. In other words, you might need a winning lottery ticket and an industrialsize cooling unit before you can start using such a system.) The answer is that not all logical addresses of an application are actually mapped to physical storage, and those that are may not be mapped to physical memory. Ever since the introduction of Windows 3.1, Windows has been able to use a swap file. The swap mechanism expands the amount of memory that the system can use by storing unused blocks of data on disk and loading them as needed. Although swap files are several orders of magnitude slower than RAM, their use enables the system to run more applications or applications that are more resource-intensive. The reason swap files can be used efficiently is that most applications allocate blocks of memory that are rarely used. For example, if you use a word processor to edit two documents simultaneously, it may happen that while you work on one document, you do not touch the other for extended periods of time. The operating system may free up physical memory in which the other document resides, swapping the document to disk; the physical memory then becomes available for other applications. When, after some time, you switch to the other document, you may notice some disk activity and a slight delay before the document is displayed; this is when the operating system loads the relevant portions of the swap file back into memory, possibly swapping out other blocks of data not recently used in the process. Figure 8.1 shows how the operating system and the computers hardware accomplish the mapping of logical addresses. A table that is often called the page table contains information on all blocks or pages of memory. In effect, this table maps blocks in an applications logical address space to blocks in physical memory or portions of the swap file. When a logical address is mapped to actual physical memory, the mapping is dereferenced and the data is read or written as requested. As the operation is supported by the processors hardware, it does not require any extra time to resolve memory addresses this way. When the logical address maps to a block in the systems swap file, a different series of events takes place. The attempt to reference such an invalid address triggers the operating system into action. The operating system loads the requested block of data from the swap file into memory, possibly swapping out other blocks of data from memory to disk to make space. After the requested data is in physical memory and the page table is updated, control is returned to the application. The access to the requested memory location can now be completed successfully. All this is completely transparent to the application; the only sign that would indicate that the requested block of memory was not readily available is the delay caused by the swapping operation.

Memory Management CHAPTER 8

161

FIGURE 8.1.
Mapping of logical addresses to physical memory.

Application 1 0xFFFFFFFF

System Reserved

0x80000000 0x7FFFFFFF Application Code Page Table Application Stack Application Data Allocated Data 0x00000000 Application 2 0xFFFFFFFF Paging File

System Reserved

Physical Memory

0x80000000 0x7FFFFFFF Allocated Data Application Code Application Stack Application Data Allocated Data 0x00000000

8
MEMORY MANAGEMENT

The fact that logical addresses may map to physical memory locations, blocks of the swap file, or nothing at all implies interesting possibilities. Furthermore, the existence of a mechanism that maps the contents of a file (namely, the swap file) to logical addresses also carries the potential for useful features. Indeed, the Win32 API provides the means for applications to explicitly manage virtual memory and to access disk data through memory-mapped files. These and other memory management mechanisms are explored in the next section.

32-Bit Programs
Because many Windows programmers have extensive experience in programming 16-bit Windows, and because the Win32 API inherits many idiosyncrasies from its 16-bit past, it is perhaps helpful to begin our review of 32-bit memory management issues by highlighting the differences between 16- and 32-bit programs. A number of issues, such as integer size, the disappearance of the far and near specifiers, or differences in address calculations affect coding practices. The considerations presented here can also be useful as a set of guidelines for porting applications from 16 to 32 bits.

162

Under the Hood: Windows and the Win32 API PART II

Integer Size
One of the most striking differences between the 16-bit and 32-bit environments can be demonstrated by the simple example shown in Listing 8.1. This program can be compiled from the command line using cl intsize.cpp.

Listing 8.1. Determining integer size.


#include <iostream.h> void main(void) { cout << sizeof(int) = << sizeof(int); }

When you run this program, it will print the following result:
sizeof(int) = 4

UNIX programmers are probably relieved to see this result. The nightmare of trying to port UNIX programs that implicitly rely on integers and pointers both being of the same size (32 bits) is gone. Programmers of 16-bit Windows, on the other hand, are facing the added difficulty of having to review older code for any signs of an explicit dependence on the 16-bit integer size. One thing that has not changed is the size of types defined by Windows. Specifically, the types WORD and DWORD remain 16 and 32 bits wide, respectively. Use of these types when saving application data to disk ensures that the contents of a disk file remain readable by both the 16- and the 32-bit versions of the same application. In contrast, if an application used the int type when writing to disk, the contents of the disk file would be operating system dependent.

Type Modifiers and Macros


An obvious consequence of 32-bit addressing is that you no longer need to use type modifiers to distinguish between near and far pointers, or to specify huge data. Does this mean that existing programs must be modified and all references to the _near, _far, or _huge keywords must be removed? Fortunately not; the 32-bit C/C++ compiler simply ignores these keywords to ensure backward compatibility. Similarly, all the types that used to be defined in the windows.h header file, such as LPSTR for a far pointer to characters or LPVOID for a far pointer to a void type, still remain available. In the 32-bit environment, these types are simply defined to be equivalent to their near counterparts; thus, LPSTR is the same as PSTR, and LPVOID is the same as PVOID. To maintain backward compatibility (and from experience, I know that occasionally you must recompile your code with a 16-bit compiler), it is generally a good idea to continue using the correct types. This is further encouraged by the fact that the published interface to most Windows functions uses the correct (near or far) types.

Memory Management CHAPTER 8

163

Address Calculations
Naturally, if your program performs address calculations specific to the segmented Intel architecture, it needs to be modified. (Such calculations would also be in violation of the platform-independent philosophy of the Win32 API, making it difficult to compile your program under Windows NT on the MIPS, Alpha, or other platforms.) A particular case concerns the use of the LOWORD macro. In Windows 3.1, memory allocated with GlobalAlloc was aligned on a segment boundary, with the offset set to 0. Some programmers used this fact to set addresses by simply modifying the low word of a pointer variable using the LOWORD macro. Under the Win32 API, the assumption that an allocated memory block starts on a segment boundary is no longer valid. The questionable practice of using LOWORD this way will no longer work.

Library Functions
In the 16-bit environment, many functions in the standard C library had two versions: one for near addresses, and one for far addresses. It was often necessary to use both. For example, in medium model programs, one frequently had to use _fstrcpy to copy characters from or to a far memory location. In the 32-bit environment, these functions are obsolete. The header file windowsx.h defines these obsolete function names to refer to their regular counterparts. By including this file in your program that contains older source code, you can avoid having to manually comb through your source files and remove or change these obsolete function references.

8
MEMORY MANAGEMENT

Memory Models
Ever since the introduction of the IBM PC, programmers have learned to hate the multitude of compiler switches and options that control addressing behavior. Tiny, small, compact, medium, large, huge, custom memory models, address conversions, 64KB code and data segmentsto make a long story short, in 32-bit Windows, this nightmare is no longer. There is only one memory model, in which both addresses and code reside in a flat 32-bit memory space.

Selector Functions
The Windows 3.1 API contains a set of functions (for example, AllocSelector, FreeSelector) that enable applications to directly manipulate physical memory. These functions are not available in the Win32 API; 32-bit applications should not attempt to manipulate physical memory in any way. Dealing with physical memory is a task best left to device drivers.

164

Under the Hood: Windows and the Win32 API PART II

Simple Memory Management


As mentioned at the beginning of this chapter, memory allocation in the 32-bit environment is greatly simplified. It is no longer necessary to separately allocate memory and lock it for use. The distinction between global and local heaps has disappeared. On the other hand, the 32-bit environment presents a set of new challenges.

Memory Allocation via malloc and new


The venerable set of memory management functions in Windows versions prior to 3.1, such as GlobalAlloc and GlobalLock, addressed a problem specific to real mode programming of the 8086 processor family. Because applications used actual physical addresses to access objects in memory, there was no other way for the operating system to perform memory management functions. It was necessary for applications to abide by a convoluted mechanism by which they regularly relinquished control of these objects. While not using an object, applications only held a system-defined handle to it; an actual address was only obtained when the object was being read or written to, and released afterwards. This enabled the operating system to move these objects around as necessary. In other words, applications had to actively take part in memory management and cooperate with the operating system. Because malloc not only allocated memory but also locked it in place, use of this function caused dangerous fragmentation of available memory. Windows 3.1 uses Intel processors in protected mode. In protected mode, applications no longer have access to physical addresses. The operating system is able to move a memory block around even while applications hold valid addresses to it that they obtained through a call to GlobalLock or LocalLock. Using malloc not only became safe, it became the recommended practice. Several implementations of this function (such as those in Microsoft C/C++ Version 7 and later) also solved another problem. Because of a system-wide limit of 8,192 selectors, the number of times applications could call memory allocation functions without subsequently freeing up memory was limited. By providing a suballocation scheme, the newer malloc implementations greatly helped applications that routinely allocated a large number of small memory blocks. The 32-bit environment further simplifies memory allocation by eliminating the difference between global and local heaps. (It is actually possible, although definitely not recommended, to allocate memory with GlobalAlloc and free it using LocalFree.) The bottom line? In a Win32 application, allocate memory with malloc or new, release it with free or delete, and let the operating system worry about all other aspects of memory management. For most applications, this approach is perfectly sufficient.

Memory Management CHAPTER 8

165

The Problem of Stray Pointers


Working with a 32-bit linear address space has one unexpected consequence. In the 16-bit environment, every call to GlobalAlloc reserved a new selector. In protected mode in the Intel segmented architecture, selectors define blocks of memory; as part of the selector, the length of the block is also specified. Attempting to address memory outside the allocated limits of a selector resulted in a protection violation. In the 32-bit environment, automatic and static objects, global and local dynamically allocated memory, the stack, and everything else belonging to the same application shares the applications heap and is accessed through flat 32-bit addresses. The operating system is less likely to catch stray pointers. The possibility of memory corruption through such pointers is greater, increasing the programmers responsibility in ensuring that pointers stay within their intended bounds. Consider, for example, the following code fragment:
HGLOBAL hBuf1, hBuf2; LPSTR lpszBuf1, lpszBuf2; hBuf1 = GlobalAlloc(GPTR, 1024); hBuf2 = GlobalAlloc(GPTR, 1024); lpszBuf1 = GlobalLock(hBuf1); lpszBuf2 = GlobalLock(hBuf2); lpszBuf1[2000] = X; /* Error! */

8
MEMORY MANAGEMENT

In this code fragment, an attempt is made to write past the boundaries of the first buffer allocated via GlobalAlloc. In the 16-bit environment, this results in a protection violation when the attempt is made to address a memory location outside the limits of the selector reserved by the first GlobalAlloc call. In the 32-bit environment, however, the memory location referenced by lpszBuf1[2000] is probably valid, pointing to somewhere inside the second buffer. An attempt to write to this address will succeed and corrupt the contents of the second buffer. On the bright side, it is practically impossible for an application to corrupt another applications memory space through stray pointers. This increases the overall stability of the operating system.

Sharing Memory Between Applications


Because each 32-bit application has a private virtual address space, it is no longer possible for such applications to share memory by simply passing pointers to each other in Windows messages. The GMEM_DDESHARE flag, used in GlobalAlloc calls, is no longer functional. Passing the handle of a 32-bit memory block to another application is meaningless and futile; the handle only refers to a random spot in the private virtual address space of the recipient program. If it is necessary for two applications to communicate using shared memory, they can do this by using the DDEML library or by using memory-mapped files, which are described later in this chapter.

166

Under the Hood: Windows and the Win32 API PART II

Virtual Memory and Advanced Memory Management


In the Win32 programming environment, applications have improved control over how they allocate and use memory. An extended set of memory management functions is provided. Figure 8.2 shows the different levels of memory management functions in the Win32 API.

FIGURE 8.2.
Memory management functions in the 32-bit environment.

Application Program

Handle based Windows API Heap Functions

C/C++ runtime (malloc, new) Memory-Mapped File Functions

Virtual Memory Functions Virtual Memory Manager

Physical Memory

Swap File on Disk

Win32 Virtual Memory Management


Figure 8.1 might appear to suggest that pages of virtual memory must always be mapped to either physical memory or a paging (or swap) file. This is not the case; Win32 memory management makes a distinction between reserved pages and committed pages. A committed page of virtual memory is a page that is backed by physical storage, either in physical memory or in the paging file. In contrast, a reserved page is not backed by physical storage at all. Why would you want to reserve addresses without allocating corresponding physical storage? One possibility is that you might not know in advance how much space is needed for a certain operation. This mechanism enables you to reserve a contiguous range of addresses in the virtual memory space of your process, without committing physical resources to it until such resources are actually needed. When a reference to an uncommitted page is made, the operating system generates an exception that your program can catch through structured exception handling. In turn, your program can instruct the operating system to commit the page, and then it can continue the processing that was interrupted by the exception. Incidentally, this is how Windows 95 performs many of its own memory management functions, such as stack allocation or manipulating the page table itself.

Memory Management CHAPTER 8

167

One real-life example concerns sparse matrices, which are two-dimensional arrays that have most of their array elements equal to a constant value, typically zero. Sparse matrices appear frequently in scientific or engineering applications. It is possible to reserve memory for the entire matrix but commit only those pages that contain non-constant elements, thus reducing the consumption of physical resources significantly while still keeping the application code simple.

Virtual Memory Functions


An application can reserve memory through the VirtualAlloc function. With this function, the application can explicitly specify the address and the size of the memory block about to be reserved. Additional parameters specify the type of the allocation (committed or reserved) and access protection flags. For example, the following code reserves 1MB of memory, starting at address 0x10000000, for reading and writing:
VirtualAlloc(0x10000000, 0x00100000, MEM_RESERVE, PAGE_READWRITE);

Later, the application can commit pages of memory by repeated calls to the VirtualAlloc function. Memory (reserved or committed) can be freed using VirtualFree. A special use of VirtualAlloc concerns the establishment of guard pages. Guard pages act as one-shot alarms, raising an exception when the application attempts to access them. Guard pages can thus be used to protect against stray pointers that point past array boundaries, for example. can be used to lock a memory block in physical memory (RAM), preventing the system from swapping out the block to the paging file on disk. This can be used to ensure that critical data can be accessed without disk I/O. This function should be used sparingly because it can severely degrade system performance by restricting the operating systems capability to manage memory. Memory that was locked through VirtualLock can be unlocked using the VirtualUnlock function.
VirtualLock

8
MEMORY MANAGEMENT

An application can change the protection flags of committed pages of memory using the function. VirtualProtectEx can be used to change the protection flags of a block of memory belonging to another process. Finally, VirtualQuery can be used to obtain information about pages of memory; VirtualQueryEx obtains information about memory owned by another process.
VirtualProtect

Listing 8.2 shows another command-line application, one that demonstrates the use of virtual memory functions. This program can be compiled with cl -GX sparse.cpp.

Listing 8.2. Handling sparse matrices using virtual memory management.


#include <iostream.h> #include <windows.h> #define PAGESIZE 0x1000 void main(void) {

continues

168

Under the Hood: Windows and the Win32 API PART II

Listing 8.2. continued


double (*pdMatrix)[10000]; double d; LPVOID lpvResult; int x, y, i, n; pdMatrix = (double (*)[10000])VirtualAlloc(NULL, 100000000 * sizeof(double), MEM_RESERVE, PAGE_NOACCESS); if (pdMatrix == NULL) { cout << Failed to reserve memory.\n; exit(1); } n = 0; for (i = 0; i < 10; i++) { x = rand() % 10000; y = rand() % 10000; d = (double)rand(); cout << MATRIX[ << x << , << y << ] = << d << \n; try { pdMatrix[x][y] = d; } catch (...) { if (d != 0.0) { n++; lpvResult = VirtualAlloc((LPVOID)(&pdMatrix[x][y]), PAGESIZE, MEM_COMMIT, PAGE_READWRITE); if (lpvResult == NULL) { cout << Cannot commit memory.\n; exit(1); } pdMatrix[x][y] = d; } } } cout << Matrix populated, << n << pages used.\n; cout << Total bytes committed: << n * PAGESIZE << \n; for(;;) { cout << Enter row: ; cout.flush(); cin >> x; cout << Enter column: ; cout.flush(); cin >> y; try { d = pdMatrix[x][y]; } catch (...) { cout << Exception handler was invoked.\n; d = 0.0; }

Memory Management CHAPTER 8


cout << MATRIX[ << x << , << y << ] = << d << \n; } }

169

This program creates a double-precision matrix of 10,000 by 10,000 elements. However, instead of allocating a whopping 800,000,000 bytes of memory, it only allocates memory on an as-needed basis. This mechanism is especially suitable for matrices that have very few nonzero elements; in this example, only 10 out of 100,000,000 elements are set to random nonzero values. The program first reserves, but does not commit, 800,000,000 bytes of memory for the matrix. Next, it assigns random values to 10 randomly selected elements. If the element falls on a page of virtual memory that is not yet committed (has no backing in physical memory or in the paging file), an exception is raised. The exception is caught using the C++ exception handling mechanism. The exception handler checks whether the value to be assigned is nonzero; if so, it commits the page in question and repeats the assignment.

NOTE
In this simple example, we assume that the exception we catch is always a Win32 structured exception indicating a memory access violation. In complex programs, this assumption may not always be valid and a more elaborate exception handling mechanism may be necessary to reliably identify exceptions.

8
MEMORY MANAGEMENT

In the last part of the program, the user is invited to enter row and column index values. The program then attempts to retrieve the value of the specified matrix element. If the element falls on a page that has not been committed, an exception is raised; this time, it is interpreted as an indication that the selected matrix element is zero. The rudimentary user-interface loop of this program does not include a halting condition; the program can be stopped using Ctrl+C. The programs output looks similar to the following:
MATRIX[41,8467] = 6334 MATRIX[6500,9169] = 15724 MATRIX[1478,9358] = 26962 MATRIX[4464,5705] = 28145 MATRIX[3281,6827] = 9961 MATRIX[491,2995] = 11942 MATRIX[4827,5436] = 32391 MATRIX[4604,3902] = 153 MATRIX[292,2382] = 17421 MATRIX[8716,9718] = 19895 Matrix populated, 10 pages used. Total bytes committed: 40960 Enter row: 41

170

Under the Hood: Windows and the Win32 API PART II


Enter column: 8467 MATRIX[41,8467] = 6334 Enter row: 41 Enter column: 8400 MATRIX[41,8400] = 0 Enter row: 1 Enter column: 1 Exception handler was invoked. MATRIX[1,1] = 0 Enter row:

Heap Functions
In addition to their default heap, processes can create additional heaps using the HeapCreate function. Heap management functions can then be used to allocate and free memory blocks in the newly created private heap. A possible use of this mechanism involves the creation of a private heap at startup, specifying a size that is sufficient for the applications memory allocation needs. Failure to create the heap using HeapCreate can cause the process to terminate; however, if HeapCreate succeeds, the process is assured that the memory it requires is present and available. After a heap is created via HeapCreate, processes can allocate memory from it using HeapAlloc. HeapRealloc can be used to change the size of a previously allocated memory block, and HeapFree deallocates memory blocks and returns them to the heap. The size of a previously allocated block can be obtained using HeapSize. It is important to note that the memory allocated by HeapAlloc is no different from memory obtained using the standard memory allocation functions such as GlobalAlloc, GlobalLock, or malloc. Heap management functions can also be used on the default heap of the process. A handle to the default heap can be obtained using GetProcessHeap. The function GetProcessHeaps returns a list of all heap handles owned by the process. A heap can be destroyed using the function HeapDestroy. This function should not be used on the default heap handle of the process that is returned by GetProcessHeap. (Destroying the default heap would mean destroying the applications stack, global and automatic variables, and so on, with obviously disastrous consequences). The function HeapCompact attempts to compact the specified heap by coalescing adjacent free blocks of memory and decommitting large free blocks. Note that objects allocated on the heap by HeapAlloc are not movable, so the heap can easily become fragmented. HeapCompact will not unfragment a badly fragmented heap.

Memory Management CHAPTER 8

171

Windows API and C Runtime Memory Management


At the top of the hierarchy of memory management functions are the standard Windows and C runtime memory management functions. As noted earlier, these functions are likely to prove adequate for the memory management requirements of most applications. Handle-based memory management functions provided in the Windows API include GlobalAlloc and LocalAlloc, GlobalLock and LocalLock, GlobalFree and LocalFree. The C/C++ runtime library contains the malloc family of functions (malloc, realloc, calloc, free, and other functions). These functions are safe to use and provide compatibility with the 16-bit environment, should it become necessary to build applications that can be compiled as both 16-bit and 32-bit programs.

Miscellaneous and Obsolete Functions


In addition to the API functions already described, a number of miscellaneous functions are also available to the Win32 programmer. Several other functions that were available under Windows 3.1 have been deleted or become obsolete. Memory manipulation functions include CopyMemory, FillMemory, MoveMemory, and ZeroMemory. These functions are equivalent to their C runtime counterparts such as memcpy, memmove, or memset. A set of Windows API functions is provided to verify whether a given pointer provides a specific type of access to an address or range of addresses. These functions are IsBadCodePtr, IsBadStringPtr , IsBadReadPtr , and IsBadWritePtr . For the latter pair, huge versions (IsBadHugeReadPtr, IsBadHugeWritePtr) are also provided for backward compatibility with Windows 3.1. Information about available memory can be obtained using GlobalMemoryStatus. This function replaces the obsolete GetFreeSpace function. Other obsolete functions include all functions that manipulate selectors (for example, manipulate the processors stack (SwitchStackBack, SwitchStackTo); manipulate segments (LockSegment, UnlockSegment); or manipulate MS-DOS memory (GlobalDOSAlloc, GlobalDOSFree).
AllocSelector , ChangeSelector , FreeSelector );

8
MEMORY MANAGEMENT

Memory-Mapped Files and Shared Memory


Earlier in this chapter, I mentioned that applications are no longer capable of communicating using global memory created with the GMEM_DDESHARE flag. Instead, they must use memorymapped files to share memory. What are memory-mapped files? Normally, the virtual memory mechanism enables an operating system to map nonexistent memory to a disk file, called the paging file. It is possible to look at this the other way around and see the virtual memory mechanism as a method of referring to the contents of a file, namely the paging file, through pointers as if the paging file were a memory object. In other words, the mechanism maps the contents of the paging file to memory addresses. If this can be done with

172

Under the Hood: Windows and the Win32 API PART II

the paging file, why not with other files? Memory-mapped files represent this natural extension to the virtual memory management mechanism. You can create a file mapping by using the CreateFileMapping function. You can also use the OpenFileMapping function to enable an application to open an existing named mapping. The MapViewOfFile function maps a portion of the file to a block of virtual memory. The special thing about memory-mapped files is that they are shared between applications. That is, if two applications open the same named file mapping, they will, in effect, create a block of shared memory. Isnt it a bit of an overkill to be forced to use a disk file when the objective is merely to share a few bytes between two applications? Actually, it is not necessary to explicitly open and use a disk file in order to obtain a mapping in memory. Applications can submit the special handle value of 0xFFFFFFFF to CreateFileMapping in order to obtain a mapping to the system paging file itself. This, in effect, creates a block of shared memory. Listings 8.3 and 8.4 demonstrate the use of shared memory objects for intertask communication. They implement a very simple mechanism where one program, the client, deposits a simple message (a null-terminated string) in shared memory for the other program. This other program, the server, receives the message and displays it. These programs are written for the Windows 95/98 or Windows NT command line. To see how they work, start two MS-DOS windows, start the server program first in one of the windows, and then start the client program in the other. You will see the client send its message to the server; the server, in turn, displays the message it receives and then terminates.

Listing 8.3. Intertask communication using shared memory: the server.


#include <iostream.h> #include <windows.h> void main(void) { HANDLE hmmf; LPSTR lpMsg; hmmf = CreateFileMapping((HANDLE)0xFFFFFFFF, NULL, PAGE_READWRITE, 0, 0x1000, MMFDEMO); if (hmmf == NULL) { cout << Failed to allocated shared memory.\n; exit(1); } lpMsg = (LPSTR)MapViewOfFile(hmmf, FILE_MAP_WRITE, 0, 0, 0); if (lpMsg == NULL) { cout << Failed to map shared memory.\n; exit(1); } lpMsg[0] = \0; while (lpMsg[0] == \0) Sleep(1000); cout << Message received: << lpMsg << \n; UnmapViewOfFile(lpMsg);

Memory Management CHAPTER 8

173

Listing 8.4. Intertask communication using shared memory: the client.


#include <iostream.h> #include <windows.h> void main(void) { HANDLE hmmf; LPSTR lpMsg; hmmf = CreateFileMapping((HANDLE)0xFFFFFFFF, NULL, PAGE_READWRITE, 0, 0x1000, MMFDEMO); if (hmmf == NULL) { cout << Failed to allocated shared memory.\n; exit(1); } lpMsg = (LPSTR)MapViewOfFile(hmmf, FILE_MAP_WRITE, 0, 0, 0); if (lpMsg == NULL) { cout << Failed to map shared memory.\n; exit(1); } strcpy(lpMsg, This is my message.); cout << Message sent: << lpMsg << \n; UnmapViewOfFile(lpMsg); }

8
MEMORY MANAGEMENT

These two programs are nearly identical. They both start by creating a file mapping of the system paging file with the name MMFDEMO. After the mapping is successfully created, the server sets the first byte of the mapping to zero and enters a wait loop, checking once a second to see whether the first byte is nonzero. The client, in turn, deposits a message string at the same location and exits. When the server notices that the data is present, it prints the result and also exits. Both programs can be compiled from the command line: cl
mmfsrvr.cpp and cl mmfclnt.cpp.

Shared Memory and Based Pointers


A shared memory-mapped file object may not necessarily appear at the same address for all processes. Although shared memory objects are mapped to identical locations in the address spaces of Windows 95/98 processes, the same is not true in Windows NT. This can be a problem if applications want to include pointers in the shared data. One solution to this problem is to use based pointers and set them to be relative to the start of the mapping area. Based pointers are a Microsoft-specific extension of the C/C++ language. A based pointer is declared using the _based keyword, in a fashion similar to the following:
void *vpBase; void _based(vpBase) *vpData;

References through the based pointer always point to data relative to the specified base. Their utility extends beyond shared memory; based pointers can also be very useful when saving data that contains pointers to disk.

174

Under the Hood: Windows and the Win32 API PART II

Threads and Memory Management


The multithreaded nature of 32-bit Windows presents some additional challenges when it comes to memory management. As threads may concurrently access the same objects in memory, it is possible that one threads operation on a variable is interrupted by another; obviously, a synchronization mechanism is needed to avoid this. In other situations, threads may want private copies of a data object, instead of a shared copy.

Interlocked Variable Access


The first of the two problems is solved in many cases by interlocked variable access. This mechanism allows a thread to change the value of an integer variable and check the result without the fear of being interrupted by another thread. Under normal circumstances, if you increment or decrement a variable within a thread, it is possible that another thread changes the value of this variable once again before the first thread has a chance to examine its value. The functions InterlockedIncrement and InterlockedDecrement can be used to atomically increment or decrement a 32-bit value and check the result. A third function, InterlockedExchange, can be used to atomically set a variables value and retrieve the old value, without the fear of being interrupted by another thread.

Thread-Local Storage
While automatic variables are always local to the instance of the function in which they are allocated, the same is not true for global or static objects. If your code relies heavily on such objects, it may prove to be very difficult to make your application thread-safe. Fortunately, the Win32 API offers a mechanism to allocate thread-local storage. The TlsAlloc function can be used to reserve a TLS Index, which is a DWORD sized space. Threads can use this space, for example, to store a pointer to a private block of memory through the TlsSetValue and TlsGetValue functions. The TlsFree function can be used to release the TLS index. If this doesnt sound easy, dont despair. The Visual C++ compiler provides an alternative mechanism that is much easier to use. Data objects can be declared thread-local using the thread type modifier. For example,
_declspec(thread) int i;

Using _declspec(thread) is problematic in DLLs because of a problem in extending the global memory allocation of a DLL at runtime to accommodate thread-local objects. It is recommended that you use the TLS APIs, such as TlsAlloc, in code that is intended to run in a DLL.

Memory Management CHAPTER 8

175

Accessing Physical Memory and I/O Ports


Programmers of 16-bit Windows are used to the idea of accessing physical memory or the input/output ports of Intel processors directly. For example, it is possible to write a 16-bit application that accesses a custom hardware device through memory-mapped I/O. It is natural to expect that those programming practices can be carried over to the 32-bit operating system. However, this is not the case. Win32 is a platform-independent operating system specification. As such, anything that introduces platform (hardware) dependence is fundamentally incompatible with the operating system. This includes all kinds of access to actual physical hardware, such as ports, physical memory addresses, or anything else. So what can you do if your task is to write an application that communicates directly with hardware? The answer is that you require one of the various DDKs (device driver kits). Through the DDK, it is possible to create a driver library that encapsulates all low-level access to the device and keep your high-level Win32 application free of platform dependencies.

Summary
Memory management in Win32 is markedly different from memory management in 16-bit Windows. Developers need no longer be concerned about issues related to the Intel segmented architecture; on the other hand, new capabilities mean new responsibilities for the programmer. Win32 applications run in separate address spaces. A pointer in the context of one application is meaningless in the context of another. All applications have access to a 4GB address space through 32-bit addresses (although the different Win32 implementations reserve certain portions of this address space for special purposes). Win32 operating systems use virtual memory management to map a logical address in an applications address space to a physical address in memory or a block of data in the systems swap or paging file. Applications can explicitly use virtual memory management capabilities to create memory-mapped files, or to reserve, but not commit, huge blocks of virtual memory. Memory-mapped files offer a very efficient intertask communication mechanism. By gaining access to the same memory-mapped file object, two or more applications can utilize such a file as shared memory. Special features address the unique problems of memory management in threads. Through interlocked variable access, threads can perform atomic operations on shared objects. Through thread-local storage, threads can allocate privately owned objects in memory. Many of the old Windows and DOS memory management functions are no longer available. Because of the platform independence of Win32, applications can no longer access physical memory directly. If it is necessary to directly access hardware (as in the case when custom hardware is used), it may be necessary to utilize the appropriate device driver kit.

8
MEMORY MANAGEMENT

File Management CHAPTER 9

177

File Management
IN THIS CHAPTER
s File System Overview 178 s Win32 File Objects 179

s Low-Level I/O 187 s Stream I/O 188 s Special Device 189

9
FILE MANAGEMENT

178

Under the Hood: Windows and the Win32 API PART II

The Win32 API offers a set of new functions and concepts for accessing and managing disk files. This is in addition to the low-level and stream I/O functions that are available as part of the C and C++ runtime libraries. This chapter reviews all forms of file handling that are available to 32-bit Windows applications. Figure 9.1 illustrates the relationship between C/C++ style stream I/O, DOS/UNIX-style low-level I/O, and the Win32 file I/O functions.

FIGURE 9.1.
I/O functions.

Stream I/O fopen() printf() scanf() iostream

Low-level I/O _open() _read() _write()

Win32 I/O CreateFile() ReadFile() WriteFile()

Win32 applications should preferably use Win32 file I/O operations, which provide full access to Win32 security features and other attributes and also allow asynchronous, or overlapped, input and output operations.

File System Overview


A typical file is a collection of data stored on nonvolatile media, such as a magnetic disk. Files are organized into file systems. File systems implement a particular scheme for storing files on physical media, and for representing various file attributes such as filenames, permissions, and ownership information. File system information can be obtained by calling the function GetVolumeInformation. Information about the nature of the storage device can be obtained by a call to GetDriveType.

Supported File Systems


Windows NT 4.0 recognizes three file systems (support for OS/2s High Performance File System, or HPFS, was recently dropped). File Allocation Table (FAT) file systems are compatible with earlier versions of DOS. The New Technology File System (NTFS) is the native file system of Windows NT. Finally, an extension of the FAT file system, the Protected Mode FAT file system (VFAT), supports long filenames on volumes otherwise compatible with earlier versions of MS-DOS.

File Management CHAPTER 9

179

Windows 95 does not recognize NTFS volumes; however, beginning with Windows 95 OEM Service Release 2, there is support for the FAT32 file systeman enhanced, more robust and flexible version of the old FAT file system. This file system is also supported by the new version of Windows NT5.0. From an application point of view, the major difference between the various file systems is the support for special attributes. For example, NTFS volumes support the concept of file ownership and security attributes, which are unavailable in the case of FAT file systems.

CD-ROM
ISO-9660 CD-ROM volumes appear as regular FAT volumes to applications. Both Windows NT and Windows 95 support long filenames on CD-ROM.

Network Volumes
Windows supports file sharing across a network. Network file systems may appear under local drive letters through network redirection. Alternatively, applications may access files across a network using UNC (universal naming convention) names, such as \\server\vol1\myfile.txt. Different networks may or may not support long filenames.

File and Volume Compression


Starting with Version 3.51, Windows NT now supports per-file compression on NTFS volumes. On the other hand, Windows 95 supports DriveSpace compression of FAT volumes. Unfortunately, the two compression mechanisms are not compatible; at present, only uncompressed FAT volumes can be accessed by both Windows 95 and Windows NT.

Win32 File Objects


In 32-bit Windows, an open file is treated as an operating system object. It is referenced through a Win32 handle; this is not to be confused with the DOS/UNIX style file handles which are basically integers assigned by the operating system to represent open files. Because a file is a kernel object, in addition to file system operations, many other handle-based operations are also possible. For example, it is possible to use the WaitForSingleObject function on a file handle opened for console I/O.

9
FILE MANAGEMENT

180

Under the Hood: Windows and the Win32 API PART II

Creating and Opening Files


A file object is created by a call to the CreateFile function. This function can be used to both create a new file and open an existing file. The function name may appear to be a misnomer unless you realize that what the function creates is the file object, which represents either a new or an existing file on the storage device. Parameters to this function specify the access mode (read or write), file sharing mode, security attributes, creation flags, file attributes, and an optional file that serves as an attribute template. For example, to open the file C:\README.TXT for reading, one would issue the following call to CreateFile:
hReadme = CreateFile(C:\\README.TXT, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);

The first parameter is a filename. Applications can also use the UNC name. The length of the filename is limited to the value of the MAX_PATH constant. Under Windows NT, this can be circumvented by prepending \\?\ to the path and calling the wide version of CreateFile, CreateFileW. The prefix \\?\ tells the operating system not to parse the path name, which can therefore contain up to approximately 32,000 Unicode characters. Another parameter that deserves special interest is the fourth parameter; this is of type LPSECURITY_ATTRIBUTES. Through this parameter, applications request security attributes for the new file object and may also specify the security attributes for newly created files. However, for this parameter to have any effect, it must be supported by the operating system and the file system. In other words, unless the operating system is Windows NT and the file is on an NTFS volume, advanced security features will not be available. Nevertheless, one member of the SECURITY_ATTRIBUTES structure, the bInheritHandle member, is still useful; it controls whether a handle to the object is inherited by child processes. Win32 applications should not use the OpenFile function for opening files; this function is provided only for compatibility with 16-bit Windows. An open file object can be closed by calling the CloseHandle function.

Simple Input and Output


Input and output are accomplished with the help of the ReadFile and WriteFile functions. Need I say more? The simple program in Listing 9.1 (compile with cl filecopy.c) copies the contents of one file to another using the Win32 functions CreateFile, ReadFile, and WriteFile.

Listing 9.1. Copying a file using Win32 file functions.


#include <windows.h> #include <iostream.h> void main(int argc, char *argv[]) {

File Management CHAPTER 9


HANDLE hSrc, hDest; DWORD dwRead, dwWritten; char pBuffer[1024]; if (argc != 3) { cout << Usage: << argv[0] << srcfile destfile\n; exit(1); } hSrc = CreateFile(argv[1], GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); if (hSrc == INVALID_HANDLE_VALUE) { cout << Unable to open << argv[1] << \n; exit(1); } hDest = CreateFile(argv[2], GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0); if (hDest == INVALID_HANDLE_VALUE) { cout << Unable to create << argv[2] << \n; CloseHandle(hSrc); exit(1); } do { ReadFile(hSrc, pBuffer, sizeof(pBuffer), &dwRead, NULL); if (dwRead != 0) WriteFile(hDest, pBuffer, dwRead, &dwWritten, NULL); } while (dwRead != 0); CloseHandle(hSrc); CloseHandle(hDest); }

181

For random-access files, Win32 provides the SetFilePointer function to position the file pointer before reading or writing. The file pointer is a 64-bit value that determines the position of the next read or write operation within the file. The SetFilePointer function fails if it is called with a handle to a device that cannot perform seek operations, such as the console or a communication port.
SetFilePointer can also be used to retrieve the current value of the file pointer. Call this func-

9
FILE MANAGEMENT

tion as follows:
dwPos = SetFilePointer(hFile, 0, NULL, FILE_CURRENT);

This call does not move the file pointer but returns its present value.

Asynchronous I/O Operations


A recurring problem when programming interactive applications that perform file I/O is the issue of responsiveness. Typical file system calls are blocking calls; for example, a call to scanf may not return until there are enough characters in the operating systems input buffer to complete

182

Under the Hood: Windows and the Win32 API PART II

the call. This is rarely a problem with fast, hard-diskbased file systems; however, when the input operation is performed, for example, on a communication port, the problem becomes much more acute. In 32-bit Windows, there are several solutions to this problem. An obvious solution is to use multiple threads; a dedicated thread may perform the input function and remain blocked indefinitely, without affecting the responsiveness of the applications user interface, managed by another thread. A simple communication program using the multithreaded approach is demonstrated in Listing 9.2. This program can be compiled from the command line by typing cl commthrd.c. (I return to the subject of using the console and communication ports in more detail later in this chapter.)

Listing 9.2. Simple communication program using multiple threads.


#include <windows.h> volatile char cWrite; DWORD WINAPI ReadComm(LPVOID hCommPort) { HANDLE hConOut; DWORD dwCount; char c; hConOut = CreateFile(CONOUT$, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); while (TRUE) { ReadFile((HANDLE)hCommPort, &c, 1, &dwCount, NULL); if (dwCount == 1) WriteFile(hConOut, &c, 1, &dwCount, NULL); if (cWrite) { if (cWrite == 24) break; c = cWrite; WriteFile(hCommPort, &c, 1, &dwCount, NULL); cWrite = \0; } } CloseHandle(hConOut); return 0; } void main(void) { HANDLE hConIn, hCommPort; HANDLE hThread; DWORD dwThread; DWORD dwCount; COMMTIMEOUTS ctmoCommPort; DCB dcbCommPort; char c; hConIn = CreateFile(CONIN$, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); SetConsoleMode(hConIn, 0);

File Management CHAPTER 9


hCommPort = CreateFile(COM2, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); memset(&ctmoCommPort, 0, sizeof(ctmoCommPort)); ctmoCommPort.ReadTotalTimeoutMultiplier = 1; SetCommTimeouts(hCommPort, &ctmoCommPort); dcbCommPort.DCBlength = sizeof(DCB); GetCommState(hCommPort, &dcbCommPort); SetCommState(hCommPort, &dcbCommPort); hThread = CreateThread(NULL, 0, ReadComm, (LPDWORD)hCommPort, 0, &dwThread); while (TRUE) { ReadFile(hConIn, &c, 1, &dwCount, NULL); cWrite = c; if (c == 24) break; while (cWrite); } if (WaitForSingleObject(hThread, 5000) == WAIT_TIMEOUT) TerminateThread(hThread, 1); CloseHandle(hConIn); CloseHandle(hCommPort); }

183

This program uses the COM2 port for communications. In order to test it, you should have a modem attached to that port. If your modem is attached to a different port, change the port name in the second CreateFile call and recompile the application. To test the program, simply run it and try to issue modem commands (for example, ATI1); the modem should respond with the appropriate response codes. After opening the communication port COM2 for reading and writing and the console for input, the program proceeds with initializing the port. As part of the initialization, it sets up a one millisecond read time-out; it also initializes communications through a DCB structure. (Yes, the seemingly superfluous calls to GetCommState/SetCommState are actually necessary; without them, the port will not function properly under Windows 95/98.) After initializing the communication port, the program creates a secondary thread, which opens the console for writing. The purpose of the secondary thread is to perform input and output on the communication port in a loop, while the primary thread does the same on the console. Whenever the primary thread receives a character on the console, it places the character in a global variable and waits until the secondary thread processes it. Whenever the secondary thread receives a character from the communication port, it writes that character to the console.

9
FILE MANAGEMENT

NOTE
A seemingly more elegant solution would be to eliminate the use of the global variable cWrite and write to the communication port directly in the main thread. This would also help us avoid using a read time-out value, making the secondary thread somewhat more
continues

184

Under the Hood: Windows and the Win32 API PART II

continued

efficient. However, this approach would not work: If the communication port is opened for synchronous I/O, and a blocking call is made in one thread (for example, a call to ReadFile that blocks until a character becomes available), all other calls for the same port will also become blocked. Thus it is not possible, for instance, to have a blocking ReadFile pending in one thread of your application while performing repeated WriteFile calls in another. (Actually, this technique works under Windows 95/98 but fails under Windows NT. Thanks to all the readers who have called my attention to this difference!)

The loops are terminated when a Ctrl+X character is received from the keyboard. Although using multiple threads is always a viable option for 32-bit applications, it may not always be the most convenient solution. Another approach available to 32-bit applications is the use of overlapped I/O. Overlapped I/O allows an application to initiate an I/O operation in a nonblocking fashion. For example, if an application uses the ReadFile function for overlapped input, the function will return even if the input operation has not yet been completed. After the operation is complete, the application can retrieve results using the GetOverlappedResult function. Applications can also use the ReadFileEx and WriteFileEx functions for overlapped I/O operations.

NOTE
Under Windows 95/98, overlapped I/O operations cannot be used on disk files.

Overlapped input can also be used in combination with a synchronization event. Processes can use the synchronization event to receive notification when the I/O operation has been completed. Using events and the WaitForMultipleObjects function, it is possible to wait for input on several input devices at once. This is exactly the technique demonstrated by the second version of this simple communication program, shown in Listing 9.3. This program can also be compiled by a simple command-line instruction, cl commovio.c.

Listing 9.3. Simple communication program using overlapped I/O.


#include <windows.h> void main(void) { HANDLE hConIn, hConOut, hCommPort; HANDLE hEvents[2]; DWORD dwCount; DWORD dwWait; COMMTIMEOUTS ctmoCommPort; DCB dcbCommPort; OVERLAPPED ovr, ovw;

File Management CHAPTER 9


INPUT_RECORD irBuffer; BOOL fInRead; char c; int i; hConIn = CreateFile(CONIN$, GENERIC_READ /*| GENERIC_WRITE*/, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); SetConsoleMode(hConIn, 0); hConOut = CreateFile(CONOUT$, GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); hCommPort = CreateFile(COM2, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, 0); memset(&ctmoCommPort, 0, sizeof(ctmoCommPort)); SetCommTimeouts(hCommPort, &ctmoCommPort); dcbCommPort.DCBlength = sizeof(DCB); GetCommState(hCommPort, &dcbCommPort); SetCommState(hCommPort, &dcbCommPort); SetCommMask(hCommPort, EV_RXCHAR); memset(&ovr, 0, sizeof(ovr)); memset(&ovw, 0, sizeof(ovw)); ovr.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); hEvents[0] = ovr.hEvent; hEvents[1] = hConIn; fInRead = FALSE; while (1) { if (!fInRead) while (ReadFile(hCommPort, &c, 1, &dwCount, &ovr)) if (dwCount == 1) WriteFile(hConOut, &c, 1, &dwCount, NULL); fInRead = TRUE; dwWait = WaitForMultipleObjects(2, hEvents, FALSE, INFINITE); switch (dwWait) { case WAIT_OBJECT_0: if (GetOverlappedResult(hCommPort, &ovr, &dwCount, FALSE)) if (dwCount == 1) WriteFile(hConOut, &c, 1, &dwCount, NULL); fInRead = FALSE; break; case WAIT_OBJECT_0 + 1: ReadConsoleInput(hConIn, &irBuffer, 1, &dwCount); if (dwCount == 1 && irBuffer.EventType == KEY_EVENT && irBuffer.Event.KeyEvent.bKeyDown) for (i = 0; i < irBuffer.Event.KeyEvent.wRepeatCount; i++) { if (irBuffer.Event.KeyEvent.uChar.AsciiChar) { WriteFile(hCommPort, &irBuffer.Event.KeyEvent.uChar.AsciiChar, 1, &dwCount, &ovw);

185

9
FILE MANAGEMENT

continues

186

Under the Hood: Windows and the Win32 API PART II

Listing 9.3. continued


if (irBuffer.Event.KeyEvent.uChar.AsciiChar == 24) goto EndLoop; } } } } EndLoop: CloseHandle(ovr.hEvent); CloseHandle(hConIn); CloseHandle(hConOut); CloseHandle(hCommPort); }

As before, if your modem is not attached to COM2, it may be necessary to change the port name in the second CreateFile call and recompile the program before using it. Like its multithreaded counterpart, this program also begins by opening the console and the communication port and initializing the port. Part of the port initialization is a call to the SetCommMask function, which enables read event notifications for that port. The communication port is opened with the FILE_FLAG_OVERLAPPED attribute. This enables overlapped I/O operations. When the programs main loop is entered, a call is made to ReadFile, passing to it a pointer to an OVERLAPPED structure. may return data immediately if data is available on the port. If not, ReadFile still returns, but signals an error; GetLastError can be used to check for the error code ERROR_IO_PENDING. (For simplicity, this part has been left out from the code; we just assume that any ReadFile error indicates pending input.)
ReadFile

The heart of this program is the call to WaitForMultipleObjects. The function waits on two objects: an event object that was specified as part of the OVERLAPPED structure used in reading the communication port and the console input object. In the case of the latter, it is not necessary to use overlapped I/O; the console object has its own signaled state indicating that data is waiting in the consoles input buffer. When WaitForMultipleObjects returns, it indicates that data arrived either on the console or on the communication port. It is the subsequent switch statement that distinguishes between the two. Retrieving the console event requires code that is a bit tricky. Unfortunately, a simple ReadFile would not suffice as it only retrieves the key down event; it leaves the key up event in the consoles input buffer, leaving the console object in a signaled state. A subsequent ReadFile would then result in a blocking read, waiting until a key is pressed once again and another key down event is generated. Because of this behavior, it is necessary to use low-level console functions to retrieve (and discard) all console events, so when WaitForMultipleObjects is called again, the console object would no longer be signaledat least not until the user presses a key again.

File Management CHAPTER 9

187

Low-Level I/O
Figure 9.1 makes it obvious that the term low-level I/O is somewhat of a misnomer for file descriptor-based I/O operations. Indeed, this term is a relic, a leftover from DOS and UNIX; although Windows NT provides these functions for compatibility with those operating systems, they are effectively implemented using CreateFile, ReadFile, and WriteFile.

File Descriptors
A file descriptor is an integer identifying an open file. A file descriptor is obtained when an application uses the _open or _creat functions. Note that throughout the runtime library documentation, file descriptors are often referred to as file handles; once again, this is not to be confused with Win32 handles for file objects. A handle returned by CreateFile and a file descriptor obtained by calling _open are not compatible.

Standard File Descriptors


Win32 console applications have access to the standard input and output file descriptors. These are summarized in Table 9.1.

Table 9.1. Standard file descriptors. File descriptor Stream name


0 1 2
stdin stdout stderr

Description Standard Input Standard Output Standard Error

9
FILE MANAGEMENT

_stdaux.

Note that MS-DOS programs also have access to two additional file descriptors, _stdprn and These file descriptors are not available for Win32 applications.

Low-Level I/O Functions


A file can be opened for low-level I/O using the _open function. A new file can be created for low-level I/O using creat. Both of these functions have wide character versions, that is, versions which accept a Unicode filename string under Windows NT: _wopen and _wcreat. Reading and writing can be performed by calling the _read or _write functions. Seeking within the file is accomplished by calling _lseek. The current position within the file can be retrieved by calling _tell.

188

Under the Hood: Windows and the Win32 API PART II

The contents of any buffers maintained by Windows can be committed to disk by calling _commit. The file can be closed by calling _close. The _eof function can be used to test for an end-of-file condition. All low-level I/O functions use the errno global variable to indicate other error conditions. The names of all these functions begin with an underscore to indicate that they are not part of the standard ANSI function library. However, for programs that may use the old names of these functions, Microsoft provides the oldnames.lib library.

Stream I/O
C/C++ stream I/O functions are among the most frequently used I/O functions. Not too many programs exist that contain no calls to printf or at least one FILE pointer.

Stream I/O in C
C programs that use stream I/O utilize the FILE structure and related family of functions. A file is opened for stream I/O by calling the fopen function. This function, if successful, returns a pointer to a FILE structure, which can be used in subsequent operations, such as calls to fscanf, fprintf, fread, fwrite, fseek, ftell, or fclose. The Visual C++ runtime library supports all standard stream I/O functions as well as several functions that are Microsoft specific. For applications that mix calls to stream I/O and low-level I/O functions, the _fileno function can be used to obtain a file descriptor for a given stream (identified by a FILE pointer). The _fdopen function can be used to open a stream and associate it with a file descriptor that identifies a previously opened file. Applications can also access the standard input, standard output, and standard error through the predefined streams stdin, stdout, and stderr.

Stream I/O in C++ (The iostream Classes)


The Visual C++ runtime library contains a complete implementation of the C++ iostream classes. The iostream class library implements a family of C++ classes, shown in Figure 9.2. The base class of all iostream classes is the ios class. Normally, applications do not derive classes from ios directly. Instead, they use one of the derived classes: istream or ostream. Variants of the istream class include istrstream (operates on an array of characters stored in memory), ifstream (operates on a disk file), and istream_withassign (a variant of istream that enables assignments to work). Variants of ostream include ostrstream (stream output to a character array), ofstream (stream output to a file), and ostream_withassign (variant of ostream that allows assignment).

File Management CHAPTER 9

189

FIGURE 9.2.
The C++ iostream classes.

ios

streambuf

istream

filebuf

ifstream

stdiobuf

ifstream_withassign

strstreambuf

istrstream

ostream

iostream

ofstream

fstream

ostream_withassign

stdiostream

ostrstream

strstream

The predefined stream object cin, representing standard input, is of type istream_withassign. The predefined objects cout, cerr, and clog, which represent standard output and standard error, are of type ostream_withassign. The class iostream combines the functionality of the istream and ostream classes. Derived classes include fstream (for file I/O), strstream (for stream I/O on a character array), and stdiostream (for standard I/O). All ios-derived objects make use of the streambuf class or the derived classes filebuf, stdiobuf, or strstreambuf for I/O buffering.

9
FILE MANAGEMENT

Special Devices
In addition to handling disk files, the Win32 file management routines can be used to handle many other types of devices. These include the console, communication ports, named pipes, and mailslots. Functions such as ReadFile or WriteFile may also accept socket handles created by the WinSock functions socket or accept depending on the WinSock implementation. The next section discusses console and communication port I/O.

Console I/O
Win32 applications can use the CreateFile, ReadFile, and WriteFile functions to perform console input and output. Consoles provide an interface for character-based applications.

190

Under the Hood: Windows and the Win32 API PART II

Unless its input or output are redirected, an application inherits file handles to the console that can be obtained by calling the function GetStdHandle. However, if the applications standard handles are redirected, GetStdHandle returns the redirected handles. In this case, applications can open the console explicitly by using the special filenames CONIN$ and CONOUT$. When opening the console for input or output, be sure to specify the FILE_SHARE_READ or FILE_SHARE_WRITE sharing mode, respectively. Also, use the OPEN_EXISTING creation mode. For example,
CreateFile(CONIN$, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);

In order to be able to perform certain operations on a console opened for reading (such as flushing the input buffer or setting the console mode), it may be necessary to open the console for reading and writing (GENERIC_READ | GENERIC_WRITE). By default, the console is opened for line-oriented input. The SetConsoleMode function can be used to change the input and output mode. For example, to set the console to raw input mode (every character is returned immediately, no control character processing takes place) use the following call:
SetConsoleMode(hConsole, 0);

The ReadFile function can be used to read keyboard input from the console. However, it is recommended that applications use ReadConsole instead where applicable; unlike ReadFile, ReadConsole can handle both ASCII and Unicode input on Windows NT. To write to the console, the WriteFile function can be used. WriteConsole has the same functionality, but it can also handle Unicode output on Windows NT, and thus it is the preferred function for console output. What I have explained so far may give the impression that using the console is just a glorified way of accomplishing what can easily be done using standard C/C++ library functions. This is not so. A console has many capabilities other than providing a facility for keyboard input and character output. In addition to character input and output, a console also handles mouse input and provides some window management functions for the console window. The low-level console input functions ReadConsoleInput and PeekConsoleInput can be used to retrieve information about keyboard, mouse, and window events. On the output side, the low-level output function WriteConsoleOutputAttribute can be used to write text and background color attributes to the console. Graphics Windows applications do not by default have access to a console. These applications can explicitly create a console by calling the AllocConsole function. A process can detach itself from its console by calling the FreeConsole function. The console windows title and position can be controlled using the SetConsoleTitle and SetConsoleWindowInfo functions, respectively.

File Management CHAPTER 9

191

Communication Ports
Communication ports are opened and used via the CreateFile, ReadFile, and WriteFile functions. That said, there are several other functions that applications must use to set up the communication ports and control their behavior. The basic setup of a communication port takes place using a DCB, or device control block structure. Members of this structure specify the baud rate, parity, data and stop bits, handshaking, and other aspects of port behavior. The current settings can be obtained using the GetCommState function and can be set by calling the SetCommState function. A helper function, BuildCommDCB, can be used to fill in parts of this structure on the basis of a string formatted in the style of an MS-DOS MODE command. Read and write time-out behavior is controlled through the COMMTIMEOUTS structure. Current time-outs can be retrieved by calling GetCommTimeouts and set by calling SetCommTimeouts. The helper function BuildCommDCBAndTimeouts can be used to fill in both a DCB structure and a COMMTIMEOUTS structure using command strings. The size of the input and output buffers can be controlled by calling the SetupComm function. The WaitCommEvent function can be used to wait for a specific event to occur on the communication port. The SetCommBreak function places the communication line in a break state. The state can be cleared by calling the ClearCommBreak function. The ClearCommError function can be used to clear an error condition. This function also reports the status of the communication device. The PurgeComm function can be used to clear any I/O buffers associated with the communication port, and to interrupt pending I/O operations. The TransmitCommChar function transmits a character on the communication port ahead of any pending data in the output buffer. This function can be used, for example, to transmit interrupt characters such as Ctrl+C. The communication port can also be opened for overlapped I/O operations. An event mask, which controls which events will set the state of the event object (specified as part of the OVERLAPPED structure), can be set using the SetCommMask function. The current event mask can be retrieved by calling the GetCommMask function. Low-level access to port functions is provided by the functions DeviceIoControl.
EscapeCommFunction

9
FILE MANAGEMENT

and

Further information about the communication port and its status can be obtained by calling the GetCommProperties and GetCommModemStatus functions. Windows 95 also offers the function CommConfigDialog, which displays a driver-specific configuration dialog for the specified communication port.

192

Under the Hood: Windows and the Win32 API PART II

Summary
Win32 applications can access files through three distinct sets of file management functions: stream and low-level I/O functions that are part of the C/C++ runtime libraries and Win32 file management functions. A file is a collection of data on a storage device, such as a disk. Files are organized on the device into file systems. Windows NT recognizes the DOS FAT, protected-mode FAT, NTFS, and HPFS file systems. Windows 95/98, in contrast, can only deal with FAT and protected-mode FAT file systems. Windows NT supports file-level compression on NTFS file systems; Windows 95, in turn, supports volume compression through DriveSpace. On NTFS volumes, Windows NT also supports advanced security features. In addition to disks, files may also be accessed on CD-ROM and remote volumes across the network. The Win32 file management functions treat an open file as an operating system object, referred to by a handle. At the heart of Win32 file management are the functions CreateFile, ReadFile, and WriteFile. Through file handles, I/O operations can be performed synchronously and asynchronously; for the latter, applications can use the technique of overlapped I/O. Overlapped I/O enables an application to regain control after an I/O call, even before the I/O operation is finished, and do something else until it is notified of the operations completion. Win32 file management routines can also be used to perform I/O on the standard input, standard output, and standard error. Handles to these can be obtained by calling the GetStdHandle function. Applications can also access files through DOS/UNIX style low-level I/O functions. The names of these functions are preceded by an underscore (for example, _open, _read) to indicate that they are not part of the standard ANSI library. However, applications can be linked with the oldnames.lib library file if the use of the old names without underscores is desired. In addition to low-level I/O, applications can also use stream I/O. This includes C functions such as fopen, fprintf, fscanf, or fclose; it also includes the C++ iostream classes. The Win32 file management functions can also be used to access special devices. They include the console, communication ports, named pipes and mailslots, and sockets opened by calls to socket or accept. Several special functions provide fine control over console and communication port I/O.

The Windows Clipboard CHAPTER 10

193

The Windows Clipboard

10

IN THIS CHAPTER
s Clipboard Formats 194 s Clipboard Operations 196 s A Simple Implementation 199

10
THE WINDOWS CLIPBOARD

194

Under the Hood: Windows and the Win32 API PART II

Not too long ago, the Windows Clipboard represented the only means for applications to exchange data with each other. Before the glorious days of OLE embedding and Drag and Drop, users had to use clipboard operations such as cutting, copying, and pasting to transfer data from one application to another, or even to move data within the same application. These days, the clipboard is often forgotten; yet just because it is overshadowed by OLE, it does not mean that applications can stop supporting clipboard operations. Furthermore, various clipboard-related concepts survive even when applications exchange data using more advanced methods. What exactly is the clipboard? Perhaps it is best defined as a Win32 facility where applications can place data so that it becomes available to all running applications, in a variety of formats that are supported by the operating system, or are application specific.

Clipboard Formats
Applications place data on the clipboard using the SetClipboardData function. In addition to providing a handle to the data object, this function also accepts a parameter specifying the format of the data. Applications are encouraged to provide data in a variety of formats; for example, a word processor program may place data on the clipboard using both a private format and a plain text format that is usable by other applications, such as the Notepad. The three types of clipboard formats available to applications include standard formats, registered formats, and private formats.

Standard Clipboard Formats


A multitude of standard clipboard formats exists, identified by symbolic constants. These formats are summarized in Table 10.1. In the cases when the application is supposed to provide a handle of a specific type when calling SetClipboardData, the handle type is indicated. In other cases, the handle passed to SetClipboardData is typically a handle to a block of memory allocated via the GlobalAlloc function.

Table 10.1. Standard clipboard formats. Format Description


Text formats Text containing characters from the OEM character set Text containing characters from the ANSI character set Text containing Unicode characters

CF_OEMTEXT CF_TEXT CF_UNICODETEXT

The Windows Clipboard CHAPTER 10

195

Format

Description

Bitmap formats CF_BITMAP Device-dependent bitmap (HBITMAP) CF_DIB Device-independent bitmap (HBITMAPINFO) CF_TIFF Tagged Image File Format Metafile formats CF_ENHMETAFILE Enhanced metafile (HENHMETAFILE) CF_METAFILEPICT Windows metafile (METAFILEPICT) Substitute formats for private formats CF_DSPBITMAP Bitmap representation of private data CF_DSPENHMETAFILE Enhanced metafile representation of private data CF_DSPMETAFILEPICT Metafile representation of private data CF_DSPTEXT Text representation of private data Sound formats CF_RIFF Resource Interchange File Format CF_WAVE Standard wave file format audio data Special formats CF_DIF Data Interchange Format from Software Arts CF_OWNERDISPLAY Data displayed by the owner of the clipboard data CF_PALETTE Color palette (HPALETTE) CF_PENDATA Microsoft Pen Extensions data CF_PRIVATEFIRST through Private data
CF_PRIVATELAST CF_SYLK

Microsoft Symbolic Link Format Windows 95only formats through Application-defined GDI objects List of files (HDROP) Locale information for CF_TEXT data

CF_GDIOBJFIRST CF_GDIOBJLAST CF_HDROP CF_LOCALE

10
THE WINDOWS CLIPBOARD

Under certain circumstances, Windows is capable of synthesizing data in formats not explicitly provided by an application. For example, if the application provides data in CF_TEXT format, Windows can render that data in the CF_OEMTEXT format at the request of another application. Windows can perform this conversion of formats between the text formats CF_TEXT,

196

Under the Hood: Windows and the Win32 API PART II


CF_OEMTEXT,

and (under Windows NT)

CF_UNICODETEXT;

the bitmap formats

CF_BITMAP

and

CF_DIB; and the metafile formats CF_ENHMETAFILE and CF_METAFILEPICT. Finally, Windows can

also synthesize a CF_PALETTE format from the CF_DIB format.

Registered Formats
Applications that need to place data on the clipboard in a format other than any of the standard formats can register a new clipboard format using the RegisterClipboardFormat function. For example, an application that wishes to place RTF text on the clipboard may make the following call to register this format:
cfRTF = RegisterClipboardFormat(Rich Text Format);

If several applications call RegisterClipboardFormat with the same format name, the format is only registered once. There are many clipboard formats registered by Windows. For example, some registered formats are related to OLE, some others to the Windows 95 shell. The name of a registered format can be obtained by calling the GetClipboardFormatName function.

Private Formats
Sometimes it is not necessary for an application to register a new clipboard format. This is the case when the clipboard is used, for example, to transfer data internally within the application and the data is not expected to be used by other applications. For such application-defined private formats, an application can use the CF_PRIVATEFIRST through CF_PRIVATELAST range of values. In order to enable clipboard viewers to display data stored in a private format, the clipboard owner must provide data in any of the display formats CF_DSPBITMAP , CF_DSPTEXT , CF_DSPMETAFILEPICT, or CF_DSPENHMETAFILE. These formats are identical to their standard counterparts (CF_BITMAP, CF_TEXT, CF_METAFILEPICT, and CF_ENHMETAFILE) except that they are used solely for display purposes and not for pasting.

Clipboard Operations
In order to utilize the clipboard, an application has to perform a variety of operations. These include setting up the clipboard data, obtaining ownership of the clipboard, transferring the data, and responding to clipboard-related events. The application should also provide, as part of its user interface, clipboard-specific user commands (such as commands under its Edit menu).

Transferring Data to the Clipboard


Before order data can be transferred to the clipboard, an application has to do two things. First, the data object must be allocated; second, ownership of the clipboard must be obtained.

The Windows Clipboard CHAPTER 10

197

The data object must be a handle. This handle can refer to a block of memory allocated using GlobalAlloc with the GMEM_MOVEABLE and GMEM_DDESHARE flags (note that the presence of the GMEM_DDESHARE flag does not indicate that the block of memory is shared between applications). It can also be a handle to a GDI object such as a bitmap. It is important to note that once the handle is passed to the clipboard, the application transfers the objects ownership; it should no longer lock the object and should definitely not make an attempt to delete it.
OpenClipboard

The application obtains ownership of the clipboard by opening the clipboard using and then emptying the clipboard by calling the EmptyClipboard function. All handles to data previously transferred to the clipboard will be freed. Next, the application transfers data to the clipboard using SetClipboardData and closes it by calling CloseClipboard.

The application can call SetClipboardData multiple times if data is available in several formats. For example, an application may call SetClipboardData using the CF_DIB and CF_ENHMETAFILE formats to provide a graphics image in both bitmap and metafile forms.

Delayed Rendering
Delayed rendering is a performance-enhancing technique that is most useful for applications that routinely place large blocks of data on the clipboard. An application can specify delayed rendering by passing NULL as the second parameter of SetClipboardData. The system informs the application that data in a specific format must be rendered by sending the application a WM_RENDERFORMAT message. In response to this message, the application must call SetClipboardData and place the requested data on the clipboard. An application that placed data on the clipboard using delayed rendering may also receive a
WM_RENDERALLFORMATS message. This message is sent to the clipboard owner before it is destroyed

to ensure that the data on the clipboard remains available to other applications. When processing a WM_RENDERFORMAT or WM_RENDERALLFORMATS message, the application must not open the clipboard before calling SetClipboardData or close it afterwards.

Pasting Data from the Clipboard


An application can use the IsClipboardFormatAvailable function to determine if data in a specific format is available on the clipboard. If it wishes to obtain a copy of the data on the clipboard, the application can call OpenClipboard, followed by a call to GetClipboardData. The handle obtained by calling GetClipboardData shall not be assumed to remain persistent; applications should immediately copy any data associated with that handle, preferably before calling CloseClipboard. After the call to CloseClipboard, it is possible for other applications to empty the clipboard, rendering the handle obtained through GetClipboardData useless.

10
THE WINDOWS CLIPBOARD

198

Under the Hood: Windows and the Win32 API PART II

The IsClipboardFormatAvailable function can also be used to update the applications Edit menu items. For example, if IsClipboardFormatAvailable indicates that no clipboard data is available in a format that the application understands, the application should disable its Paste command. Applications can also obtain information about the data formats available in the clipboard by calling CountClipboardFormats or EnumClipboardFormats.

Controls and the Clipboard


Edit controls have built-in clipboard support (also supported by the edit control in combo boxes). Edit controls respond to a series of messages by performing clipboard operations. When receiving a WM_COPY message, edit controls copy the current selection to the clipboard using the CF_TEXT format. When receiving a WM_CUT message, edit controls transfer the current selection to the clipboard using the CF_TEXT format and erase the selection from the control. In response to a WM_PASTE message, edit controls take the clipboard contents (if anything is available in the CF_TEXT format) and use it to replace the current selection. Finally, edit controls also process the WM_CLEAR message (erasing the current selection).

Clipboard Messages
There are several Windows messages associated with the clipboard. Applications that use delayed rendering must process the WM_RENDERALLFORMATS messages.
WM_RENDERFORMAT

and

The WM_DESTROYCLIPBOARD message is sent to the clipboard owner when the contents of the clipboard are destroyed. In response to this message, an application may free up any resources associated with rendering or drawing clipboard items. A series of messages is sent to applications that place data on the clipboard using the C F _ O W N E R D I S P L A Y format. These include W M _ A S K C B F O R M A T N A M E , W M _ D R A W C L I P B O A R D , WM_HSCROLLCLIPBOARD, WM_VSCROLLCLIPBOARD, and WM_PAINTCLIPBOARD. Another set of messages is sent to or used by clipboard viewer applications.

Clipboard Viewers
A clipboard viewer is an application that displays the current contents of the clipboard. An example of a clipboard viewer is the Windows Clipboard Viewer application, CLIPBRD.EXE. The IDataObject viewer (DOBJVIEW.EXE) that comes with Visual C++ can also act as a clipboard viewer. A clipboard viewer merely exists for the users convenience and does not disrupt or alter clipboard operations.

The Windows Clipboard CHAPTER 10

199

There can be several clipboard viewers in operation. An application inserts a window in the chain of clipboard viewers by calling the SetClipboardViewer function with the handle of the window. Once added to the chain, the clipboard viewer will receive WM_CHANGECBCHAIN and WM_DRAWCLIPBOARD messages. The clipboard viewer can remove itself from the chain by calling the ChangeClipboardChain function.

A Simple Implementation
The program shown in Listing 10.1 (yes, yet another Hello, World program) puts all this nice theory into practice. This very simple application provides an implementation for the four basic clipboard commands: Cut, Copy, Paste, and Delete. The programs resource file is shown in Listing 10.2, and its header file in Listing 10.3. To compile this application from the command line, you need to enter the following two commands in a DOS window (still not complex enough to warrant a make file):
rc hellocf.rc cl hellocf.c hellocf.res user32.lib gdi32.lib

Listing 10.1. A simple clipboard-aware application.


#include <windows.h> #include hellocf.h HINSTANCE hInstance; char *pszData; void DrawHello(HWND hwnd) { HDC hDC; PAINTSTRUCT paintStruct; RECT clientRect; if (pszData != NULL) { hDC = BeginPaint(hwnd, &paintStruct); if (hDC != NULL) { GetClientRect(hwnd, &clientRect); DPtoLP(hDC, (LPPOINT)&clientRect, 2); DrawText(hDC, pszData, -1, &clientRect, DT_CENTER | DT_VCENTER | DT_SINGLELINE); EndPaint(hwnd, &paintStruct); } } } void CopyData(HWND hwnd) { HGLOBAL hData; LPVOID pData; OpenClipboard(hwnd); EmptyClipboard(); hData = GlobalAlloc(GMEM_DDESHARE | GMEM_MOVEABLE, strlen(pszData) + 1);

10
THE WINDOWS CLIPBOARD

continues

200

Under the Hood: Windows and the Win32 API PART II

Listing 10.1. continued


pData = GlobalLock(hData); strcpy((LPSTR)pData, pszData); GlobalUnlock(hData); SetClipboardData(CF_TEXT, hData); CloseClipboard(); } void DeleteData(HWND hwnd) { free(pszData); pszData = NULL; InvalidateRect(hwnd, NULL, TRUE); } void PasteData(HWND hwnd) { HANDLE hData; LPVOID pData; if (!IsClipboardFormatAvailable(CF_TEXT)) return; OpenClipboard(hwnd); hData = GetClipboardData(CF_TEXT); pData = GlobalLock(hData); if (pszData) DeleteData(hwnd); pszData = malloc(strlen(pData) + 1); strcpy(pszData, (LPSTR)pData); GlobalUnlock(hData); CloseClipboard(); InvalidateRect(hwnd, NULL, TRUE); } void SetMenus(HWND hwnd) { EnableMenuItem(GetMenu(hwnd), ID_EDIT_CUT, pszData ? MF_ENABLED : MF_GRAYED); EnableMenuItem(GetMenu(hwnd), ID_EDIT_COPY, pszData ? MF_ENABLED : MF_GRAYED); EnableMenuItem(GetMenu(hwnd), ID_EDIT_PASTE, IsClipboardFormatAvailable(CF_TEXT) ? MF_ENABLED : MF_GRAYED); EnableMenuItem(GetMenu(hwnd), ID_EDIT_DELETE, pszData ? MF_ENABLED : MF_GRAYED); } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_COMMAND: switch (LOWORD(wParam)) { case ID_FILE_EXIT: DestroyWindow(hwnd); break; case ID_EDIT_CUT: CopyData(hwnd); DeleteData(hwnd); break; case ID_EDIT_COPY: CopyData(hwnd); break;

The Windows Clipboard CHAPTER 10


case ID_EDIT_PASTE: PasteData(hwnd); break; case ID_EDIT_DELETE: DeleteData(hwnd); break; } break; case WM_PAINT: DrawHello(hwnd); break; case WM_DESTROY: PostQuitMessage(0); break; case WM_INITMENUPOPUP: if (LOWORD(lParam) == 1) { SetMenus(hwnd); break; } default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hThisInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { HWND hwnd; MSG msg; WNDCLASS wndClass; HANDLE hAccTbl; pszData = malloc(14); strcpy(pszData, Hello, World!); hInstance = hThisInstance; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszMenuName = HelloMenu; wndClass.lpszClassName = Hello; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(Hello, Hello, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); hAccTbl = LoadAccelerators(hInstance, HelloMenu); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0))

201

10
THE WINDOWS CLIPBOARD

continues

202

Under the Hood: Windows and the Win32 API PART II

Listing 10.1. continued


{ if (!TranslateAccelerator(hwnd, hAccTbl, &msg)) { TranslateMessage(&msg); DispatchMessage(&msg); } } return msg.wParam; }

Listing 10.2. Resource file for the sample clipboard-aware application.


#include windows.h #include hellocf.h HelloMenu MENU BEGIN POPUP &File BEGIN MENUITEM E&xit, ID_FILE_EXIT END POPUP &Edit BEGIN MENUITEM Cu&t\tCtrl+X, ID_EDIT_CUT, GRAYED MENUITEM &Copy\tCtrl+C, ID_EDIT_COPY, GRAYED MENUITEM &Paste\tCtrl+V, ID_EDIT_PASTE, GRAYED MENUITEM &Delete\tDel, ID_EDIT_DELETE, GRAYED END END HelloMenu ACCELERATORS BEGIN X, ID_EDIT_CUT, VIRTKEY, CONTROL C, ID_EDIT_COPY, VIRTKEY, CONTROL V, ID_EDIT_PASTE, VIRTKEY, CONTROL VK_DELETE, ID_EDIT_DELETE, VIRTKEY END

Listing 10.3. Header file for the sample clipboard-aware application.


#define #define #define #define #define ID_FILE_EXIT ID_EDIT_CUT ID_EDIT_COPY ID_EDIT_PASTE ID_EDIT_DELETE 1000 1001 1002 1003 1004

To see how this application works, try using its clipboard functions. You can use the Cut or Copy functions to copy the text it displays to the clipboard; you can also use another application (for example, the Windows Notepad) to create a block of text, copy it to the clipboard, and then paste it into this application.

The Windows Clipboard CHAPTER 10

203

This program has a simple data object; a pointer that is initially set to point to the character string Hello, World! The application also has a simple set of menus containing an Edit menu with the clipboard functions Cut, Copy, Paste, and Delete. Clipboard operations are performed in response to the user selecting these Edit menu commands. The function CopyData copies the current string to the clipboard by first gaining ownership of it through EmptyClipboard and then performing a SetClipboardData call. The function PasteData copies data from the clipboard; it does so by first freeing any current data, and then obtaining clipboard data by calling GetClipboardData. The function SetMenus updates the enabled state of menu items in the Edit menu based on the availability of data in the CF_TEXT format on the clipboard. If no such data is available, the Paste command is disabled. The state of the Cut, Copy, and Delete menu items is also updated to reflect whether the application has any data that can be placed on the clipboard.
DeleteData

To ensure that the applications window is properly updated when the data changes, both the and the PasteData functions call InvalidateRect.

Note that the data handle allocated in CopyData is never freed by the application. After this handle has been passed to the clipboard, freeing it (when the clipboard is emptied) is no longer the responsibility of the application. Similarly, the application never frees the handle obtained through GetClipboardData (in PasteData); after the clipboard data has been obtained and the clipboard is closed, this handle is simply discarded.

Summary
The Windows clipboard represents one of the oldest mechanisms for transferring data between applications. The clipboard is a Windows facility where applications can place data in a variety of formats, to be retrieved later by other applications. Windows defines several standard clipboard formats. Applications can also register additional clipboard formats or use private clipboard formats. An application transfers data to the clipboard by first gaining ownership of it by calling the function EmptyClipboard. It may transfer data immediately or choose to use delayed rendering by passing a NULL handle to the SetClipboardData function. When using delayed rendering, applications must process the WM_RENDERFORMAT and WM_RENDERALLFORMATS messages. Edit controls (and the edit control parts of combo boxes) have built-in clipboard support. They respond to the WM_CUT, WM_COPY, WM_PASTE, and WM_CLEAR messages by performing a cut, copy, paste, or delete operation using data in the CF_TEXT format. Clipboard viewers are programs that show the current contents of the clipboard. These programs merely serve the users convenience and do not alter the clipboards contents or affect clipboard operations.

10
THE WINDOWS CLIPBOARD

204

Under the Hood: Windows and the Win32 API PART II

The Registry CHAPTER 11

205

The Registry
IN THIS CHAPTER

11
THE REGISTRY

11

s Registry Structure 206 s Manually Editing the Registry 208 s Commonly Used Registry Keys 209 s Application Programs and the Registry 212

206

Under the Hood: Windows and the Win32 API PART II

Ah, the Registry. This mysterious object appeared with the introduction of OLE under Windows 3.1. No matter how hard we programmer types were trying to ignore it, it is here to stay; in fact, while we were looking the other way it quietly took over the role of initialization, or INI files, among other things. But what is it? Perhaps more importantly, what should you, the Win32 programmer, know about it? How can you access and manipulate it from within your applications? These are the questions that I attempt to answer in this chapter.

Registry Structure
The Registry is a hierarchically organized store of information. Each entry in this tree-like information structure is called a key. A key may contain any number of subkeys; it can also contain data entries called values. In this form, the Registry stores information about the system, its configuration, hardware devices, and software applications. It also assumes the role of the ubiquitous INI files by providing a place where application-specific settings can be stored. A Registry key is identified by its name. Keynames consist of printable ASCII characters except the backslash (\), space, and wildcard (* or ?) characters. The use of keynames that begin with a period (.) is reserved. Keynames are not case sensitive.

Registry Values
A value in the Registry is identified by its name. Value names consist of the same characters as keynames. The value itself can be a string, binary data, or a 32-bit unsigned value. There are some apparent differences between the behavior of the Windows 95/98 and the Windows NT Registries. The Windows 95/98 Registry appears to allow the assignment of a value to a Registry key (as opposed to a value name); this appears as the default value for that key. Upon closer examination, however, one finds that this is a superficial difference. The default value for a key is really a value with an empty name; empty names are also permitted in the Windows NT Registry. Perhaps the only difference is that the value with the empty name appears to be always defined for a key in the Windows 95/98 Registry, while it must be explicitly created in the Windows NT Registry. In the end, even this difference turns out to be a difference between the two Registry editors, regedit.exe and regedt32.exe. Another apparent difference between the two Registries is the existence of a variety of string types in the Windows NT Registry, while Windows 95/98 appears to support only one string type. But is this really the case? Jumping a bit ahead of myself here, let me show you some of the output created by the Registry reader program that is examined later in this chapter. For example, look at the following output:
Enter key: HKEY_CURRENT_USER\Environment\include Expandable string: f:\msvc20\include;f:\msvc20\mfc\include

However, if you examine the same value using the Windows 95 Registry Editor, it will appear as a binary value. But this is a shortcoming of the Registry Editor, not the Registry itself.

The Registry CHAPTER 11

207

Table 11.1 contains a list of all value types which can go into the Windows 95/98 and Windows NT Registries.

11
THE REGISTRY

Table 11.1. Registry value types. Symbolic identifier


REG_BINARY REG_DWORD REG_DWORD_LITTLE_ENDIAN REG_DWORD_BIG_ENDIAN REG_EXPAND_SZ REG_LINK REG_MULTI_SZ REG_NONE REG_RESOURCE_LIST REG_SZ

Description Raw binary data Double word in machine format (low-endian on Intel) Double word in little-endian format Double word in big-endian format String with unexpanded environment variables Unicode symbolic link Multiple strings ended by two null characters Undefined type Device-driver resource list Null-terminated string

Registry Capacity
Generally, storing items larger than a kilobyte or two in the Registry is not recommended. For larger items, use a separate file, and use the Registry for storing the filename. Under Windows 95/98, Registry values are limited to 64KB. Another consideration when using the Registry is that storing a key generally requires substantially more storage space than storing a value. Whenever possible, organize values under a common key rather than using several keys for the same purpose.

Predefined Registry Keys


The Registry contains several predefined keys. The HKEY_LOCAL_MACHINE key contains entries that describe the computer and its configuration. This includes information about the processor, system board, memory, and installed hardware and software. The HKEY_CLASSES_ROOT key is the root key for information relating to document types and OLE/COM. This key is a subordinate key to HKEY_LOCAL_MACHINE. (It is equivalent to HKEY_LOCAL_MACHINE\SOFTWARE\Classes.) Information that is stored here is used by shell applications such as the Program Manager, File Manager, or the Explorer, and by OLE/ActiveX applications.

208

Under the Hood: Windows and the Win32 API PART II

The HKEY_USERS key serves as the root key for the default user preference settings as well as individual user preferences. The HKEY_CLASSES_USER key is the root key for information relating to the preferences of the current (logged in) user. Under Windows 95/98, there are two additional predefined keys. The HKEY_CURRENT_CONFIG key contains information about the current system configuration settings. This key is equivalent to a subkey (such as 0001) of the key HKEY_LOCAL_MACHINE\Config. The HKEY_DYN_DATA key provides access to dynamic status information, such as information about plug and play devices. The predefined keys and their relationships are illustrated in Figure 11.1.

FIGURE 11.1.
Predefined Registry keys.

HKEY_LOCAL_MACHINE Config HARDWARE SOFTWARE Classes ...

HKEY_CURRENT_CONFIG Display Fonts Settings System CurrentControlSet

Win95

HKEY_CLASSES_ROOT * .386 .doc .txt ... HKEY_USERS .Default ...

HKEY_CURRENT_USER Control Panel Environment Network Software ...

HKEY_DYN_DATA Config Manager PerfStats

Win95

Manually Editing the Registry


The Registry can be manually edited using the Registry Editor. This program is named regedt32.exe under Windows NT and regedit.exe under Windows 95/98. The Windows

The Registry CHAPTER 11

209

NT program regedit.exe is a version of the Windows 95/98 Registry editor that offers a better search capability, but has limitations when it comes to handling various value types. Figure 11.2 shows the Windows 95 Registry Editor in operation.

11
THE REGISTRY

FIGURE 11.2.
The Windows 95 Registry Editor.

Needless to say, using the Registry Editor is a last resort solution. Programmers may need to frequently access the Registry this way (for example, to remove keys that have been placed there by misbehaving applications that are under development). However, end users should never be required to manually change Registry settings. Many Registry settings are controlled implicitly through configuration applications such as the Control Panel. Other Registry settings are created during application installation. OLE applications that have been created using AppWizard update their Registry settings every time they run.

Commonly Used Registry Keys


Information about Registry keys is often difficult to find. For this reason, I decided to collect here information on some frequently used Registry keys that are of interest to the programmer.

Subtrees in HKEY_LOCAL_MACHINE
Keys in HKEY_LOCAL_MACHINE contain information about the computers software and hardware configuration. Of these, the Config and Enum subkeys are specific to plug-and-play capabilities. The Config subkey is where Windows stores various hardware configurations; the Enum subkey contains bus enumerators that build the tree of hardware devices.

210

Under the Hood: Windows and the Win32 API PART II

Both Windows 95/98 and Windows NT maintain the System subkey under HKEY_LOCAL_MACHINE. The System\CurrentControlSet subkey contains configuration information for services and device drivers. Other subkeys in HKEY_LOCAL_MACHINE include Software and Classes. The Software subkey is where information about installed software packages can be found. The Classes subkey is where HKEY_CLASSES_ROOT points to. The Software subtree is of particular interest to application programmers. This is where you should store configuration and installation information specific to your application. Microsoft recommends that you build a series of subtrees under HKEY_LOCAL_MACHINE\Software. These subkeys should represent your company name, the name of your product, and the products version number:
HKEY_LOCAL_MACHINE\Software\CompanyName\ProductName\1.0

For example, configuration information pertaining to the version of Microsoft FrontPage that is installed on my computer can be found under the following key:
HKEY_LOCAL_MACHINE\Software\Microsoft\Microsoft FrontPage\3.0

What you store under such a key is entirely application-dependent. Note that you should not store anything here that is user-specific; user-specific information pertinent to your application should be organized under a subkey of HKEY_CURRENT_USER. Of particular interest is the key
HKEY_LOCAL_MACHINE\Software\Microsoft\Windows\CurrentVersion

which describes the current Windows configuration. Another important key is the key
HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion

which actually has a curious, unexpected role under Windows 95/98. I presume the reason is to maintain compatibility with 32-bit debuggers originally written for Windows NT, but debugger information stored under the key
HKEY_LOCAL_MACHINE\Software\Microsoft\Windows NT\CurrentVersion\Aedebug

will affect debugger behavior under Windows 95/98.

Subtrees in HKEY_CLASSES_ROOT
The HKEY_CLASSES_ROOT key contains two types of subkeys: subkeys that correspond to filename extensions and class definition subkeys.

The Registry CHAPTER 11

211

A filename extension subkey has a name that corresponds to the filename extension (such as .doc). The key typically contains one unnamed value, which holds the name of the class definition subkey. A class definition subkey describes the behavior of a document class. The information stored here includes data on shell and OLE-related properties. A subkey under HKEY_CLASSES_ROOT is CLSID. This is the place where COM class identifiers are stored. When you create an MFC application using the Visual C++ AppWizard, a series of subkeys that are to be installed under HKEY_CLASSES_ROOT are also created. These identify the document type and filename extension of your new application and also its OLE properties such as the OLE class identifier. For example, creating an MFC application named Test with a file extension .tst for its document files yielded the following Registry entries under HKEY_CLASSES_ROOT:
.TST = Test.Document Test.Document\shell\open\command = TEST.EXE %1 Test.Document\shell\open\ddeexec = [open(%1)] Test.Document\shell\open\ddeexec\application = TEST Test.Document = Test Document Test.Document\protocol\StdFileEditing\server = TEST.EXE Test.Document\protocol\StdFileEditing\verb\0 = &Edit Test.Document\Insertable = Test.Document\CLSID = {FC168A60-F1EA-11CE-87C3-00403321BFAC}

11
THE REGISTRY

The following entries were also created under HKEY_CLASSES_ROOT\CLSID:


{FC168A60-F1EA-11CE-87C3-00403321BFAC} = Test Document {FC168A60-F1EA-11CE-87C3-00403321BFAC}\DefaultIcon = TEST.EXE,1 {FC168A60-F1EA-11CE-87C3-00403321BFAC}\LocalServer32 = TEST.EXE {FC168A60-F1EA-11CE-87C3-00403321BFAC}\ProgId = Test.Document {FC168A60-F1EA-11CE-87C3-00403321BFAC}\MiscStatus = 32 {FC168A60-F1EA-11CE-87C3-00403321BFAC}\AuxUserType\3 = test {FC168A60-F1EA-11CE-87C3-00403321BFAC}\AuxUserType\2 = Test {FC168A60-F1EA-11CE-87C3-00403321BFAC}\Insertable = {FC168A60-F1EA-11CE-87C3-00403321BFAC}\verb\1 = &Open,0,2 {FC168A60-F1EA-11CE-87C3-00403321BFAC}\verb\0 = &Edit,0,2 {FC168A60-F1EA-11CE-87C3-00403321BFAC}\InprocHandler32 = ole32.dll

Subtrees in HKEY_USERS
The key HKEY_USERS contains a subkey named .Default and zero or more subkeys corresponding to users on the system. The .Default subkey corresponds to the default user profile. Other entries correspond to profiles of existing users.

Subtrees in HKEY_CURRENT_USER
The HKEY_CURRENT_USER key corresponds to the profile of the currently logged-in user. This key has several subkeys, some common to both Windows 95/98 and Windows NT, some specific to one or the other.

212

Under the Hood: Windows and the Win32 API PART II

Application configuration information specific to the current user should be stored under the subkey Software. Information should be organized by keys corresponding to company name, product name, and product version number. For example, user settings for Microsoft Excel 5.0 can be found under the key
HKEY_CURRENT_USER\Software\Microsoft\Excel\5.0

The users settings and preferences for Windows, its components, and applets can be found under the key
HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion

and its subkeys.

The Registry and INI Files


In new applications, the Registry should be used instead of INI files. This is obvious; but what about old applications, how would they behave under 32-bit Windows? As it turns out, their behavior is different under Windows NT and Windows 95/98. In order to maintain maximum backward compatibility, Windows 95/98 still maintain INI files, such as a win.ini file or a system.ini file. These files do not exist under Windows NT. Instead, Windows NT maps these files to the Registry. Which files are mapped and which are not is determined by the settings under the key
SOFTWARE\Microsoft\Windows NT\CurrentVersion\IniFileMapping

which contains a subkey for every mapped INI file. Values under such a subkey correspond to sections in the INI file and typically point to other keys in the Registry. The mapping of INI files affects the operation of functions such as ReadProfileString or WriteProfileString. If a mapping exists for the specified INI file, these functions will read from and write to the Registry as opposed to an actual INI file.

Application Programs and the Registry


The Win32 API offers a variety of functions for manipulating the Registry.

Opening a Registry Key


All access to the Registry is performed through handles. In order to access a key in the Registry, applications must use a handle to an existing, open key. There are several predefined key handles that are assumed to be always open. These handles include the following: HKEY_LOCAL_MACHINE, HKEY_CLASSES_ROOT, HKEY_USERS, HKEY_CURRENT_USER.

The Registry CHAPTER 11

213

A Registry key is accessed through the function RegOpenKeyEx. For example, in order to obtain a handle to the Registry key HKEY_LOCAL_MACHINE\Software, one would issue the following call:
RegOpenKeyEx(HKEY_LOCAL_MACHINE, Software, 0, KEY_READ, &hKey);

11
THE REGISTRY

To access a subkey under the key HKEY_LOCAL_MACHINE\Software, you can use the concatenated form as the keyname parameter:
RegOpenKeyEx(HKEY_LOCAL_MACHINE, Software\\Classes, 0, KEY_READ, &hKey);

When an application is finished using a Registry key, it should close the key by calling RegCloseKey.

Querying a Value
A Registry value can be retrieved by calling the RegQueryValueEx function. Before this function can be called, the appropriate key must be opened using RegOpenKeyEx.
RegQueryValueEx offers a mechanism that enables applications to find out the memory require-

ments for storing a value before the value is actually retrieved. If you call this function with a NULL pointer passed as the data buffer pointer, the function will return the requested length of the data buffer without actually copying any data. Thus, it is possible to call RegQueryValueEx twice: first to obtain the length of the buffer, and next to actually copy the data, as in the following example:
RegQueryValueEx(hKey, MyValue, NULL, &dwType, NULL, &dwSize); pData = malloc(dwSize); RegQueryValueEx(hKey, MyValue, NULL, &dwType, pData, &dwSize);

Logical as it may appear, it is not possible to use concatenated keys and value names delimited by a backslash as the value name parameter to RegQueryValueEx. Thus, the following call is an error:
RegQueryValueEx(hKey, MyKey\\Value, NULL, &dwType, pData, &dwSize); // ERROR!

Setting a Value
A value in the Registry can be set using the RegSetValueEx function. Before this function can be used, the appropriate key must be opened with KEY_SET_VALUE access using RegOpenKeyEx.

Creating a New Key


Applications can also create a new subkey in the Registry. The RegCreateKeyEx function creates the new key, opens it, and obtains a key handle. This function can also be used to open existing keys; thus it is ideal in situations when the application wishes to access a key whether it already exists or notduring an installation procedure, for example.

214

Under the Hood: Windows and the Win32 API PART II

Under Windows NT, when creating a new key, the application also assigns security attributes to it. The keys security attributes determine who can access the key for reading and writing. Security information can be obtained about an open key using RegGetKeySecurity and set using RegSetKeySecurity (that is, if the application has the necessary privileges).

Other Registry Functions


There are several other functions that assist in dealing with the Registry efficiently. For example, the RegEnumKeyEx and RegEnumValue functions can be used to enumerate the subkeys and values under a specific Registry key. Registry keys can be deleted using the RegDeleteKey function. Several other functions exist to deal with saving and loading subkeys, connecting to remote Registries, and performing other administrative functions.

A Working Example
To demonstrate the use of the Registry from application programs, I created a simple command-line program to read Registry settings. This program is shown in Listing 11.1. To compile this program from the command line, you must specify the advanced API library: cl readreg.cpp advapi32.lib.

Listing 11.1. A simple Registry reader.


#include <windows.h> #include <iostream.h> #include <iomanip.h> #include <string.h> #define STR_HKEY_LOCAL_MACHINE HKEY_LOCAL_MACHINE #define STR_HKEY_CLASSES_ROOT HKEY_CLASSES_ROOT #define STR_HKEY_USERS HKEY_USERS #define STR_HKEY_CURRENT_USER HKEY_CURRENT_USER #define LEN_HKEY_LOCAL_MACHINE (sizeof(STR_HKEY_LOCAL_MACHINE)-1) #define LEN_HKEY_CLASSES_ROOT (sizeof(STR_HKEY_CLASSES_ROOT)-1) #define LEN_HKEY_USERS (sizeof(STR_HKEY_USERS)-1) #define LEN_HKEY_CURRENT_USER (sizeof(STR_HKEY_CURRENT_USER)-1) #define SWAP_ENDIAN(x) (((x<<24)&0xFF000000)|((x<<8)&0xFF0000)|\ ((x>>8)&0xFF00)|((x>>24)|0xFF)) void printval(unsigned char *pBuffer, DWORD dwType, DWORD dwSize) { switch (dwType) { case REG_BINARY: cout << Binary data:; { for (unsigned int i = 0; i < dwSize; i++) { if (i % 16 == 0) cout << \n; cout.fill(0); cout << hex << setw(2) << (unsigned int)(pBuffer[i]) << ; } }

The Registry CHAPTER 11


cout << \n; break; case REG_DWORD: cout.fill(0); cout << Double word: << hex << setw(8) << *((unsigned int *)pBuffer) << \n; break; case REG_DWORD_BIG_ENDIAN: // Intel specific! cout.fill(0); cout << Big-endian double word: << hex << setw(8) << SWAP_ENDIAN(*((unsigned int *)pBuffer)) << \n; break; case REG_EXPAND_SZ: cout << Expandable string: << pBuffer << \n; break; case REG_LINK: cout << Unicode link.; break; case REG_MULTI_SZ: cout << Multiple strings:\n; { char *pStr; int i; for (i = 0, pStr = (char *)pBuffer; *pStr != \0; i++, pStr += strlen((char *)pStr) + 1) { cout << String << i << : << pStr << \n; } } break; case REG_NONE: cout << Undefined value type.\n; break; case REG_RESOURCE_LIST: cout << Resource list.\n; break; case REG_SZ: cout << String: << pBuffer << \n; break; default: cout << Invalid type code.\n; break; } } void main(void) { char szKey[1000]; char *pKey; HKEY hKey, hSubKey; DWORD dwType; DWORD dwSize; unsigned char *pBuffer; int nKey; while (1) { cout << Enter key: ; cin.getline(szKey, 1000);

215

11
THE REGISTRY

continues

216

Under the Hood: Windows and the Win32 API PART II

Listing 11.1. continued


nKey = strcspn(szKey, \\); hKey = NULL; if (!strncmp(szKey, STR_HKEY_LOCAL_MACHINE, nKey) && nKey == LEN_HKEY_LOCAL_MACHINE) hKey = HKEY_LOCAL_MACHINE; if (!strncmp(szKey, STR_HKEY_CLASSES_ROOT, nKey) && nKey == LEN_HKEY_CLASSES_ROOT) hKey = HKEY_CLASSES_ROOT; if (!strncmp(szKey, STR_HKEY_USERS, nKey) && nKey == LEN_HKEY_USERS) hKey = HKEY_USERS; if (!strncmp(szKey, STR_HKEY_CURRENT_USER, nKey) && nKey == LEN_HKEY_CURRENT_USER) hKey = HKEY_CURRENT_USER; if (hKey == NULL || szKey[nKey] != \\) { cout << Invalid key.\n; continue; } pKey = szKey + nKey + 1; nKey = strcspn(pKey, \\); while (pKey[nKey] == \\) { pKey[nKey] = \0; if (RegOpenKeyEx(hKey, pKey, NULL, KEY_READ,&hSubKey) == ERROR_SUCCESS) { RegCloseKey(hKey); hKey = hSubKey; } else { RegCloseKey(hKey); hKey = NULL; break; } pKey += nKey + 1; nKey = strcspn(pKey, \\); } if (hKey == NULL) { cout << Invalid key.\n; continue; } if (RegQueryValueEx(hKey, pKey, NULL, &dwType, NULL, &dwSize) == ERROR_SUCCESS) { pBuffer = (unsigned char *)malloc(dwSize); if (pBuffer == NULL) { cout << Insufficient memory.\n; break; } if (RegQueryValueEx(hKey, pKey, NULL, &dwType, pBuffer, &dwSize) == ERROR_SUCCESS) printval(pBuffer, dwType, dwSize);

The Registry CHAPTER 11


else cout << Error reading key.\n; free(pBuffer); } else cout << Error reading key.\n; RegCloseKey(hKey); } }

217

11
THE REGISTRY

This program demonstrates many aspects of handling the Registry. Execution begins in the main function, where the program immediately enters a forever loop (terminate the program by hitting Ctrl+C). After displaying a prompt and reading in a Registry key typed by the user, the program first checks for the presence of the name of any of the top-level keys in the string the user typed. If such a keyname is found, the program begins an iteration. The string typed by the user is expected to contain backslash characters as key delimiters. In the iteration part, subsequent strings are extracted from the input string using the strcspn function. The iteration proceeds until the last string is extracted, which is assumed to be a value name. The iteration allows empty (zero-length) names for both keys and values. During the iteration, a key handle is obtained through RegOpenKeyEx for every string extracted from the user input. If obtaining the key handle fails (presumably because the user specified an invalid key), the iteration stops with an error. However, if the iteration succeeds, the value corresponding to the last extracted string is retrieved by calling RegQueryValueEx. This function is in fact called twice: once to determine the amount of memory required to store the data and a second time to actually retrieve the data. The value is printed in the printval function. This function takes a pointer to the value and takes the values type and its length, from which it produces formatted output. Here is some sample output produced by this program (note that all these keys might not be present on your computer):
Enter key: HKEY_CURRENT_USER\Software\Microsoft\Hover!\HighScoreNames Multiple strings: String 0: Viktor String 1: Steve String 2: John String 3: Patrick String 4: Jennifer String 5: Julie String 6: Jon String 7: Tony String 8: Larry String 9: Harley Enter key: HKEY_LOCAL_MACHINE\Enum\Monitor\Default_Monitor\0001\ConfigFlags Binary data: 00 00 00 00 Enter key: HKEY_CURRENT_USER\Environment\include Expandable string: f:\msvc20\include;f:\msvc20\mfc\include

218

Under the Hood: Windows and the Win32 API PART II

You may even find this program useful when trying to determine the exact type of a value in the Windows 95/98 Registry, above and beyond the information provided by the Registry Editor. With some work, the program could be modified to have the capability to actually write to the Registry as well.

Summary
The Registry is a place where Windows and applications can store configuration data. The Registry is a tree-like, hierarchically organized information store. Registry entries, or keys, are identified by a name and may contain any number of subkeys or values. At the top level of the Registry are the root keys HKEY_USERS and HKEY_LOCAL_MACHINE. Other predefined keys include HKEY_CLASSES_ROOT, HKEY_CURRENT_USER, HKEY_CURRENT_CONFIG, and HKEY_DYN_DATA. A Registry value can be a four-byte integer, a string or a series of strings, or arbitrary binary data. Registry values are usually created by application programs, installation procedures, or configuration utilities such as the Control Panel. However, the Registry can also be manually edited using the Registry Editor. Applications typically store configuration information under HKEY_LOCAL_MACHINE\Software, and user-specific data under HKEY_CURRENT_USER\Software. In both cases, subkeys should be created to correspond to a company name, product name, and product version number. Additionally, applications that manage specific document types created a filename extension and a class definition entry under HKEY_CLASSES_ROOT. OLE applications also store information here. The Win32 API provides a series of functions for applications to access the Registry. Using one of the predefined Registry keys, applications can access any subkey for reading and writing, query values, and set values. Applications can also create new keys or delete existing keys.

Exception Handling CHAPTER 12

219

Exception Handling
IN THIS CHAPTER

12

s Exception Handling in C and C++ 220 s Mixing C and C++ Exceptions 228

12
EXCEPTION HANDLING

220

Under the Hood: Windows and the Win32 API PART II

The Win32 API supports structured exception handling. Through this mechanism, applications can handle various hardware- and software-related conditions. Structured exception handling is not to be confused with the concept of exceptions in the C++ language, which is a feature of that language. The Win32 exception handling mechanism is not dependent on its implementation in a specific language. To avoid confusion, I decided to follow the conventions used in Microsoft documentation and use the term C exception to refer to Win32 structured exceptions, and C++ exception to refer to the typed exception handling mechanism of the C++ language.

Exception Handling in C and C++


Microsoft provides a set of extensions to the C language, which enable C programs to handle Win32 structured exceptions. This exception handling mechanism is markedly different from the typed exceptions in the C++ language. This section offers a review of both mechanisms in the context of exceptions in the Win32 environment.

C Exceptions
What is, indeed, an exception? How do exceptions work? In order to understand the exception handling mechanism, first take look at the program shown in Listing 12.1.

Listing 12.1. A program that generates an exception.


void main(void) { int x, y; x = 5; y = 0; x = x / y; }

Needless to say, an integer division by zero is likely to cause a program to terminate abnormally. If you compile the preceding program and run it under Windows 95, it generates the error dialog shown in Figure 12.1.

FIGURE 12.1
Division by zero error.

What exactly happened here? Obviously, when you attempt to divide by zero, the processor will generate an error condition (the actual mechanism is hardware-dependent and not of our concern). This error condition is detected by the operating system, which looks for an exception handler for the specific error condition. As no such handler was detected, the default exception handling mechanism took over, displaying the dialog.

Exception Handling CHAPTER 12

221

Using the C exception handling mechanism, it is possible to catch this exception and handle the divide by zero condition gracefully. Consider the program shown in Listing 12.2.

Listing 12.2. Handling the divide by zero exception.


#include windows.h #include stdio.h void main(void) { int x, y; _ _try { x = 5; y = 0; x = x / y; } _ _except (GetExceptionCode() == EXCEPTION_INT_DIVIDE_BY_ZERO ? EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) { printf(Divide by zero error.\n); } }

12
EXCEPTION HANDLING

Running this program no longer produces the dialog; instead, the message Divide error. is printed and the program terminates gracefully.

by zero

The block of statements following the _ _try instruction is often called a guard block. This block of statements is executed unconditionally. When an exception is raised within the guard block, the expression following the _ _except statement (often called the filter expression) is evaluated. This expression should be an integer expression yielding one of the following values in Table 12.1.

Table 12.1. Filter expression values. Symbolic constant


EXCEPTION_CONTINUE_EXECUTION

Value -1 0 1

Description Continue execution at the location where exception was raised. Pass control to next exception handler. Execute exception handler.

EXCEPTION_CONTINUE_SEARCH EXCEPTION_EXECUTE_HANDLER

If the filter expressions value is -1 (EXCEPTION_CONTINUE_EXECUTION), execution continues at the location where the exception was raised. That is, at the location, not afterwhich means that the offending piece of code may get executed again. Whether it actually does get executed or not depends on the type of the exception. For example, in the case of an integer division by zero, it does; in the case of a floating-point division by zero, it does not. In any case, care should be taken to avoid creating an infinite loop by returning control to the point where the error occurs without eliminating the conditions which caused the exception in the first place.

222

Under the Hood: Windows and the Win32 API PART II

In the other two cases, the first thing that happens is that the guard block goes out of scope. Any function calls that might have been interrupted by the exception are terminated and the stack is unwound. If the filter expression evaluates to 1 (EXCEPTION_EXECUTE_HANDLER), control is transferred to the statement block following the _ _except statement. The third filter value, 0 (EXCEPTION_CONTINUE_SEARCH), hints at the possibility of nested exceptions. Indeed, consider the program shown in Listing 12.3. In this program, two exceptions are generated, one for a floating-point division by zero, one for an integer division by zero. The two exceptions are handled very differently.

Listing 12.3. Nesting exception handlers.


#include <stdio.h> #include <float.h> #include <windows.h> int divzerofilter(unsigned int code, int *j) { printf(Inside divzerofilter\n); if (code == EXCEPTION_INT_DIVIDE_BY_ZERO) { *j = 2; printf(Handling an integer division error.\n); return EXCEPTION_CONTINUE_EXECUTION; } else return EXCEPTION_CONTINUE_SEARCH; } void divzero() { double x, y; int i, j; _ _try { x = 10.0; y = 0.0; i = 10; j = 0; i = i / j; printf(i = %d\n, i); x = x / y; printf(x = %f\n, x); } _ _except (divzerofilter(GetExceptionCode(), &j)) { } } void main(void) { _controlfp(_EM_OVERFLOW, _MCW_EM); _ _try {

Exception Handling CHAPTER 12


divzero(); } _ _except (GetExceptionCode() == EXCEPTION_FLT_DIVIDE_BY_ZERO ? EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) { printf(Floating point divide by zero error.\n); } }

223

When an exception is raised inside the divzero function, the filter expression is evaluated. This results in a call to the divzerofilter function. The function checks whether the exception was an integer division by zero exception; if so, it corrects the value of the divisor (j) and returns the EXCEPTION_CONTINUE_EXECUTION value, which causes the exception handling mechanism to return control to the point where the exception was raised. In the case of any other exceptions, divzerofilter returns EXCEPTION_CONTINUE_SEARCH; this causes the exception handling mechanism to seek another exception handler. This other exception handler has been installed in the main function. This handler handles floating-point division by zero exceptions. Instead of returning to the point at which execution was interrupted, it simply prints an error message. Running this program produces the following output:
Inside divzerofilter Handling an integer division error. i = 5 Inside divzerofilter Floating point divide by zero error.

12
EXCEPTION HANDLING

As you can see, both times an exception is raised, the exception filter installed in the function divzero is activated. However, in the case of the floating-point division, the exception remains unhandled; therefore, the exception is propagated to the next level, and the exception handler is installed in the main function.

NOTE
To handle floating-point exceptions, it was necessary to call the _controlfp function. This function can be used to enable floating-point exceptions. By default, floating-point exceptions on the Intel architecture are disabled; instead, the floating-point library generates IEEEcompatible infinite results.

A discussion of C exception handling would not be complete without a list of some of the commonly occurring C exceptions. These exceptions are shown in Table 12.2.

224

Under the Hood: Windows and the Win32 API PART II

Table 12.2. Filter expression values. Symbolic constant


EXCEPTION_ACCESS_VIOLATION EXCEPTION_PRIV_INSTRUCTION EXCEPTION_STACK_OVERFLOW EXCEPTION_FLT_DIVIDE_BY_ZERO EXCEPTION_FLT_OVERFLOW EXCEPTION_FLT_UNDERFLOW EXCEPTION_INT_DIVIDE_BY_ZERO EXCEPTION_INT_OVERFLOW

Description Reference to invalid memory location Attempt to execute privileged instruction Stack overflow Floating-point division Floating-point result too large Floating-point result too small Integer division Integer result too large

In addition to system-generated exceptions, applications can raise software exceptions using the RaiseException function. Windows reserves exception values with bit 29 set for userdefined software exceptions.

C Termination Handling
Closely related to the handling of C exceptions is the topic of C termination handling. To better understand the problem for which termination handling provides a solution, consider the program shown in Listing 12.4.

Listing 12.4. Resource allocation problem.


#include <stdio.h> #include <windows.h> void badmem() { char *p; printf(allocating p\n); p = malloc(1000); printf(p[1000000] = %c\n, p[1000000]); printf(freeing p\n); free(p); } void main(void) { _ _try { badmem(); } _ _except (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION ? EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) { printf(An access violation has occurred.); } }

Exception Handling CHAPTER 12

225

In this program, the function badmem allocates the p character array. However, its execution is interrupted when it refers to an invalid array element. Because of this, the function never has a chance to free up the allocated array, as demonstrated by its output:
allocating p An access violation has occurred.

This problem can be solved by installing a termination handler in the badmem function, as shown in Listing 12.5.

Listing 12.5. A termination handler.


#include <stdio.h> #include <windows.h> void badmem() { char *p; _ _try { printf(allocating p\n); p = malloc(1000); printf(p[1000000] = %c\n, p[1000000]); } _ _finally { printf(freeing p\n); free(p); } } void main(void) { _ _try { badmem(); } _ _except (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION ? EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) { printf(An access violation has occurred.); } }

12
EXCEPTION HANDLING

Running this program produces the desired result:


allocating p freeing p An access violation has occurred.

As you can see, the instructions in the badmem function are now enclosed in a _ _try block, which is now followed by the _ _finally keyword. The _ _finally keyword is special in that the instruction block that follows it is always executed, no matter under what circumstances the function terminates. So when badmem goes out of scope due to the exception, the instructions in the

226

Under the Hood: Windows and the Win32 API PART II

block are given a chance to clean up any resources that might have been allocated within this function.
_ _finally

C++ Exception Handling


The Win32 exception handling mechanism uses the GetExceptionCode function to determine the nature of the exception. In contrast, C++ exception handling is type-based; the nature of the exception is determined by its type. Most examples that demonstrate C++ exception handling do so in the context of a class declaration. This is not necessary, and in my opinion often hides the simplicity of C++ exception handling. Consider the simple example in Listing 12.6. (When you compile this example or any other program that uses C++ exceptions, do not forget to add the -GX switch to the cl command line.)

Listing 12.6. C++ Exception handling.


#include <iostream.h> int divide(int x, int y) { if (y == 0) throw int(); return x / y; } void main(void) { int x, y; try { x = 5; y = 0; x = divide(x, y); } catch (int) { cout << A division by zero was attempted.\n; } }

In this example, the function divide raises an exception of type int when a division by zero is attempted. This exception is caught by the exception handler in main.

Termination Handling in C++


C++ exceptions can also be used for termination handling. For termination handling, a C++ program can wrap a block of code using a catchall exception handler, and perform resource cleanup before propagating all exceptions to a higher level handler by using throw. Consider the example in Listing 12.7, which is a C++ variant of the program shown in Listing 12.5.

Exception Handling CHAPTER 12

227

Listing 12.7. Termination handling with C++ exceptions.


#include <stdio.h> #include <windows.h> void badmem() { char *p; try { printf(allocating p\n); p = (char *)malloc(1000); printf(p[1000000] = %c\n, p[1000000]); } catch(...) { printf(freeing p\n); free(p); throw; } } void main(void) { try { badmem(); } catch(...) { printf(An exception was raised.); } }

12
EXCEPTION HANDLING

Running this program produces the following output:


allocating p freeing p An exception was raised.

The exception handler in the function badmem plays the role of the __finally block in the C exception handling mechanism. Although these examples demonstrate the power of C++ exception handling with C-style code, the use of classes in exception handling has some obvious advantages. For example, when the exception is thrown, an object of the type of the exception is actually created; thus it is possible to provide additional information about the exception in the form of member variables. Also, appropriate use of constructors and destructors can replace the relatively inelegant resource cleanup mechanism shown in Listing 12.7.

C++ Exception Classes


Visual C++ provides an implementation of the exception class hierarchy as part of the Standard Template Library. This hierarchy consists of the exception class and derived classes

228

Under the Hood: Windows and the Win32 API PART II

representing various conditions, such as runtime errors. The exception class and derived classes are declared in the header file exception.

Mixing C and C++ Exceptions


Although the C compiler does not support C++ exceptions, the C++ compiler supports both C++ exceptions and the Microsoft extensions for C exceptions. Sometimes it is necessary to mix these two in order to use the C++ exception syntax while catching Win32 structured exceptions. There are basically two methods for this: You can use an ellipsis handler, or you can use a translator function.

The Ellipsis Handler


In the termination handling example shown in Listing 12.7, we already made use of the ellipsis handler. This catchall handler, which has the form
catch(...) { }

can be used to catch exceptions of any type, including C exceptions. This offers a simple exception handling mechanism like the one used in Listing 12.7. Unfortunately, the ellipsis handler does not have any information about the actual type of the structured exception. This should be easy, you say. (Well, I certainly said that when I first read about the differences between C and C++ exception handling.) Why not just catch an exception of type unsigned int (after all, the Microsoft Visual C++ documentation states that this is the type of C exceptions) and examine its value? Consider the program in Listing 12.8.

Listing 12.8. Failed attempt to catch C exceptions as C++ exceptions of type unsigned int.
#include <windows.h> #include <iostream.h> void main(void) { int x, y; try { x = 5; y = 0; x = x / y; } catch (unsigned int e) { if (e == EXCEPTION_INT_DIVIDE_BY_ZERO) { cout << Division by zero.\n; } else throw; } }

Exception Handling CHAPTER 12

229

Alas, this elegant solution is no solution at all. C exceptions can only be caught by an ellipsis handler. But not all is lost just yet; could we not simply use the GetExceptionCode function in the C++ catch block and obtain the structured exception type? For example, consider the program in Listing 12.9.

Listing 12.9. C++ exception handlers cannot call GetExceptionCode.


#include <windows.h> #include <iostream.h> void main(void) { int x, y; try { x = 5; y = 0; x = x / y; } catch (...) { // The following line results in a compiler error if (GetExceptionCode() == EXCEPTION_INT_DIVIDE_BY_ZERO) { cout << Division by zero.\n; } else throw; } }

12
EXCEPTION HANDLING

As they say, nice try but no cigar. The function GetExceptionCode is implemented as an intrinsic function and can only be called as part of the filter expression in a C __except statement. It seems that some other mechanism is necessary to differentiate between C exceptions in C++ code. There is yet another possible solution. We could create a C exception handler to catch all C exceptions and throw a C++ exception of type unsigned int with the value of the C exception code. An example program for this is shown in Listing 12.10.

Listing 12.10. Raising C++ exceptions in a C exception filter.


#include <windows.h> #include <iostream.h> int divide(int x, int y) { try { x = x / y; } catch(unsigned int e) {

continues

230

Under the Hood: Windows and the Win32 API PART II

Listing 12.10. continued


cout << Inside C++ exception.\n; if (e == EXCEPTION_INT_DIVIDE_BY_ZERO) { cout << Division by zero.\n; } else throw; } return x; } unsigned int catchall(unsigned int code) { cout << inside catchall: << code << \n; if (code != 0xE06D7363) throw (unsigned int)code; return EXCEPTION_CONTINUE_SEARCH; } void main(void) { int x, y; _ _try { x = 10; y = 0; x = divide(x, y); } _ _except(catchall(GetExceptionCode())) {} }

This approach has but one problem. When the catchall function throws a C++ exception that is not handled by a C++ exception handler, it will be treated as yet another C exception, resulting in another call to catchall. This would go on forever, were it not for the test for the value 0xE06D7363, which appears to be a magic value associated with C++ exceptions. But we are getting into seriously undocumented stuff here; there has to be another solution! At this point, you might ask the obvious question: If C++ programs can use the Microsoft C exception handling mechanism, why go through this dance at all? Why not just use _ _try and _ _except and get it over with? Indeed, this is a valid solution; however, to improve code portability, you may want to use the C++ exception handling mechanism when possible, and localize any dependence on Microsoft extensions as much as possible.

Translating C Exceptions
Fortunately, the Win32 API provides a function that allows a much more elegant solution for translating a C exception into a C++ exception. The name of the function is _set_se_translator. Using this function, you can finally obtain an elegant, satisfactory solution for translating C exceptions to C++ exceptions. An example for this is shown in Listing 12.11.

Exception Handling CHAPTER 12

231

Listing 12.11. Using _set_se_translator to translate C exceptions.


#include <windows.h> #include <iostream.h> #include <eh.h> int divide(int x, int y) { try { x = x / y; } catch(unsigned int e) { cout << Inside C++ exception.\n; if (e == EXCEPTION_INT_DIVIDE_BY_ZERO) { cout << Division by zero.\n; } else throw; } return x; } void se_translator(unsigned int e, _EXCEPTION_POINTERS* p) { throw (unsigned int)(e); } void main(void) { int x, y; _set_se_translator(se_translator); x = 10; y = 0; x = divide(x, y); }

12
EXCEPTION HANDLING

Summary
Win32 programmers using the C++ language must face two separate, only partially compatible exception handling mechanisms. Win32-structured exceptions are often generated by the operating system. These exceptions are not dependent on any language-specific implementation and are used to communicate a condition to the applications exception handler using a 32-bit unsigned value. In contrast, C++ exceptions are typed expressions; the nature of the exception is often derived from the type of the object that is used when the expression is thrown. C programs can use the _ _try and _ _except keywords (which are Microsoft extensions to the C language) to handle structured exceptions. It is possible for exception handlers to be nested. The type of the expression is obtained by calling the GetExceptionCode function in the

232

Under the Hood: Windows and the Win32 API PART II


_ _except

filter expression. Depending on the value of the filter expression, an exception may be handled by the exception handler, execution may continue at the point where the exception occurred, or control can be transferred to the next exception handler. An unhandled exception causes an application error.

C programs can also use termination handlers. These handlers, installed using the _ _try and can ensure that a function which is abnormally terminated by an exception is given a chance to perform cleanup.
_ _finally keywords,

C++ programs use the C++ try and catch keywords to handle exceptions. The type of the exception is declared following the catch keyword. The catch keyword with an ellipsis declaration (...) can be used to catch all exceptions; one possible use of this construct is to act as a termination handler, analogous to the _ _finally block in C exception handling. As C++ programs can also use C exceptions, it is possible to mix the two exception handling mechanisms. C++ programs can catch C exceptions using an ellipsis handler. Unfortunately, this method does not allow C++ programs to obtain the exception code. However, C++ programs can install an exception translator function, which can be used to translate C structured exceptions into C++ typed exceptions.

IN THIS PART
s Exploring an MFC Skeleton Application 235 s Working with Documents and Views 257 s Dialogs and Property Sheets 273

PART
333

III

s MFC Support for Common Dialogs and Common Controls 297 s Using ActiveX Controls 321 s Device Context and GDI Objects s Serialization: File and Archive Objects 351 s Collection Classes 367

s Internet Support Classes 387 s Exceptions, Multithreading, and other MFC Classes 397

Microsoft Foundation Classes

Exploring an MFC Skeleton Application CHAPTER 13

235

Exploring an MFC Skeleton Application

13

IN THIS CHAPTER
s MFC and Applications 236 s Foundation Class Fundamentals 237 s A Simple MFC Application Skeleton 238 s Adding Code to the Application 253

13
EXPLORING AN MFC SKELETON APPLICATION

236

Microsoft Foundation Classes PART III

The Microsoft Foundation Classes (MFC) Library is arguably the most distinguishing component of the Visual C++ development system. This vast collection of C++ classes encapsulates most of the Win32 API and provides a powerful framework for typical (and not-so-typical) applications.

MFC and Applications


A typical MFC application is created using the Visual C++ AppWizard. However, using the AppWizard to create an MFC application is not necessary, nor is the use of MFC restricted to such AppWizard-generated programs. Many simple MFC classes can be used in simple programs including, to the surprise of many programmers, command-line (console) applications. For example, consider the simple program shown in Listing 13.1. This MFC program can be compiled from the command line (cl -MD -D_AFXDLL hellomfc.cpp).

Listing 13.1. A simple MFC console application.


#include <afx.h> CFile& operator<<(CFile& file, const CString& string) { file.Write(string, string.GetLength()); return file; } void main(void) { CFile file((int)GetStdHandle(STD_OUTPUT_HANDLE)); CString string = Hello, MFC!; file << string; }

This example notwithstanding, the primary goal of the MFC is to provide an encapsulation for the Windows API. Its major classes, such as CWnd, CDialog, or CGdiObject, represent the results of this design philosophy. Ideally, an MFC application never has to call Windows API functions directly; instead, it constructs an object of the appropriate type and utilizes the objects member functions. The objects constructor and destructor take care of any necessary initialization and cleanup, including creating and destroying Windows objects. For example, an application that must draw into a window can do so by constructing a CClientDC object and calling the objects member functions (which closely map GDI drawing functions). The CClientDC constructor makes the appropriate calls to create a device context, set up the mapping mode, and perform other initializations. When the object goes out of scope or is destroyed using the delete operator, the destructor automatically releases the device context. This kind of encapsulation would make writing application programs easier even without the benefit of the Developer Studio, its AppWizard, and other powerful features.

Exploring an MFC Skeleton Application CHAPTER 13

237

Foundation Class Fundamentals


The classes in MFC are loosely organized into several major categories. Of these, the two major categories are Application Architecture classes and Window Support classes. Other categories contain classes that encapsulate a variety of system, GDI, or miscellaneous services, including Internet support. Most classes in MFC, with the exception of certain support classes, are derived from a common root, the CObject class. The major MFC categories are illustrated in Figure 13.1.

FIGURE 13.1.
Overview of MFC.

CObject

Application Architecture Application and Threads Document Templates

Window support

Graphics Support

Frame Windows

Device Contexts

13
View Windows GDI Objects

EXPLORING AN MFC SKELETON APPLICATION

Document Classes

Dialogs System Support

Document Items

Controls Exceptions and Synchronization

OLE Classes Files

Other (ODBC / DAO, Winsock, etc.) Non-CObject Derived Run-time Model and Serialization Simple Value Types and Structures OLE Wrappers and OLE Automation Collection Templates Internet Server Support Other Support Classes Collections

238

Microsoft Foundation Classes PART III

The CObject class implements two important features: serialization and runtime type information. (Note that the CObject class predates, and therefore does not use RTTI, the new C++ runtime type information mechanism.) Serialization is the conversion of an object to and from a persistent form. Or, in simpler terms, it means writing an object to disk or reading it from the disk or any other forms of persistent storage. The MFC Library uses CArchive objects for serialization. Serialization is also used to place an object on the Clipboard or prepare the object for OLE embedding. All classes that represent windows are derived from the root window class, CWnd, itself derived from CObject. These classes provide a very high-level encapsulation of window behavior and specific window types. For example, the sample program in Listing 13.2 demonstrates the use of the CColorDialog class that represents the color picker common dialog. As this example demonstrates, the dialog can be displayed by simply creating a CColorDialog object and calling its DoModal member function. This program can also be compiled from the command line, using cl -MD -D_AFXDLL colors.cpp.

Listing 13.2. Using an MFC dialog class in a non-MFC application.


#include <afx.h> #include <afxdlgs.h> int WINAPI WinMain(HINSTANCE d1, HINSTANCE d2, LPSTR d3, int d4) { CColorDialog dlg; dlg.DoModal(); return 0; }

Two of the most important key MFC classes are CView and CDocument. These classes together encapsulate a document (any collection of data with which an application works) and its visual presentation. Most MFC applications have at least one CDocument-derived and one CViewderived class, providing a logical and visual representation of the data they are designed to manipulate.

A Simple MFC Application Skeleton


What is a typical MFC framework application like? How does it utilize the document and view classes? How do you create and build such an application? These are the questions that I attempt to answer in this chapter. Have you not guessed it yet? We are going to build YAHWA! No, I am not swearing in Yiddish; it is short for Yet Another Hello World Application. But this time, it is going to be a framework application built using the MFC AppWizard. We will use this application to experiment with MFC features and to explore the relationships between the applications various classes.

Exploring an MFC Skeleton Application CHAPTER 13

239

Creating the YAH Project


I would have liked to name my project YAHWA but with a five-character project name, AppWizard would have generated filenames that are longer than eight characters. Alas, such filenames would have been mangled when put on an ISO9660 CD-ROM. Instead of getting into that mess, I figured it is easier to just use a shorter name. The YAH project is created through AppWizard. From the Developer Studio, select the New command under the File menu; select the Projects tab in the New dialog; and select MFC AppWizard (EXE). Type the name of the project (YAH) and select a directory where the new project would be placed. Click on the OK button. YAH should be a single document interface (SDI) project; set this option in the first AppWizard dialog step that appears. Most other default settings should be accepted, except for a few settings that can be accessed by clicking the Advanced button in AppWizard step 4. In this advanced dialog, enter YAH as the file extension and change the main frame caption to Hello, World! (or whatever you find suitable). See Figure 13.2.

FIGURE 13.2.
AppWizard advanced options for the YAH project.

13
EXPLORING AN MFC SKELETON APPLICATION

After these changes, you can let AppWizard create the project. After the project has been created, the Developer Studio opens the project and displays the project workspace in ClassView. (See Figure 13.3.)

Exploring the Application Object


As you can see, AppWizard created five classes for the YAH project. Take a look at CYAHApp. This class is derived from CWinApp and represents the application itself.

240

Microsoft Foundation Classes PART III

FIGURE 13.3.
YAH classes.

The CYAHApp class is declared in YAH.h; this file can be opened either by double-clicking on the CYAHApp class in ClassView or double-clicking the filename in FileView. A look at the declaration (see Listing 13.3) reveals three member functions: a constructor, an override of the virtual function InitInstance, and a member function CAppAbout. How do these functions relate to a typical WinMain function in a non-MFC application?

Listing 13.3. CYAHApp class declaration.


class CYAHApp : public CWinApp { public: CYAHApp(); // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CYAHApp) public: virtual BOOL InitInstance(); //}}AFX_VIRTUAL // Implementation //{{AFX_MSG(CYAHApp) afx_msg void OnAppAbout(); // NOTE - the ClassWizard will add and remove member // functions here. DO NOT EDIT what you see in these blocks // of generated code ! //}}AFX_MSG DECLARE_MESSAGE_MAP() };

A look at the implementation of AfxWinMain in the MFC source file winmain.cpp reveals the answer. The major initialization steps performed here are shown in Figure 13.4. How can the application object be constructed before AfxWinMain is executed? Simple; in YAH.cpp, a global object of type CYAHApp named theApp is declared. By the time execution of AfxWinMain begins, this object will have been constructed (which implies that its constructor will have been called).

Exploring an MFC Skeleton Application CHAPTER 13

241

The member functions InitApplication and InitInstance can be overridden. They correspond to one-time and instance-specific initializations. These functions are called explicitly by AfxWinMain before the message loop is entered.

FIGURE 13.4.
Major initialization steps.
Construct application object

AfxWinMain

Initialize MFC Library

Call InitApplication member function of application object

13
Call InitInstance member function of application object

EXPLORING AN MFC SKELETON APPLICATION

Call Run member function of application object

Look at the file YAH.cpp, the implementation file for the CYAHApp class (if you want to open this file from ClassView, you may have to expand the CYAHApp class and double-click on one of the member functions). Actually, to be precise, look at the first half of this file; the second half contains a declaration and the implementation of the CAboutDlg class, which is the skeleton applications About dialog; this does not concern us at the moment. The relevant parts of YAH.cpp are shown in Listing 13.4.

Listing 13.4. CYAHApp class implementation.


/////////////////////////////////////////////////////////////////// // CYAHApp BEGIN_MESSAGE_MAP(CYAHApp, CWinApp) //{{AFX_MSG_MAP(CYAHApp) ON_COMMAND(ID_APP_ABOUT, OnAppAbout) // NOTE - the ClassWizard will add and remove mapping macros here. // DO NOT EDIT what you see in these blocks of generated code!

continues

242

Microsoft Foundation Classes PART III

Listing 13.4. continued


//}}AFX_MSG_MAP // Standard file based document commands ON_COMMAND(ID_FILE_NEW, CWinApp::OnFileNew) ON_COMMAND(ID_FILE_OPEN, CWinApp::OnFileOpen) // Standard print setup command ON_COMMAND(ID_FILE_PRINT_SETUP, CWinApp::OnFilePrintSetup) END_MESSAGE_MAP() /////////////////////////////////////////////////////////////////// // CYAHApp construction CYAHApp::CYAHApp() { // TODO: add construction code here, // Place all significant initialization in InitInstance } /////////////////////////////////////////////////////////////////// // The one and only CYAHApp object CYAHApp theApp; /////////////////////////////////////////////////////////////////// // CYAHApp initialization BOOL CYAHApp::InitInstance() { AfxEnableControlContainer(); // Standard initialization // If you are not using these features and wish to reduce the size // of your final executable, you should remove from the following // the specific initialization routines you do not need. #ifdef _AFXDLL Enable3dControls(); // Call this when using MFC in a shared DLL #else Enable3dControlsStatic(); // Call this when linking statically #endif // Change the registry key under which our settings are stored. // You should modify this string to be something appropriate // such as the name of your company or organization. SetRegistryKey(_T(Local AppWizard-Generated Applications)); LoadStdProfileSettings(); // Load standard INI file options // Register the applications document templates. Document // templates serve as the connection between documents, frame // windows and views. CSingleDocTemplate* pDocTemplate; pDocTemplate = new CSingleDocTemplate( IDR_MAINFRAME, RUNTIME_CLASS(CYAHDoc), RUNTIME_CLASS(CMainFrame), // main SDI frame window RUNTIME_CLASS(CYAHView)); AddDocTemplate(pDocTemplate); // Enable DDE Execute open EnableShellOpen(); RegisterShellFileTypes(TRUE); // Parse command line for standard shell commands, DDE, file open CCommandLineInfo cmdInfo; ParseCommandLine(cmdInfo); // Dispatch commands specified on the command line if (!ProcessShellCommand(cmdInfo)) return FALSE; // The one and only window has been initialized, so show and // update it. m_pMainWnd->ShowWindow(SW_SHOW); m_pMainWnd->UpdateWindow(); // Enable drag/drop open

Exploring an MFC Skeleton Application CHAPTER 13


m_pMainWnd->DragAcceptFiles(); return TRUE; }

243

The framework only created an override version of InitInstance, not InitApplication. Because Win32 applications run in separate memory spaces, application-specific (as opposed to instance-specific) initializations are now rare (as they would normally only affect the current instance of the application anyway, in contrast with 16-bit Windows where such initializations affected all copies of the application). In InitInstance, a number of initializations take place. These initialization steps reflect many of the choices you select when you create the project through AppWizard. For example, we selected the default 3D look for the YAH application; correspondingly, the 3D look is enabled here in InitInstance. Perhaps the most important initialization step is the creation of a document template. An object of type CSingleDocTemplate (because we selected an SDI application) is created and added to the applications document templates using the AddDocTemplate member function. The information stored in document templates is used when the user selects the New command from the File menu. The default implementation of this command is in the function CWinApp::OnFileNew. This function uses the template information to decide what kind of objects it must create to represent the new document object and its corresponding view. There are applications that can handle many kinds of documents. For example, a graphics application may be able to handle both bitmap and vector graphics files. A programmers editor may handle source (text) files and provide graphics editing for resource files. How can an MFC application accommodate multiple document types? Well, first you must create a multiple-document interface (MDI) application. SDI applications created with AppWizard do not support multiple document types. Afterwards, it takes little effort to add additional document types. After declaring and implementing a new document class and a corresponding view class, make sure these classes are added in the form of a new document template to the application object by calling AddDocTemplate in your application objects InitInstance member function. Subsequently, when the user selects the File New command, the framework automatically presents a dialog where the user can select the desired document type.

13
EXPLORING AN MFC SKELETON APPLICATION

The Message Map


In the files
BEGIN_MESSAGE_MAP,

and YAH.cpp , we encountered the macros DECLARE_MESSAGE_MAP , and END_MESSAGE_MAP. What do these macros represent and how do they connect to the applications main message loop in the application objects Run member function?
YAH.h

244

Microsoft Foundation Classes PART III

The Run member function dispatches messages to their target windows much like any nonMFC application would in its message loop. In fact, it calls the very same function, ::DispatchMessage, for this purpose. Thus, the first recipient of a message is always a window. The message handler function in an object capable of receiving messages (that is, a command target object, including window objects) generally dispatches, or routes, messages in the following order: 1. To any currently active child command target objects 2. To itself 3. To other command target objects For example, a command message that is ultimately processed by the applications document class may be routed through its frame window and view window first before eventually reaching the message handler in the document class. Table 13.1 summarizes how messages are handled by the major MFC command target classes.

Table 13.1. Message routing. Class


MDI frame windows (CMDIFrameWnd)

Routing order 1. Active MDI child window 2. This window 3. Application object 1. Active view 2. This window 3. Application object 1. This window 2. Attached document object 1. This document 2. Document template 1. This window 2. Owner window 3. Application object

Document frame windows (CMDIChildWnd, CFrameWnd) View Document Dialog box

So how do those message map macros relate to message processing? Simple. The DECLARE_MESSAGE_MAP macro declares an array of message map entries as part of your class declaration. The BEGIN_MESSAGE_MAP and END_MESSAGE_MAP macros enclose a series of initializers for this array that represent the individual messages that your class can respond to.

Exploring an MFC Skeleton Application CHAPTER 13

245

Look at the message map entries in YAH.cpp. These default entries connect a few standard commands in the File menu to default implementations supplied as part of the CWinApp class. ON_COMMAND is one of several macros that make creating message map entries easier. Normally, message map entries are created automatically by the ClassWizard; however, there are times when it is necessary to manually add entries (for example, when processing an applicationspecific message).

The Frame, the Document, and the View


In a simple non-MFC Windows application, you would typically create one window and use its client area to present your applications output. MFC applications, on the other hand, use at least two windows: a frame window and a view window. The frame window manages the applications menus, toolbars, and other user-interface components. The view window, in turn, is dedicated to presenting data from the applications document. The document object is not a visual object. It is an object that represents the applications data; it typically corresponds to the contents of a file. The document object closely interacts with the view window for presenting the data and for user interaction. The relationship between the frame and view windows and the document object is depicted in Figure 13.5.

13
EXPLORING AN MFC SKELETON APPLICATION

FIGURE 13.5.
Frames, views, and documents.

Frame Window File Edit View Help View Window

Document Object

The next sections present a look at the declaration and implementation of these three classes.

The Frame Window Class


The applications frame window is supported by the MainFrm.h (see Listing 13.5).
CMainFrame

class, which is declared in

246

Microsoft Foundation Classes PART III

Listing 13.5. CMainFrame class declaration.


class CMainFrame : public CFrameWnd { protected: // create from serialization only CMainFrame(); DECLARE_DYNCREATE(CMainFrame) // Attributes public: // Operations public: // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CMainFrame) virtual BOOL PreCreateWindow(CREATESTRUCT& cs); //}}AFX_VIRTUAL // Implementation public: virtual ~CMainFrame(); #ifdef _DEBUG virtual void AssertValid() const; virtual void Dump(CDumpContext& dc) const; #endif protected: // control bar embedded members CStatusBar m_wndStatusBar; CToolBar m_wndToolBar; // Generated message map functions protected: //{{AFX_MSG(CMainFrame) afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct); // NOTE - the ClassWizard will add and remove member // functions here. DO NOT EDIT what you see in these blocks // of generated code! //}}AFX_MSG DECLARE_MESSAGE_MAP() };

Nothing is surprising here: There is a constructor, a destructor, an overridden PreCreateWindow, an overridden OnCreate, and some debug member functions. However, I would like to call your attention to the two member variables m_wndStatusBar and m_wndToolBar. These correspond to the applications single toolbar and status bar. For any control bars that you might want to add to your program, this is the preferred way to do it; declare them as member variables of the frame window class and add supporting code in the frame window classs implementation file. The implementation of CMainFrame can be found in MainFrm.cpp (see Listing 13.6). In this file the OnCreate member function deserves a closer look. It is here in this function that the toolbar and status bar are initialized.

Listing 13.6. CMainFrame class implementation.


/////////////////////////////////////////////////////////////////// // CMainFrame

Exploring an MFC Skeleton Application CHAPTER 13


IMPLEMENT_DYNCREATE(CMainFrame, CFrameWnd) BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd) //{{AFX_MSG_MAP(CMainFrame) // NOTE - the ClassWizard will add and remove mapping macros here. // DO NOT EDIT what you see in these blocks of generated code ! ON_WM_CREATE() //}}AFX_MSG_MAP END_MESSAGE_MAP() static UINT indicators[] = { ID_SEPARATOR, // status line indicator ID_INDICATOR_CAPS, ID_INDICATOR_NUM, ID_INDICATOR_SCRL, }; /////////////////////////////////////////////////////////////////// // CMainFrame construction/destruction CMainFrame::CMainFrame() { // TODO: add member initialization code here } CMainFrame::~CMainFrame() { } int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct) { if (CFrameWnd::OnCreate(lpCreateStruct) == -1) return -1; if (!m_wndToolBar.Create(this) || !m_wndToolBar.LoadToolBar(IDR_MAINFRAME)) { TRACE0(Failed to create toolbar\n); return -1; // fail to create } if (!m_wndStatusBar.Create(this) || !m_wndStatusBar.SetIndicators(indicators, sizeof(indicators)/sizeof(UINT))) { TRACE0(Failed to create status bar\n); return -1; // fail to create } // TODO: Remove this if you dont want tool tips or a // resizeable toolbar m_wndToolBar.SetBarStyle(m_wndToolBar.GetBarStyle() | CBRS_TOOLTIPS | CBRS_FLYBY | CBRS_SIZE_DYNAMIC); // TODO: Delete these three lines if you dont want the toolbar // to be dockable m_wndToolBar.EnableDocking(CBRS_ALIGN_ANY); EnableDocking(CBRS_ALIGN_ANY); DockControlBar(&m_wndToolBar); return 0; } BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs) { // TODO: Modify the Window class or styles here by modifying // the CREATESTRUCT cs return CFrameWnd::PreCreateWindow(cs); }

247

13
EXPLORING AN MFC SKELETON APPLICATION

248

Microsoft Foundation Classes PART III

The Document Class


The declaration of the document class in YAHDoc.h in Listing 13.7 provides overrides for two functions: OnNewDocument and Serialize. OnNewDocument is called when the user selects the File New command. Although in MDI applications, OnNewDocument is called only when a new document object is created, the situation is different for SDI programs. In this case, OnNewDocument is called to reinitialize the applications one and only document object, so many initialization operations that would normally go into the constructor really belong here instead.

Listing 13.7. CYAHDoc class declaration.


class CYAHDoc : public CDocument { protected: // create from serialization only CYAHDoc(); DECLARE_DYNCREATE(CYAHDoc) // Attributes public: // Operations public: // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CYAHDoc) public: virtual BOOL OnNewDocument(); virtual void Serialize(CArchive& ar); //}}AFX_VIRTUAL // Implementation public: virtual ~CYAHDoc(); #ifdef _DEBUG virtual void AssertValid() const; virtual void Dump(CDumpContext& dc) const; #endif protected: // Generated message map functions protected: //{{AFX_MSG(CYAHDoc) // NOTE - the ClassWizard will add and remove member // functions here. DO NOT EDIT what you see in these blocks // of generated code ! //}}AFX_MSG DECLARE_MESSAGE_MAP() };

The Serialize member function is called when the document is loaded or saved. This member function must be overridden; you must write your own saving and loading code in the override version in order to save and load your document data. Apropos serializationisnt there a glaring inconsistency here? Why is the DECLARE_DYNCREATE macro used in the class declaration when this class obviously supports serialization? Shouldnt it be DECLARE_SERIAL instead?

Exploring an MFC Skeleton Application CHAPTER 13

249

Using DECLARE_SERIAL is unnecessary even though the class has a Serialize member function because the operator >> is never used to retrieve a document from a CArchive. The Serialize member function is called explicitly, from CDocument::OnOpenDocument . The use of DECLARE_SERIAL (and IMPLEMENT_SERIAL) is necessary only for classes that are loaded from a CArchive object using the >> operator. Both CYAHDoc override functions are implemented in the file YAHDoc.cpp (see Listing 13.8). Their default implementations do nothing; you must supply the code to initialize your document type, and save and load document data.

Listing 13.8. CYAHDoc class implementation.


/////////////////////////////////////////////////////////////////// // CYAHDoc IMPLEMENT_DYNCREATE(CYAHDoc, CDocument) BEGIN_MESSAGE_MAP(CYAHDoc, CDocument) //{{AFX_MSG_MAP(CYAHDoc) // NOTE - the ClassWizard will add and remove mapping macros here. // DO NOT EDIT what you see in these blocks of generated code! //}}AFX_MSG_MAP END_MESSAGE_MAP() /////////////////////////////////////////////////////////////////// // CYAHDoc construction/destruction CYAHDoc::CYAHDoc() { // TODO: add one-time construction code here } CYAHDoc::~CYAHDoc() { } BOOL CYAHDoc::OnNewDocument() { if (!CDocument::OnNewDocument()) return FALSE; // TODO: add reinitialization code here // (SDI documents will reuse this document) return TRUE; } ///////////////////////////////////////////////////////////////////////////// // CYAHDoc serialization void CYAHDoc::Serialize(CArchive& ar) { if (ar.IsStoring()) { // TODO: add storing code here } else { // TODO: add loading code here } }

13
EXPLORING AN MFC SKELETON APPLICATION

250

Microsoft Foundation Classes PART III

The View Class


The default declaration of the view class in YAHView.h includes several function overrides (see Listing 13.9). Perhaps the most significant of these is OnDraw; this function is responsible for presenting a visual representation of the data of the document that corresponds to this view.

Listing 13.9. CYAHView class declaration.


class CYAHView : public CView { protected: // create from serialization only CYAHView(); DECLARE_DYNCREATE(CYAHView) // Attributes public: CYAHDoc* GetDocument(); // Operations public: // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CYAHView) public: virtual void OnDraw(CDC* pDC); // overridden to draw this view virtual BOOL PreCreateWindow(CREATESTRUCT& cs); protected: virtual BOOL OnPreparePrinting(CPrintInfo* pInfo); virtual void OnBeginPrinting(CDC* pDC, CPrintInfo* pInfo); virtual void OnEndPrinting(CDC* pDC, CPrintInfo* pInfo); //}}AFX_VIRTUAL // Implementation public: virtual ~CYAHView(); #ifdef _DEBUG virtual void AssertValid() const; virtual void Dump(CDumpContext& dc) const; #endif protected: // Generated message map functions protected: //{{AFX_MSG(CYAHView) // NOTE - the ClassWizard will add and remove member // functions here. DO NOT EDIT what you see in these blocks // of generated code ! //}}AFX_MSG DECLARE_MESSAGE_MAP() };

Notice that this class, like the document class, is also declared with the DECLARE_DYNCREATE macro. Use of this macro is necessary because when a new document is created, the view object is created dynamically. The implementation of the view class in YAHView.cpp contains few surprises (see Listing 13.10). The override functions are only skeletons; you must supply your own implementation. However, only the OnDraw member function must be edited in order to obtain a functional

Exploring an MFC Skeleton Application CHAPTER 13

251

application. In order to have printing capability, it is not necessary to adjust any printingrelated member functions here, although you would probably want to do so because the default printing behavior may not be satisfactory.

Listing 13.10. CYAHView class implementation.


/////////////////////////////////////////////////////////////////// // CYAHView IMPLEMENT_DYNCREATE(CYAHView, CView) BEGIN_MESSAGE_MAP(CYAHView, CView) //{{AFX_MSG_MAP(CYAHView) // NOTE - the ClassWizard will add and remove mapping macros here. // DO NOT EDIT what you see in these blocks of generated code! //}}AFX_MSG_MAP // Standard printing commands ON_COMMAND(ID_FILE_PRINT, CView::OnFilePrint) ON_COMMAND(ID_FILE_PRINT_DIRECT, CView::OnFilePrint) ON_COMMAND(ID_FILE_PRINT_PREVIEW, CView::OnFilePrintPreview) END_MESSAGE_MAP() /////////////////////////////////////////////////////////////////// // CYAHView construction/destruction CYAHView::CYAHView() { // TODO: add construction code here } CYAHView::~CYAHView() { } BOOL CYAHView::PreCreateWindow(CREATESTRUCT& cs) { // TODO: Modify the Window class or styles here by modifying // the CREATESTRUCT cs return CView::PreCreateWindow(cs); } /////////////////////////////////////////////////////////////////// // CYAHView drawing void CYAHView::OnDraw(CDC* pDC) { CYAHDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here } /////////////////////////////////////////////////////////////////// // CYAHView printing BOOL CYAHView::OnPreparePrinting(CPrintInfo* pInfo) { // default preparation return DoPreparePrinting(pInfo); } void CYAHView::OnBeginPrinting(CDC* /*pDC*/, CPrintInfo* /*pInfo*/) { // TODO: add extra initialization before printing } void CYAHView::OnEndPrinting(CDC* /*pDC*/, CPrintInfo* /*pInfo*/) { // TODO: add cleanup after printing }

13
EXPLORING AN MFC SKELETON APPLICATION

252

Microsoft Foundation Classes PART III

Notice that there are several message map entries here that are related to printing. They call the base class functions that implement default printing and print preview behavior.

Skeleton Application Resources


To complete our tour of the skeleton MFC application, here is a brief look at the resources that were generated by AppWizard. To see the list of resources, open the project in ResourceView and expand the single item seen in this view. The accelerator resource requires little explanation; it contains the keyboard shortcuts to many standard menu functions. The menu bar itself is defined in the applications single menu resource. AppWizard created one dialog resource, an About dialog. This dialog is displayed when the user selects the About command from the Help menu. Two icons have been generated; IDR_MAINFRAME is the application icon, and IDR_YAHTYPE is the icon representing the applications document type. The string table contains numerous strings. Many of these correspond to MFC framework messages; others represent status bar messages, tooltips, and other text items specific to this application. Of particular interest is the string resource IDR_MAINFRAME, also referred to as the document template string. This string contains up to nine substrings, separated by the newline (\n) character. Here is what it has been set to by AppWizard:
Hello, World!\n\nYAH\nYAH Files (*.yah)\n.YAH\nYAH.Document\nYAH Document

The substrings of the document template string are described in Table 13.2. The general syntax for this string is the following:
<windowTitle>\n<docName>\n<fileNewName>\n<filterName>\n <filterExt>\n<regFileTypeID>\n<regFileTypeName>

Table 13.2. Substrings of the document template string. Substring Description


<windowTitle> <docName>

<fileNewName>

<filterName> <filterExt> <regFileTypeID> <regFileTypeName>

The title of the applications main frame window Root document name for document windows (this name plus a number will be used as window titles) Document type displayed in the File New dialog when the application supports multiple types Filter used in the file dialogs Extension used in the file dialogs File type registered in the Registry Visible name of the file type registered in the Registry

Exploring an MFC Skeleton Application CHAPTER 13

253

The resource file also contains a toolbar resource and a version resource. Note how several resources share the same identifier, IDR_MAINFRAME. Such common identifiers are used when the application calls the CSingleDocTemplate (or CMultiDocTemplate) constructor. It identifies the menu, icon, accelerator table, and document template string corresponding to a specific document type.

Adding Code to the Application


Now that we have seen the basic elements of an MFC skeleton, it is time to look at actually modifying the skeleton by adding some of our own code. Well try something simple this time. In the document class, we will add a string member variable and initialize it from a resource; in the view class, we will add code to display this string in the middle of the applications view window.

Adding a String Resource


To add a string resource, open the project workspace in ResourceView and open the string table. Add a string named IDS_HELLO and set its value to Hello, World! (or whatever else may suit your taste).

13
EXPLORING AN MFC SKELETON APPLICATION

Modifying the Document


The first step in modifying our application is to add a member variable to the document class. To do this, edit the YAHDoc.h file. In the Attributes section, add a declaration for a member variable of type CString as follows:
// Attributes public: CString m_sData;

Obviously, m_sData must be initialized somewhere. We must also add this member to the Serialize member function to enable it to be saved to, and loaded from, a file. These changes are carried out in the YAHDoc.cpp file. We will initialize the string in the OnNewDocument member function to ensure that it is reinitialized every time the user selects the New command from the File menu. Here is the modified OnNewDocument:
BOOL CYAHDoc::OnNewDocument() { if (!CDocument::OnNewDocument()) return FALSE; // TODO: add reinitialization code here // (SDI documents will reuse this document)

254

Microsoft Foundation Classes PART III


m_sData.LoadString(IDS_HELLO); return TRUE; }

And here is the modified Serialize member function:


void CYAHDoc::Serialize(CArchive& ar) { if (ar.IsStoring()) { // TODO: add storing code here ar << m_sData; } else { // TODO: add loading code here ar >> m_sData; } }

We are almost finished! All that is left is to actually display the string; this must be implemented as part of our view class.

Modifying the View


To display our string, we must modify the view classs OnDraw member function. Because AppWizard already provided us with an empty implementation for this function, it is not necessary to modify the class declaration; we only add code to the existing function skeleton in YAHView.cpp:
void CYAHView::OnDraw(CDC* pDC) { CYAHDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here CRect rect; GetClientRect(&rect); pDC->DPtoLP(&rect); pDC->DrawText(pDoc->m_sData, &rect, DT_CENTER | DT_VCENTER | DT_SINGLELINE); }

Exploring an MFC Skeleton Application CHAPTER 13

255

All that is left is to recompile and run the application. If all goes well, the application window should look similar to that shown in Figure 13.6.

FIGURE 13.6.
The Yet Another Hello World application.

Summary
The MFC Library represents a powerful framework for constructing Windows applications. Classes in MFC encapsulate most Windows functionality. The root of most MFC classes is the CObject class. This class implements runtime type checking (distinct from the new C++ RTTI feature) and serialization. The major MFC categories include Application Architecture classes, Window Support classes, and other classes that encapsulate system, GDI, and miscellaneous services. MFC applications are created through AppWizard. At the heart of every MFC application is a CWinApp-derived object, which implements application initialization and the applications main message loop. Messages are dispatched and routed through message maps, which are a feature of command handler objects (such as windows). The CWinApp::Run member function dispatches messages through ::DispatchMessage; further routing takes place according to MFCs message routing rules. Generally, a command handler object routes a message first to any child command handler objects; next, to itself; and finally, to additional command handler objects.

13
EXPLORING AN MFC SKELETON APPLICATION

256

Microsoft Foundation Classes PART III

Visual presentation of an application and management of the applications data are a result of cooperation between a frame window, a view window, and a document object. The document object holds the applications data and generally corresponds to a disk file. The view window is used to present the contents of a document to the user and accept user interaction. The view window works hand-in-hand with the frame window, which manages other elements of the applications user interface, such as its menu bar, toolbars, or status bar. When implementing an MFC application, one typically edits the document and view classes simultaneously. Representations of new document objects are declared as members of the document class; the visual interfaces corresponding to the new elements are implemented as part of the view class.

Working with Documents and Views CHAPTER 14

257

14

Working with Documents and Views


IN THIS CHAPTER
s The CDocument Class s The CView Class 266 258

14
WORKING WITH DOCUMENTS AND VIEWS

258

Microsoft Foundation Classes PART III

At the core of an MFC application is the concept of a document object and a corresponding view window. The document object represents (usually) a file opened by the application; the view window provides a visual presentation of the documents data and accepts user interaction. The relationship between documents and views is a one-to-many relationship; a view can be associated with only one document, but a document may have many views associated with it. Document objects are represented by classes derived from CDocument. View window classes are derived from CView. This chapter reviews these two classes and the most common ways of utilizing their capabilities to build a versatile representation of your data; it also discusses an efficient user interface.

The CDocument Class


The CDocument class provides the basic functionality for your applications document objects. This includes the ability to create a new document, serialize document data, and provide basic cooperation between a document and a view window. MFC also provides a series of CDocumentderived classes that implement functionality specific to OLE applications.

Declaring a Document Class in Your Application


In the case of an AppWizard-created application, you often dont have to worry about declaring your document class because the AppWizard does it for you. However, it is still useful to know more about the behavior of CDocument. Not only does this knowledge enable you to enhance the AppWizard-provided application skeleton, it may also help you easily add additional document types that your application supports. The AppWizard, in contrast, only creates application skeletons that support a single document type. When you are building a simple MFC application, it is often enough to make relatively minor modifications to your applications AppWizard-supplied document class. Often no more is needed than a few member variables and perhaps a couple of member functions that provide access to those variables. For example, consider a simple communication program (terminal emulator). Its document object is a series of settings (telephone number, speed, parity, and so on) that correspond to a connection. These can easily be represented by a set of simple data items in the document class, something similar to the following:
class CTerminalDoc : public CDocument { protected: // create from serialization only CTerminalDoc(); DECLARE_DYNCREATE(CTerminalDoc) // Attributes public: CString m_sPhone; DWORD m_dwSpeed;

Working with Documents and Views CHAPTER 14


WORD m_nParity; WORD m_nBits; ...

259

In addition to the declaration of member variables, all you need to do is to initialize them to reasonable defaults in your document classs OnNewDocument member function and ensure that they are properly serialized:
... BOOL CTerminalDoc::OnNewDocument { if (!CDocument::OnNewDocument()) return FALSE; m_sPhone = _T(555-1212); m_dwSpeed = 2400; m_nParity = 0; m_nBits = 8; return TRUE; } ... void CTerminalDoc::Serialize(CArchive &ar) { if (ar.IsStoring()) { ar << m_sPhone; ar << m_dwSpeed; ar << m_nParity; ar << m_nBits; } else { ar >> m_sPhone; ar >> m_dwSpeed; ar >> m_nParity; ar >> m_nBits; } }

14
WORKING WITH DOCUMENTS AND VIEWS

For a simple application, nothing else needs to be done to have a complete, fully functional document class.

CDocument Member Functions


The CDocument class has several member functions that are frequently used by applications. The first set of member functions provides access to the associated view objects. Every document object has a list of view objects associated with it. An iterator to this list, in the form of a variable of type POSITION, can be obtained by calling the GetFirstViewPosition member function. Values of type POSITION are used throughout the MFC, primarily in association with collection classes. Applications that need to traverse a list usually obtain an iterator that is associated with the first object on the list, and then use an iterator function to access the lists elements

260

Microsoft Foundation Classes PART III

one by one. The case of CDocument and its associated views is no different; after obtaining a list iterator using GetFirstViewPosition, the elements of the list can be obtained by repeatedly calling GetNextView. Thus, to process all the views associated with a document, your code would typically look like this:
POSITION pos = GetFirstViewPosition(); while (pos != NULL) { CView *pView = GetNextView(pos); // Do something with pView }

If all you want to accomplish is to notify the views for this document that the document has changed, it may not be necessary to use an iteration at all. Instead, you can call the UpdateAllViews member function. When calling this member function, you can also specify applicationspecific data that enables the view objects to selectively update only portions of the view windows. Well take another look at this issue later in this chapter, when we discuss the CView::OnUpdate member function. Much less frequently used view-related functions are AddView and RemoveView. These functions let you manually add views to and remove views from your documents list of views. The reason these functions are not used that often is that most applications rely on the default MFC implementation with little or no modification for managing their windows. Whenever the documents data changes, you should call the SetModifiedFlag member function. Consistent use of this function ensures that the framework prompts the user before destroying an unsaved, changed document. The status of this flag can be obtained by calling the IsModified member function. The SetTitle member function can be used to set the documents title. This title is displayed as the documents title in the frame window (the main frame window in the case of an SDI application or the child frame in the case of an MDI application). The fully qualified path name for the document can be set by calling SetPathName and obtained through GetPathName. The document template object associated with the current document can be obtained by calling GetDocTemplate.

Documents, Events, and Overridable Functions


Although a CDocument object is not directly associated with a window, it is nevertheless a command target object that can receive messages. Messages are routed to CDocument objects by the associated view objects.

Working with Documents and Views CHAPTER 14

261

Although it is up to you to decide which messages will be handled by your document object and which should be left to the view window (or perhaps the frame window) for processing, there are a few sensible rules of thumb to follow. Always keep in mind that the document is an abstract presentation of your data, independent of the visual representation provided by the view window. Moreover, a document may have several views attached to it (or possibly none at all). Any messages the document responds to should be global in nature, having an immediate effect on the document data itself that should be reflected in all the views. In contrast, views should respond to messages that are specific to that window only. How does this translate into practical terms? Take, for example, the command message that is generated when the user selects the Save command from the File menu. What you are saving is the document as a whole, not a visual representation of it; thus, this command is best handled by the document class. Take, in contrast, the Copy command in the Edit menu. Whatever you are copying to the clipboard is selected through a view of the document. In fact, if multiple views exist for the same document, chances are that different selections are active in them; thus the meaning of the Copy command changes from one view to the next. Conclusion: This command should likely be handled by the view class. Then there are some borderline cases. Is the Paste command best handled by the document class or the view class? True, this command affects the entire document, not just a single view, as it inserts data into your document object. However, it may have particular effects in the current viewfor example, it may cause the current selection to be replaced by the pasted data. Therefore, the decision regarding which class should handle this command is dependent on your applications design. I should also mention that there are commands that should not be handled by either the view class or the document class, but by the frame window instead. Commands that hide and show toolbars are good examples. The presence or absence of a toolbar is not a feature of a document or one of its views; instead, this is a configuration issue with an effect thats global to the entire application. Now well return our attention to the CDocument class. The MFC framework provides default implementations for many commands; these implementations, in turn, call overridable member functions in CDocument. (These functions are overridable because they are declared virtual; thus, you can provide your overrides in a class derived from CDocument and expect the override version to be called instead of the base class version.) The OnNewDocument member function is called during the initialization of a new document object (or when an existing document is reused in an SDI application). Calling this function is typically part of handling the File New command.

14
WORKING WITH DOCUMENTS AND VIEWS

262

Microsoft Foundation Classes PART III

The OnCloseDocument member function is called when a document is about to be closed. You should override this function if it is necessary to perform any cleanup operations before your document is destroyed. The OnOpenDocument and OnSaveDocument functions are called to read a document from disk or save the document to a disk file. You should override these functions only if the default implementation (which calls your document classs Serialize member function) is not sufficient for your purposes. The DeleteContents function is called from the default implementations of OnCloseDocument and OnOpenDocument to delete the documents previous contents before opening the new file. This function deletes the documents data without actually destroying the document object. The OnFileSendMail member function sends the document object as an attachment to a mail message. It calls OnSaveDocument to save a copy of the document to a temporary disk file, which it then attaches to a MAPI mail message. The OnUpdateFileSendMail member function is used to enable the command identified by ID_FILE_SEND_MAIL in the applications menu or remove it altogether if MAPI support is not available. Both OnFileSendMail and OnUpdateFileSendMail are overridable functions, which enables you to implement customized messaging behavior.

Document Data
I already mentioned simple CDocument-derived classes, where the documents data can be implemented in the form of simple member variables. However, real-world applications tend to be more demanding, their data requirements far beyond what can be reasonably represented by a few variables of simple data types. Perhaps the best approach to implement an application with a complex series of data elements is to use a set of CObject-derived classes to represent the data elements themselves, while relying on a standard or custom collection class to embed these elements in your document class. For example, in one application I created I used classes like this:
class CMyObject : public CObject { // ... }; class CMyFirstSubObject : public CObject { // ... }; class CMySecondSubObject : public CObject { // ... };

In the declaration of the document class, I included a CObList member:


class CMyDocument : public CDocument { // ...

Working with Documents and Views CHAPTER 14


// Attributes public: CObList m_obList; // ... };

263

In a complex situation like this, it is often not sufficient to just declare member variables. Member functions are also needed that provide methods to access the documents data. For example, in the preceding case you may not want to allow other classes (such as the view class) to manipulate the m_obList member variable directly; instead, you may wish to provide member functions that add data to or remove data from this list. Such member functions should also ensure that all the documents views are updated properly. They should also call the documents SetModified member function to indicate that a change to the documents data has been made. If your application supports an undo capability, this is where you should update your buffered undo data. As a simple example, consider the following function, which updates the documents object list by adding a new object:
BOOL CMyDocument::AddObject(CMyObject *pObject) { try { m_obList.AddTail((CObject *)pObject); SetModifiedFlag(TRUE); UpdateAllViews(NULL, UPDATE_OBJECT, pObject); return TRUE; } catch(CMemoryException *e) { TRACE(CMyDocument::AddObject memory allocation error.\n); e->Delete(); return FALSE; } }

14
WORKING WITH DOCUMENTS AND VIEWS

Consider, for a moment, how control is passed back and forth between the document and its views. First, the user interacts with the view, which results in a new object being added. The view object than calls the document objects AddObject member. Once the new object has been added successfully, the document object calls UpdateAllViews, which, in turn, calls the OnUpdate member function of each view associated with the document. The hint passed to UpdateAllViews (in the form of the application-defined constant UPDATE_OBJECT and a pointer to a CObject) assists views in implementing an efficient window update by only repainting those regions that are affected by the appearance of the new object. This control-passing mechanism is illustrated in Figure 14.1.

264

Microsoft Foundation Classes PART III

FIGURE 14.1.
Interaction between the view and the document when an item is modified.

View Class
// The user adds an object GetDocument->AddObject( );

Document Class
AddObject ( ) { m_obList.AddTail ( ); SetModifiedFlag ( ); UpdateAllViews ( );

OnUpdate ( ) { // The view is updated }

Another advantage of using MFC collection classes is that they support serialization. For example, to load and save your documents data that is stored in the form of a CObList collection, all you need to do in the documents Serialize member function is this:
void CTerminalDoc::Serialize(CArchive &ar) { if (ar.IsStoring()) { // Serialize any non CObject-derived data } else { // Serialize any non CObject-derived data } m_obList.Serialize(ar); }

Be warned, though, that for this to work you must implement the Serialize member function for all your object classes. A CObject-derived class will not magically serialize itself. If you decide to use one of the general-purpose collection templates, serialization is an issue that requires special attention. The collection templates CArray, CList, and CMap rely on the SerializeElements function to serialize the objects in the collection. This function is declared as follows:
template <class TYPE> void AFXAPI SerializeElements(CArchive &ar, TYPE *pElements, int nCount);

Because the collection class templates do not require TYPE to be derived from CObject, they do not call the Serialize member function of their elements (simply because this member function is not guaranteed to exist). Instead, the default implementation of SerializeElements performs a bitwise read or write. This is definitely not what we want in most cases! (Arguably, it might have been better if the MFC provided no default implementation at all, thus forcing the programmer to write SerializeElements rather than fall prey to a subtle trap.) Here is an example of how you would implement SerializeElements for an object type you define that supports a Serialize member function:
void SerializeElements(CArchive &ar, CMyObject **pObs, int nCount) { for (int i = 0; i < nCount; i++, pObs++) (*pObs)->Serialize(ar); }

Working with Documents and Views CHAPTER 14

265

CCmdTarget and CDocItem


Often it is not sufficient to derive your documents objects from the CObject class. A prime example for this is when you wish to support OLE automation. OLE automation support requires that your objects be command targets; something that CObject does not support. For this reason, it may be beneficial to use CCmdTarget as the base class for your objects. Better yet, you should consider the CDocItem class. You can either create a collection of CDocItem objects yourself or rely on the COleDocument class for this purpose; that is, derive your document class from COleDocument instead of CDocument. COleDocument is used in OLE applications where either this class or a class derived from it serves as the base class for the OLE applications document class. COleDocument supports a collection of CDocItem objects; these are objects of type COleServerItem and COleClientItem. However, the support for a list of CDocItem objects in COleDocument is generic. You can add your own CDocItem-derived objects to the collection maintained by COleDocument and not fear that it would interfere with normal OLE behavior. How do you declare additional CDocItem members in a COleDocument? Funny thing is, you dont have to! All you need to do is use COleDocument member functions such as AddItem, RemoveItem, GetStartPosition, and GetNextItem to add, remove, and retrieve document items. The rest (such as serialization) comes for free. There is a catch, though. Because of how your document items and the OLE COleClientItem and COleServerItem objects are derived, it may be necessary to add some magic to implement certain functions. For example, consider that you declared your objects as follows:
class CMyDocItem : public CDocItem { // ... CRect m_rect; };

Further suppose that you also support the m_rect member variable in your OLE client items:
class CMyClientItem : public COleClientItem { // ... CRect m_rect; };

14
WORKING WITH DOCUMENTS AND VIEWS

Given these class declarations, how would you create a function that can take an item from your document and utilize its m_rect member? The obvious answer is also the wrong one:
MyFunc(CDocItem *pItem) { AnotherFunc(pItem->m_rect); } // Error!

266

Microsoft Foundation Classes PART III

This will not compile because the CDocItem class has no member variable named m_rect. Using a pointer to your own CDocItem-derived class does not help either:
MyFunc(CMyDocItem *pItem) { AnotherFunc(pItem->m_rect); }

This version of MyFunc does not support OLE client items of type CDocItem, only your own derived class. Obviously, you could simply create two override versions of MyFunc, but it is a real pain having to maintain two identical versions because of this problem. So the solution that remains is to create a wrapper function that takes a pointer to a CDocItem object and uses MFC runtime type information to obtain the member variable:
CRect GetRect(CDocItem *pDocItem) { if (pDocItem->IsKindOf(RUNTIME_CLASS(CMyDocItem))) return ((CMyDocItem *)pDocItem)->m_rect; else if (pDocItem->IsKindOf(RUNTIME_CLASS(CMyClientItem))) return ((CMyClientItem *)pDocItem)->m_rect; ASSERT(FALSE); return CRect(0, 0, 0, 0); } MyFunc(CDocItem *pItem) { AnotherFunc(GetRect(pItem)); }

Note that this solution requires that both CMyDocItem and CMyClientItem be declared and implemented using the DECLARE_DYNAMIC/IMPLEMENT_DYNAMIC macros. This is usually not a problem as your application probably supports serializing these items and thus the items are declared and implemented using DECLARE_SERIAL/IMPLEMENT_SERIAL (which imply DECLARE_DYNAMIC/ IMPLEMENT_DYNAMIC).

The CView Class


For every CDocument-derived class, there is a CView-derived class that provides the visual presentation of your documents data and handles user interaction through the view window. The view window is a child of a frame window. In the case of SDI applications, this is the main frame window. For MDI applications, this is the MDI child frame. Additionally, it can be the in-place frame window during OLE in-place editing (if your application supports it). A frame window may contain several view windows (for example, through the use of splitter windows).

Working with Documents and Views CHAPTER 14

267

Declaring a View Class


All data that is part of a document should be declared as part of the document class. That said, there are many data elements that pertain to a specific view and, more importantly, are transient in nature, not saved as part of the document. Consider, for example, an application that is capable of presenting its data at different zoom factors. The zoom factor is specific to an individual view. (Different views may use different zoom factors even when they present parts of the same application.) The zoom factor is also transient; it is not saved as part of the document. Under these conditions, the zoom factor would best be declared as a member variable of your view class:
class CZoomView : public CView { protected: // create from serialization only CZoomView(); DECLARE_DYNCREATE(CZoomView) // Attributes public: CZoomDoc* GetDocument(); double m_dZoom; ...

Much more important than any member variables representing a setting is a member variable that represents the current selection. This is the collection of objects in your document that the user selected for manipulation. The nature and type of that manipulation are entirely application-dependent, but may include such interapplication operations as clipboard cut and copy or OLE drag and drop. Perhaps the easiest way to implement a selection is to use a collection class just as you would in the document class. For example, you may declare the collection representing the current selection like this:
class CMyView : public CView { // ... CList<CDocItem *, CDocItem *> m_selList; // ...

14
WORKING WITH DOCUMENTS AND VIEWS

In addition to modifying the declaration of the view class, you must write at least one member function to give your view class some functionality. The function in question is the OnDraw member function. The default implementation does nothing; you must write code here that displays your documents data items. For example, if your document class is derived from COleDocument and you rely on CDocItem objects for your documents data, your OnDraw member function implementation may look like this:
void CMyView::OnDraw(CDC *pDC) {

268

Microsoft Foundation Classes PART III


CMyDoc *pDoc = GetDocument(); ASSERT_VALID(pDoc); POSITION pos = pDoc->GetStartPosition(); while (pos != NULL) { CDocItem *pObject = pDoc->GetNextItem(pos); if (pObject->IsKindOf(RUNTIME_CLASS(CMyDocItem))) { ((CMyDocItem *)pObject)->Draw(pDC); } else if (pObject->IsKindOf(RUNTIME_CLASS(CMyClientItem))) { ((CMyClientItem *)pObject)->Draw(pDC); } else ASSERT(FALSE); } }

CView Member Functions


The CView class offers a rich selection of member functions. Among the most commonly used member functions is GetDocument, which returns a pointer to the document object associated with the view. Another member function is DoPreparePrinting; this function displays the Print dialog and creates a printer device context in accordance with the users selections. The remaining CView member functions are overridables. They supplement the large number of overridable functions available as part of the CWnd class (the base class of CView) and handle most types of user-interface events. These functions are far too numerous to be listed here; among them are message handlers for keyboard, mouse, timer, system, and other messages, clipboard and MDI events, initialization and termination messages. Your application should provide overrides for these as appropriate; for example, if your application enables the user to place an object in a document by clicking and dragging the mouse, you should provide an override for the CWnd::OnLButtonDown member function. As most of these overrides are recognized by the ClassWizard, adding and manipulating them is easy. There are some notable CView overridables. One, I already mentioned; overriding OnDraw is a must for a CView-derived object to display anything. The IsSelected member function must be implemented for OLE applications. This function returns TRUE if the object that is pointed to by its argument is part of the views current selection. If you implemented your selection using the CList template collection as a list of CDocItem objects, here is how you could implement IsSelected:
BOOL CMyView::IsSelected(const CObject* pDocItem) const { return (m_selList.Find((CDocItem *)pDocItem) != NULL); }

Working with Documents and Views CHAPTER 14

269

Another important overridable is the OnUpdate member function. This function is called by the UpdateAllViews member function of the document class associated with the view. The default implementation simply invalidates the entire client area of the view window. To improve your applications performance, you may wish to override this function and invalidate only those areas that need updating. For example, you may implement OnUpdate as follows:
void CMyView::OnUpdate(CView *pView, LPARAM lHint, CObject *pObj) { if (lHint == UPDATE_OBJECT) // hint constant defined by you InvalidateRect(((CMyObject *)pObj)->m_rect); else Invalidate(); }

Normally you should not do any drawing in OnUpdate. Use your views OnDraw member function for that purpose. The OnPrepareDC member function acquires special significance if your view supports nonstandard mapping modes like zooming. It is in this function that you can set the view windows mapping mode before any actual drawing takes place. Make sure that if you create a device context for your view window, you call OnPrepareDC yourself to ensure that the proper settings are applied to the device context. Sometimes it is necessary to create a device context just to retrieve the current mapping using OnPrepareDC. For example, your views OnLButtonDown member function may need to convert the position of the mouse click from physical to logical coordinates:
void CMyView::OnLButtonDown(UITN nFlags, CPoint point) { CClientDC dc(this); OnPrepareDC(&dc); dc.DPtoLP(&point); // ...

14
WORKING WITH DOCUMENTS AND VIEWS

Other CView overridables deal with initialization and termination, OLE drag and drop support, scrolling, view activation and deactivation, and printing. Whether these functions require overriding or not depends on whether you support the particular feature, and whether the default implementation (if it exists) is sufficient for your purposes or not.

Views and Messages


In addition to messages for which default handlers already exist in CView or its parent, CWnd, a typical view class handles many other messages. These are typically command messages representing the users selection of a menu command, toolbar button, or other user-interface object. As explained earlier, when you are deciding whether it is the view or the document (or the frame) that should handle a particular message, the prevailing criteria is the scope and the effect of the message or command. If the command affects the entire document, it is best handled by the document class (unless the commands effect is through a specific view, as in some implementations of the Paste command). If the command only affects a particular view (such as setting a zoom factor), it should be handled by that view object.

270

Microsoft Foundation Classes PART III

Variants of CView
In addition to the basic CView class, the MFC Library provides several derived classes that serve specific purposes. These classes are summarized in Table 14.1.

Table 14.1. CView variants. Class Name Description


CCtrlView

CDaoRecordView CEditView CFormView CHtmlView CListView CRecordView CRichEditView CScrollView CTreeView

Supports views that are based on a control (such as a tree or edit control) Displays database records using dialog controls Provides a multiline text editor window using an edit control Based on a dialog template; displays dialog box controls Provides Web browseer capability in a view Displays a list control Displays database records using dialog controls Displays a rich-text edit control Enables the use of scrollbars Displays a tree control

A rarely overridden variant of CView is CPreviewView; this class is used by the MFC framework to provide print preview support. All these classes provide member functions that are specific to their function. Member functions of view classes derived from CCtrlView encapsulate Windows messages that are specific to the control class they represent. classes derived from it (CDaoRecordView and CRecordView) support Dialog Data Exchange. You can use these classes in a fashion similar to the way CDialog-derived classes would be used.
CFormView and

Dialog-Based Applications
Dialog-based applications represent an exception from the standard MFC document-view model. If you create a dialog-based application using the AppWizard, the resulting program will not have a document or a view class (nor a frame window class, for that matter). Instead, all functionality will be implemented by a single dialog class, derived from CDialog. Although this is sufficient for many simple applications, it also means a loss of support for many MFC features that you have come to like. A dialog-based application will have no menu, toolbar, or status bar; it will not support OLE or MAPI; it will not have printing capabilities.

Working with Documents and Views CHAPTER 14

271

An alternative to using a dialog-based application is to build your application using the CFormView class as the base class for your view window and utilize the SDI application model. This enables you to retain all the advantages of a full-featured MFC application, yet present the same dialog-like appearance, utilize a dialog box template for defining the views contents, and use dialog data exchange.

Summary
Most MFC applications are based on the document-view model. The document, an abstract object, represents the applications data and typically corresponds to the contents of a file. The view, in turn, provides presentation of the data and accepts user-interface events. The relationship is one-to-many; a document may have several associated views, but a view is always associated with exactly one document. Document classes are derived from CDocument . This class encapsulates much of the basic functionality of a document object. In the simplest case, applications need only add member variables representing application-specific data and provide overrides for the OnNewDocument (initialization) and Serialize (saving and loading) member functions to obtain a fully functional document class. More sophisticated applications often rely on collection classes to implement the set of objects that comprise a document. In particular, applications can use the COleDocument class and rely on its capability to manage a list of CDocItem objects that is not restricted to OLE client and server objects. View classes are derived from CView. View windows that are represented by CView objects are child windows of frame windows; a frame window may have several child view windows, as is the case when splitter windows are used. A view object, in addition to containing member variables representing view-specific settings, often implements a current selection. The current selection is the set of document objects that the user designated in the current view for further manipulation. As with documents, applications can use collection classes for this purpose. At the very least, a view class must provide an implementation for the OnDraw member function to draw the objects of the associated document. OLE applications must also provide an implementation for the IsSelected member function. Other, frequently overridden, member functions include OnPrepareDC and OnUpdate. The CView class has several variants specifically designed to handle scrolling views, views based on dialogs, controls, and views representing database records. You should select the class that is most appropriate for your application as the base class for your view class.

14
WORKING WITH DOCUMENTS AND VIEWS

272

Microsoft Foundation Classes PART III

Both documents and views (as well as frame objects) handle messages. A decision about which of these three classes should handle a particular message is based upon the effects of the message. If the message affects the entire document, it is the document class that should handle the message, unless the effect takes place through a specific view. In that case, or in the case when the effect is specific to a view, it is the view class that should provide handling. Lastly, messages that have a global effect on the application are best handled by the frame class.

Dialogs and Property Sheets CHAPTER 15

273

Dialogs and Property Sheets

15

IN THIS CHAPTER
s Constructing Dialogs 274 s More on Dialog Data Exchange 283 s Dialogs and Message Handling 285 s Property Sheets 286

15
DIALOGS AND PROPERTY SHEETS

274

Microsoft Foundation Classes PART III

Applications use dialogs in many situations. The MFC Library supports dialogs through the CDialog class and derived classes. A CDialog object corresponds to a dialog window, the content of which is based on a dialog template resource. The dialog template resource can be edited using any dialog editor; typically, however, you would use the dialog editor that is part of the Developer Studio for this purpose. Perhaps the most important feature of CDialog is its support for Dialog Data Exchange, a mechanism that facilitates the easy and efficient exchange of data between controls in a dialog and member variables in the dialog class. The CDialog class is derived from CWnd; thus, you can use many CWnd member functions to enhance your dialog. Furthermore, your dialog classes can have message maps; indeed, except for the most simple dialogs, it is often necessary to add message map entries to handle specific control messages. Newer applications often support tabbed dialogs, or property sheets. A property sheet is really several dialogs merged into one; the user uses tab controls to pick any one of the property pages that comprise a property sheet. Our tour of MFC dialog support starts with a review of how simple dialogs are constructed in an MFC application.

Constructing Dialogs
The basic steps in constructing a dialog and making it part of your application include creating the dialog template resource, creating a CDialog-derived class that corresponds to this resource, and constructing an object of this class at the appropriate location in your application. For our experiments with dialogs, we use a simple AppWizard-created SDI application named DLG. Other than selecting the Single document application type, this application should be created with AppWizards defaults. The next section shows you how to create a simple dialog that has an editable text field and make it part of the DLG application by connecting it to a new menu item, Dialog, under the View menu. The dialog, as displayed by DLG, is shown in Figure 15.1.

FIGURE 15.1.
A simple dialog.

Dialogs and Property Sheets CHAPTER 15

275

Adding a Dialog Template


The first step in constructing a dialog is to create the dialog template resource. This resource can be built using the integrated dialog editor that is part of the Developer Studio. Figure 15.2 shows the dialog under construction. The OK and Cancel buttons are supplied by the dialog editor when a blank dialog is created; to that, we should add a static control and an edit control. Rename the edit controls identifier to IDC_EDIT; rename the identifier of the dialog itself to IDD_DIALOG.

FIGURE 15.2.
Constructing a simple dialog.

While the dialog template is open for editing in the dialog editor, you can directly invoke the ClassWizard to construct the dialog class corresponding to this template.

Constructing the Dialog Class


Although it is possible to create a dialog class by hand, in many cases it is easier to rely on the ClassWizard for this purpose. To create a dialog class corresponding to the dialog shown in Figure 15.2, use the right mouse button anywhere in the dialog editor window to bring up a pop-up menu; from this pop-up menu, select the ClassWizard command. The ClassWizard, after detecting that it has been invoked for a newly constructed resource, presents the Adding a Class dialog that is shown in Figure 15.3. Select the Create a new class radio button and click OK. At this time, the ClassWizard displays the New Class dialog (see Figure 15.4). Here, you can enter the dialogs name and set other options, such as the filename, the resource identifier, or Automation settings. You can also add this class to the Component Gallery for later reuse in other applications. Add a suitable name for the new class, for example, CMyDialog. Should you change the filenames that the ClassWizard suggests for your new classs header and implementation files? Should you use a separate header and implementation file for every new dialog you create? This is an interesting question. At the surface, the answer would appear to

15
DIALOGS AND PROPERTY SHEETS

276

Microsoft Foundation Classes PART III

be a yes; then again, even the AppWizard itself violates this rule when it places both the declaration and implementation of your applications About dialog into the application objects implementation file. Thus, I believe that in the end, it is best left to the judgment of the programmer. I often grouped dialog classes together if they were small, simple, and related. Leaving them in separate files tended to clutter the application workspace. For now, leave the filenames at the ClassWizard-generated defaults: MyDialog.h and MyDialog.cpp. Clicking on the OK button actually creates the new class and leaves the ClassWizard main dialog open.

FIGURE 15.3.
The Adding a Class dialog.

FIGURE 15.4.
The New Class dialog.

Dialogs and Property Sheets CHAPTER 15

277

The next step is to add a member variable that corresponds to the edit field in the dialog template.

Adding Member Variables


To add a new member variable, select the Member Variables tab in ClassWizard (see Figure 15.5). The member variable for the IDC_EDIT control can be added by double-clicking this identifier in the Control IDs column. This invokes yet another dialog, shown in Figure 15.6. Type in the new variables name (m_sEdit) and click on the OK button. After the member variable has been added, you can dismiss the ClassWizard altogether by clicking on the OK button in the ClassWizard dialog.

FIGURE 15.5.
Member variables in ClassWizard.

FIGURE 15.6.
The Add Member Variable dialog.

15
DIALOGS AND PROPERTY SHEETS

278

Microsoft Foundation Classes PART III

If you still have the dialog template resource open for editing, dismiss that window as well. In a moment, well begin creating the code that will invoke our new dialog. Before we do that, however, take a look at the code that the ClassWizard has generated for us so far.

ClassWizard Results
The declaration of CMyDialog (in MyDialog.h) is shown in Listing 15.1. Part of the class declaration is the declaration of IDD, which identifies the dialog template. The class declaration also contains the member variable m_sEdit, which we created through ClassWizard.

Listing 15.1. CMyDialog class declaration.


class CMyDialog : public CDialog { // Construction public: CMyDialog(CWnd* pParent = NULL); // standard constructor // Dialog Data //{{AFX_DATA(CMyDialog) enum { IDD = IDD_DIALOG }; CString m_sEdit; //}}AFX_DATA // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CMyDialog) protected: virtual void DoDataExchange(CDataExchange* pDX); //}}AFX_VIRTUAL // Implementation protected: // Generated message map functions //{{AFX_MSG(CMyDialog) // NOTE: the ClassWizard will add member functions here //}}AFX_MSG DECLARE_MESSAGE_MAP() };

Declarations for the constructor function and an override for the function are also provided here.

DoDataExchange

member

These two functions are defined in MyDialog.cpp (see Listing 15.2). Notice that the ClassWizard inserted code into both of them; the member variable m_sEdit is initialized in the constructor and also referred to in DoDataExchange.

Listing 15.2. CMyDialog member functions.


CMyDialog::CMyDialog(CWnd* pParent /*=NULL*/) : CDialog(CMyDialog::IDD, pParent) { //{{AFX_DATA_INIT(CMyDialog)

Dialogs and Property Sheets CHAPTER 15


m_sEdit = _T(); //}}AFX_DATA_INIT } void CMyDialog::DoDataExchange(CDataExchange* pDX) { CDialog::DoDataExchange(pDX); //{{AFX_DATA_MAP(CMyDialog) DDX_Text(pDX, IDC_EDIT, m_sEdit); //}}AFX_DATA_MAP } BEGIN_MESSAGE_MAP(CMyDialog, CDialog) //{{AFX_MSG_MAP(CMyDialog) // NOTE: the ClassWizard will add message map macros here //}}AFX_MSG_MAP END_MESSAGE_MAP()

279

is the function that facilitates data exchange between member variables and dialog controls. It is invoked both when the dialog is constructed and when it is dismissed. The macros inserted by ClassWizard (such as the DDX_Text macro) facilitate data exchange in both directions; the direction is determined by the m_bSaveAndValidate member of the CDataExchange object, pointed to by the pDX parameter. We will revisit this function and the various data exchange helper macros shortly.
DoDataExchange

Invoking the Dialog


Construction of our dialog object is now complete. How are we going to invoke this dialog from our DLG application? First, we must make a design decision, if it can be dignified with that phrase: The new dialog will be invoked when the user selects a new menu item, Dialog, from the View menu. This menu item must first be added to the applications menu using the resource editor. (See Figure 15.7.)

FIGURE 15.7.
Adding the View Dialog menu item.

15
DIALOGS AND PROPERTY SHEETS

To add code that handles the new menu item, invoke the ClassWizard, and add a command handler function for ID_VIEW_DIALOG to the CMainFrame class. (Why CMainFrame? Displaying

280

Microsoft Foundation Classes PART III

this dialog has nothing to do with a specific document or any of its views, so CMainFrame appears to be the most logical choice.) This is accomplished most easily by right-clicking on the new Dialog menu item to invoke the ClassWizard, selecting ClassWizards Message Maps tab, selecting the ID_VIEW_DIALOG identifier, selecting the COMMAND item in the Messages list box, and using the Add Function button.

NOTE
The identifier ID_VIEW_DIALOG was generated automatically by ClassWizard from the names of the menu item (Dialog) and the menu (View) that it was added to. If you used different words to label this menu or menu item, the identifier will also be different. A French programmer, for instance, may use the words Affichage and Dialogue; in this case, the new menu item will automatically be labeled ID_AFFICHAGE_DIALOGUE. Any functions added by ClassWizard will also have names generated in a similar fashion.

The implementation of CMainFrame::OnViewDialog is shown in Listing 15.3. After constructing the dialog object, we assign an initial value to the member variable m_sEdit. We can access this variable directly, because it is a public member of the CMyDialog class. Were it private or protected, we would need to access it, for instance, through a pair of CMyDialog member functions (Get/Set methods). Next, we invoke the dialog via the DoModal function. After the dialog is dismissed by the user, we examine the new value of m_sEdit by simply displaying it in a message box.

Listing 15.3. The CMainFrame::OnViewDialog member function.


void CMainFrame::OnViewDialog() { // TODO: Add your command handler code here CMyDialog myDialog; myDialog.m_sEdit = Default string; if (myDialog.DoModal() == IDOK) MessageBox(myDialog.m_sEdit); }

Note that in order to be able to declare an object of type CMyDialog in CMainFrame::OnViewDialog, it is necessary to include the MyDialog.h header file in MainFrm.cpp. Thats it. The application is ready to be recompiled and run.

Modeless Dialogs
Invoking a dialog through the DoModal member function invokes the dialog as a modal dialog. However, sometimes applications require the use of modeless dialogs. The steps of creating and displaying a modeless dialog are different from the steps taken for modal dialogs.

Dialogs and Property Sheets CHAPTER 15

281

To convert our dialog in DLG to a modeless dialog, we must first modify the dialogs constructor function. In the constructor, we must make a call to the Create member function in order to construct the dialog box object. We must also call a different version of the base class constructor, as shown in Listing 15.4.

Listing 15.4. Modeless version of CMyDialog::CMyDialog.


CMyDialog::CMyDialog(CWnd* pParent /*=NULL*/) : CDialog() { Create(CMyDialog::IDD, pParent); //{{AFX_DATA_INIT(CMyDialog) m_sEdit = _T(); //}}AFX_DATA_INIT }

Invocation of the dialog from CMainFrame::OnViewDialog is also different. Instead of calling the dialogs DoModal member function, we just construct the dialog object; the call to Create within the constructor takes care of the rest. Note that we can no longer construct the dialog box on the stack. Because a modeless dialog box is long lived and continues to exist even after CMainFrame::OnViewDialog returns, we have to allocate the CDialog object differently. This new version of CMainFrame::OnViewDialog is shown in Listing 15.5 (MainFrm.cpp).

Listing 15.5. Constructing a modeless dialog in CMainFrame::OnViewDialog.


void CMainFrame::OnViewDialog() { // TODO: Add your command handler code here CMyDialog *pMyDialog; pMyDialog = new CMyDialog; pMyDialog->m_sEdit = Default string; pMyDialog->UpdateData(FALSE); pMyDialog->ShowWindow(SW_SHOW); }

Why was it necessary to call UpdateData in this function? Because we set the value of m_sEdit after the dialog box object has been constructed and initial Dialog Data Exchange took place. By calling UpdateData, we ensure that the controls in the dialog box object are updated to reflect the settings in the member variables of the CDialog object. This is yet another example that should remind us that the C++ object and the Windows object are two different entities. We must also call the ShowWindow member function to make the new dialog visible. Alternatively, we could have created the dialog box template resource with the WS_VISIBLE style. How long will this dialog exist? As long as the user does not dismiss it by clicking on the OK or Cancel button. At that time, the default implementations of CDialog::OnOK and

15
DIALOGS AND PROPERTY SHEETS

282

Microsoft Foundation Classes PART III


CDialog::OnCancel hide the dialog box but do not destroy it. Obviously, we must override these DestroyWindow

functions to properly destroy the dialog. In both of these functions, a call must be made to the member function.

We must also override the dialogs OnOK function to ensure that we process whatever the user entered in the dialog. We can no longer rely on the function calling DoModal for this purpose, for the simple reason that DoModal is never called. Calling the DestroyWindow member function from OnOK and OnCancel ensures that the Windows dialog box object is destroyed; but how will the C++ object be destroyed? The answer to that question is yet another override. We must override the PostNcDestroy member function and delete the CDialog-derived object from within it. To override the default implementations of OnOK, OnCancel, and PostNCDestroy, you must first add these functions to the CMyDialog class through ClassWizard. OnOK and OnCancel are created as event handlers forthe controls IDOK and IDCANCEL; PostNoDestroy is simply selected from the Messages list for CMyDialog. Implementations of CMyDialog::OnOK, CMyDialog::OnCancel, and are shown in Listing 15.6 (MyDialog.cpp).
CMyDialog::PostNcDestroy

Listing 15.6. Member functions in the modeless version of CMyDialog.


void CMyDialog::OnCancel() { // TODO: Add extra cleanup here CDialog::OnCancel(); DestroyWindow(); } void CMyDialog::OnOK() { // TODO: Add extra validation here MessageBox(m_sEdit); CDialog::OnOK(); DestroyWindow(); } void CMyDialog::PostNcDestroy() { // TODO: Add your specialized code here and/or call the base class CDialog::PostNcDestroy(); delete this; }

If your modeless dialog must notify the frame, document, or view, it can do so by calling a member function. The dialog class can have a member variable that stores a pointer to the frame, document, or view object from within which the dialog has been created. Other mechanisms for communication between the modeless dialog and other parts of your application are also conceivable; for example, the dialog may post a message to other windows of the application.

Dialogs and Property Sheets CHAPTER 15

283

More on Dialog Data Exchange


In the preceding example, we have used Dialog Data Exchange to map the contents of an edit control to the contents of a CString member variable in the dialog class. The Dialog Data Exchange mechanism offers many other capabilities for mapping simple variables or control objects to controls in a dialog box.

NOTE
Although Dialog Data Exchange and Dialog Data Validation are described in the context of dialog boxes, they are not limited in use to dialog boxes only. The member functions discussed, such as DoDataExchange and UpdateData, are actually member functions of CWnd, not CDialog. Dialog Data Exchange is also used outside the context of a dialog box; CFormView and classes derived from it represent one example.

Dialog Data Exchange


Dialog Data Exchange takes place in the dialog classs DoDataExchange member function. In this function, calls are made for all member variables that are mapped to controls. The calls that are made are to a family of MFC functions with names that begin with DDX_. These functions are responsible for performing the actual data exchange. For example, to perform data exchange between an edit control and a member variable of type CString, the following call is made:
DDX_Text(pDX, IDC_EDIT, m_sEdit);

Dialog Data Validation


In addition to the simple exchange of data between member variables and dialog controls, MFC also offers a data validation mechanism. Data validation is accomplished through calls to functions with names that begin with DDV_. These functions perform the necessary validation and if a validation error is encountered, display a standard error message box and raise an exception of type CUserException. They also call the Fail member function of the CDataExchange object that is passed to DoDataExchange; this object, in turn, sets the focus to the offending control. An example for a data validation function is DDV_MaxChars, which is used to validate the length of a string typed into an edit control. To validate that a string in an edit control is no longer than 100 characters, you would make the following call:
DDV_MaxChars(pDX, m_sEdit, 100);

15
DIALOGS AND PROPERTY SHEETS

Data validation calls for a given control must immediately follow the data exchange function call for the same control.

284

Microsoft Foundation Classes PART III

Using Simple Types


Dialog Data Exchange with simple types is supported for edit controls, scrollbars, check boxes, radio buttons, list boxes, and combo boxes. Table 15.1 summarizes the various types supported by Dialog Data Exchange for edit controls.

Table 15.1. Dialog Data Exchange and validation for edit controls. Control Data type DDX function
Edit control Edit control Edit control Edit control Edit control Edit control Edit control Edit control Edit control Edit control Edit control Check box Radio button List box List box List box Combo box Combo box Combo box Scrollbar
BYTE short int UINT long DWORD float double CString COleDateTime COleCurrency BOOL int int CString Cstring int CString Cstring int DDX_Text DDX_Text DDX_Text DDX_Text DDX_Text DDX_Text DDX_Text DDX_Text DDX_Text DDX_Text DDX_Text DDX_Check DDX_Radio DDX_LBIndex DDX_LBString DDX_LBStringExact DDX_CBIndex DDX_CBString DDX_CBStringExact DDX_Scroll

DDV function
DDV_MinMaxByte DDV_MinMaxInt DDV_MinMaxInt DDV_MinMaxUnsigned DDV_MinMaxLong DDV_MinMaxDWord DDV_MinMaxFloat DDV_MinMaxDouble DDV_MaxChars

DDV_MaxChars

The MFC Library provides additional versions of the DDX functions to facilitate data exchange between a dialog box and records in a database. These functions have names that begin with DDX_Field; for example, the database variant of DDX_Text would be named DDX_FieldText.

Dialogs and Property Sheets CHAPTER 15

285

Using Control Data Types


In addition to assigning a member variable to a control representing the controls value, it is also possible to assign member variables that represent the control object itself. For example, it is possible to assign a variable of type CEdit to an edit control. The Dialog Data Exchange mechanism uses the DDX_Control function to exchange data between a dialog control and a CWnd-derived control object. A control object can be used concurrently with a member variable representing the controls value. For example, it is possible to assign both a CString object representing the controls value and a CEdit object representing the control itself to an edit control in a dialog. Why would you use a control object? Through such an object, you can implement much greater control over the appearance and behavior of dialog controls. For example, as control objects are CWnd-derived, your application can use CWnd member functions to change the controls size and position. Through the control object, it is also possible to send messages to the control. They can also be used to manipulate behavior specific to certain control types; for instance, you can use CEdit::SetReadOnly to change the read-only state of an edit control. In the case of many control types (including the new common controls), you must use a control object for Dialog Data Exchange. The use of a simple data type is meaningless and not supported.

Implementing Custom Data Types


Versatile as the Dialog Data Exchange mechanism is, it would not be sufficient in many situations were it not for the capability to extend it for custom data types. Fortunately, the ClassWizard offers the capability to handle custom DDX and DDV routines. The steps required to implement custom DDX/DDV support are time-consuming and may only be beneficial for data types that you frequently reuse. That said, it is possible to add custom DDX/DDV support to a specific project, or to all projects, by modifying either your projects CLW file, or the ddx.clw file in your msdev\bin subdirectory. The exact steps to be taken for custom DDX/DDV support are described in MFC Technical Note 26 that is part of your Visual C++ online documentation.

Dialogs and Message Handling


CDialog-derived

15
DIALOGS AND PROPERTY SHEETS

objects are, as you might expect from CWnd-derived objects, capable of handling messages. In fact, in all but the simplest cases, it is necessary to add message handlers to your CDialog-derived dialog class.

286

Microsoft Foundation Classes PART III

Message handling in a dialog is no different from message handling in a view or frame window. Message-handler functions can be easily added to the dialog classs message map using ClassWizard. In the earlier examples we have already done that when we added override versions of the OnOK and OnCancel member functions. These member functions are handlers of WM_COMMAND messages. (The third override function we implemented, PostNcDestroy, is not a message handler; however, it is called from within the handler for WM_NCDESTROY messages, OnNcDestroy.) The most frequently used message handlers in a dialog class correspond to messages sent to the dialog window by one of its controls. These include BN_CLICKED messages sent by a button; the variety of CBN_ messages sent by a combo box; EN_ messages sent by an edit control; and so on. Another set of messages that dialog classes often handle consists of WM_DRAWITEM and WM_MEASUREITEM for owner-draw controls. Owner-draw controls bring up an interesting issue. Should you handle such a situation from within your dialog class, or should you assign an object of a class derived from a control class to the control and handle it there? For example, if you have an owner-draw button, you have the choice of adding a handler for WM_DRAWITEM messages to your dialog class, or deriving a class from CButton and overriding its DrawItem member function. I suppose there is no definite answer to this question. Perhaps the best rule of thumb is that you should derive your own control class if you expect the control to be reused in many dialogs; otherwise, handling WM_DRAWITEM within the dialog class may be sufficient (and also simpler).

Property Sheets
Property sheets are several overlapping dialogs in one. The user selects one of the dialogs, or property pages, by clicking on the corresponding tab in a tab control. MFC supports property sheets through two classes: CPropertySheet and CPropertyPage. CPropertySheet corresponds to the property sheet; CPropertyPage-derived classes correspond to the individual property pages within the property sheet. Using a property sheet requires several steps. First, the property pages must be constructed; next, the property sheet must be created. The following simple example reviews this process. A new application, PRP, is used to display a property sheet, as shown in Figure 15.8. Like our earlier application, DLG, PRP is also a standard SDI application created by AppWizard.

Dialogs and Property Sheets CHAPTER 15

287

FIGURE 15.8.
A sample property sheet.

Constructing Property Pages


Constructing a property page is very similar to constructing dialogs. To create an example, you can again start with an AppWizard-generated SDI project. The first step is to construct the dialog template resource for every property page that you wish to add to the property sheet. There are a few special considerations when constructing a dialog template resource for a property page object: 1. The dialogs caption should be set to the text that you wish to see appear in the tab corresponding to the dialog. 2. The dialogs style should be set to Child. 3. The dialogs border style should be set to Thin. 4. The Titlebar style should be checked. 5. The Disabled style should be checked. Although the property pages in a property sheet will overlap, it is not necessary to create them with the same size. The MFC Library will automatically adjust the size of property pages to match the size of the largest property page. In this example we construct two property pages for our applicationnothing fancy, just a simple text field in both of them. The first property page, titled Page 1, is shown in Figure 15.9. To insert a blank property page template similar to the one shown here, use the IDR_PROPPAGE_SMALL subtype of the Dialog resource type in the Insert Resource dialog (which, in turn, is invoked by right-clicking on the Dialog item in ResourceView, and selecting the Insert command). Afterwards, you can add the controls as shown. The identifier of the dialog template resource should be set to IDD_PAGE1; the identifier of the edit control should be set to IDC_EDIT1. Make sure that the dialog templates properties are set correctly. To set the dialogs caption, right-click on the dialog and select the Properties menu command to invoke the Dialog Properties property sheet (see Figure 15.10).

15
DIALOGS AND PROPERTY SHEETS

288

Microsoft Foundation Classes PART III

FIGURE 15.9.
Constructing a property page.

FIGURE 15.10.
Property page dialog resource caption setting.

To set the style, border style, and titlebar settings, select the Styles tab in the property sheet of the dialog resource (see Figure 15.11). To set the Disabled style of the dialog resource, use the More Styles tab in the dialog resource property sheet (see Figure 15.12).

FIGURE 15.11.
Property page dialog resource styles.

FIGURE 15.12.
Setting the property page dialog resource to Disabled.

Dialogs and Property Sheets CHAPTER 15

289

The second property page in our simple example is, for the sake of simplicity, nearly identical to the first. In fact, you can create the dialog resource for this second property page by simply making a copy of the first. Make sure that the identifier of the new dialog resource is set to IDD_PAGE2 and that the identifier of the edit control within it is IDC_EDIT2. (It would be perfectly legal to use the same identifier for controls in separate property pages; they act and behave like separate dialogs. Nevertheless, I prefer to use distinct identifiers; this helps reduce the possibility for errors.) Make sure also that you have changed the dialogs title and the text of the static control. Once both property page dialog resources have been constructed, it is time to invoke the ClassWizard and construct classes that correspond to these property pages. To do so, invoke the ClassWizard while the focus is on the first property page dialog resource and while it is open for editing. As with other dialogs, the ClassWizard will recognize that the dialog template has no corresponding class and offer you the opportunity to create a new class. In the Create New Class dialog, specify a name for the class corresponding to the dialog template (for example, CMyPage1). More importantly, make sure that this new class is based on CPropertyPage (and not the default CDialog). After the correct settings have been entered, create the class. While in ClassWizard, you should also add a member variable that corresponds to the edit control in the property page. Name this variable m_sEdit1. These steps should be repeated for the second property page. The class for this property page should be named CMyPage2, and the member variable corresponding to its edit control should be named m_sEdit2. Construction of our property pages is now complete. Take a brief look at the code generated by ClassWizard. The declaration of CMyPage1 is shown in Listing 15.7. (The declaration of CMyPage2 is virtually identical.)

Listing 15.7. CMyPage1 declaration.


class CMyPage1 : public CPropertyPage { DECLARE_DYNCREATE(CMyPage1) // Construction public: CMyPage1(); ~CMyPage1(); // Dialog Data //{{AFX_DATA(CMyPage1) enum { IDD = IDD_PAGE1 }; CString m_sEdit1; //}}AFX_DATA // Overrides // ClassWizard generate virtual function overrides //{{AFX_VIRTUAL(CMyPage1) protected:

15
DIALOGS AND PROPERTY SHEETS

continues

290

Microsoft Foundation Classes PART III

Listing 15.7. continued


virtual void DoDataExchange(CDataExchange* pDX); //}}AFX_VIRTUAL // Implementation protected: // Generated message map functions //{{AFX_MSG(CMyPage1) // NOTE: the ClassWizard will add member functions here //}}AFX_MSG DECLARE_MESSAGE_MAP()

As you can see, there is very little difference between this declaration and the ClassWizardgenerated declaration of a CDialog-derived dialog class. Most importantly, CPropertyPagederived classes can use Dialog Data Exchange functions just as classes derived from Cdialog can. The implementation of CMyPage1 member functions (see Listing 15.8) is also no different from the implementation of similar functions in a CDialog-derived class. Perhaps the one notable difference is that this class has been declared as dynamically creatable with the help of the DECLARE_DYNCREATE and IMPLEMENT_DYNCREATE macros.

Listing 15.8. CMyPage1 implementation.


IMPLEMENT_DYNCREATE(CMyPage1, CPropertyPage) CMyPage1::CMyPage1() : CPropertyPage(CMyPage1::IDD) { //{{AFX_DATA_INIT(CMyPage1) m_sEdit1 = _T(); //}}AFX_DATA_INIT } CMyPage1::~CMyPage1() { } void CMyPage1::DoDataExchange(CDataExchange* pDX) { CPropertyPage::DoDataExchange(pDX); //{{AFX_DATA_MAP(CMyPage1) DDX_Text(pDX, IDC_EDIT1, m_sEdit1); //}}AFX_DATA_MAP } BEGIN_MESSAGE_MAP(CMyPage1, CPropertyPage) //{{AFX_MSG_MAP(CMyPage1) // NOTE: the ClassWizard will add message map macros here //}}AFX_MSG_MAP END_MESSAGE_MAP()

As its declaration, the implementation of CMyPage2 is virtually identical to that of CMyPage1.

Dialogs and Property Sheets CHAPTER 15

291

Adding a Property Sheet Object


Now that the property pages have been constructed, the one remaining task is to create the property sheet. Again, we need to invoke the new property sheet when the user selects a new menu command, Property Sheet, from the applications View menu. Add this command to the menu using the resource editor, and invoke the ClassWizard to add a corresponding member function, CMainFrame::OnViewPropertysheet, to the CMainFrame class. In this member function, a series of tasks must be performed. First, a property sheet object must be constructed. Next, the property pages must be added to it using the AddPage member function; and finally, the property sheet must be invoked using the DoModal member function. Listing 15.9 contains the implementation of CMainFrame::OnViewPropertysheet that performs all these tasks.

Listing 15.9. The CMainFrame::OnViewPropertysheet function.


void CMainFrame::OnViewPropertysheet() { // TODO: Add your command handler code here CPropertySheet myPropSheet; CMyPage1 myPage1; CMyPage2 myPage2; myPage1.m_sEdit1 = First; myPage2.m_sEdit2 = Second; myPropSheet.AddPage(&myPage1); myPropSheet.AddPage(&myPage2); myPropSheet.DoModal(); }

Do not forget to include the header files MyPage1.h and MyPage2.h in MainFrm.cpp; otherwise, you will not be able to declare objects of type CMyPage1 or CMyPage2 and the function in Listing 15.9 will not compile. At this time, the application is ready to be compiled and run. Although in this example we made no use of the property page member variables after the property sheet was dismissed, we could access them simply through the property page objects myPage1 and myPage2.

CPropertyPage Member Functions


Our simple example did not utilize many of the advanced capabilities of the CPropertyPage class. For example, in a more realistic application, you may wish to override the CancelToClose member function whenever a change is made to a property page. This member function changes the

15
DIALOGS AND PROPERTY SHEETS

292

Microsoft Foundation Classes PART III

OK button to Close and disables the Cancel button in the property sheet. This function is best used after an irreversible change has been made in a property page. Another frequently used property page function is the SetModified function. This function can be used to enable the Apply Now button in the property sheet. Other property page overridables include OnOK (called when the OK, Apply Now, or Close button is clicked in the property sheet), OnCancel (called when the Cancel button is clicked in the property sheet), and OnApply (called when the Apply Now button is clicked in the property sheet). Property sheets can also be used to implement wizard-like behavior; that is, behavior similar to the behavior of the ubiquitous wizards that can be found in many Microsoft applications. Wizard mode can be enabled by calling the SetWizardMode member function of the property sheet; in the property pages, override the OnWizardBack, OnWizardNext, and OnWizardFinish member functions.

Modeless Property Sheets


Using the DoModal member function of a property sheet implies modal behavior. As is the case with dialogs, it is also possible to implement a modeless property sheet. To accomplish this, it is first of all necessary to derive our own property sheet class. This is important because at the very least, we must override its PostNcDestroy member function to ensure that objects of this class are destroyed when the modeless property sheet is dismissed. The new property sheet class can be created using ClassWizard. Create a new class derived from CPropertySheet, and name it CMySheet. While in ClassWizard, add the PostNcDestroy member function. The declaration of CMySheet (in the file MySheet.h), as generated by ClassWizard, is shown in Listing 15.10.

Listing 15.10. CMySheet declaration.


class CMySheet : public CPropertySheet { DECLARE_DYNAMIC(CMySheet) // Construction public: CMySheet(UINT nIDCaption, CWnd* pParentWnd = NULL, UINT iSelectPage = 0); CMySheet(LPCTSTR pszCaption, CWnd* pParentWnd = NULL, UINT iSelectPage = 0); // Attributes public: // Operations public: // Overrides // ClassWizard generated virtual function overrides

Dialogs and Property Sheets CHAPTER 15


//{{AFX_VIRTUAL(CMySheet) protected: virtual void PostNcDestroy(); //}}AFX_VIRTUAL // Implementation public: virtual ~CMySheet(); // Generated message map functions protected: //{{AFX_MSG(CMySheet) // NOTE - the ClassWizard will add and remove member // functions here. //}}AFX_MSG DECLARE_MESSAGE_MAP() };

293

In the implementation file, MySheet.cpp, it is necessary to modify the PostNcDestroy member function to destroy not only the property sheet object, but also any property pages associated with it. The implementation of this function, together with other, ClassWizard-supplied member function implementations for the CMySheet class, is shown in Listing 15.11.

Listing 15.11. CMySheet implementation.


/////////////////////////////////////////////////////////////////// // CMySheet IMPLEMENT_DYNAMIC(CMySheet, CPropertySheet) CMySheet::CMySheet(UINT nIDCaption, CWnd* pParentWnd, UINT iSelectPage) :CPropertySheet(nIDCaption, pParentWnd, iSelectPage) { } CMySheet::CMySheet(LPCTSTR pszCaption, CWnd* pParentWnd, UINT iSelectPage) :CPropertySheet(pszCaption, pParentWnd, iSelectPage) { } CMySheet::~CMySheet() { } BEGIN_MESSAGE_MAP(CMySheet, CPropertySheet) //{{AFX_MSG_MAP(CMySheet) // NOTE - the ClassWizard will add and remove mapping macros here. //}}AFX_MSG_MAP END_MESSAGE_MAP() /////////////////////////////////////////////////////////////////// // CMySheet message handlers void CMySheet::PostNcDestroy() { // TODO: Add your specialized code here and/or call the base class CPropertySheet::PostNcDestroy(); for (int i = 0; i < GetPageCount(); i++) delete GetPage(i); delete this; }

15
DIALOGS AND PROPERTY SHEETS

294

Microsoft Foundation Classes PART III

A modeless property sheet does not have OK, Cancel, and Apply Now buttons by default. If any buttons are required, these must be added by hand. We are not going to worry about these now; the modeless property sheet can still be dismissed by closing it through its control menu. How is the modeless property sheet invoked? Obviously, we have to modify the OnViewPropertysheet member function in our CMainFrame class, as using DoModal is no longer appropriate. Nor is it appropriate to create the property sheet or any of its property pages on the stack, as we do not want them destroyed when the OnViewPropertysheet function returns. The new OnViewPropertysheet function is shown in Listing 15.12.

Listing 15.12. Invoking a modeless property sheet.


void CMainFrame::OnViewPropertysheet() { // TODO: Add your command handler code here CMySheet *pMyPropSheet; CMyPage1 *pMyPage1; CMyPage2 *pMyPage2; pMyPropSheet = new CMySheet(); pMyPage1 = new CMyPage1; pMyPage2 = new CMyPage2; pMyPage1->m_sEdit1 = First; pMyPage2->m_sEdit2 = Second; pMyPropSheet->AddPage(pMyPage1); pMyPropSheet->AddPage(pMyPage2); pMyPropSheet->Create(this); }

In order for CMainFrame::OnViewPropertysheet to compile in its new form, it is necessary to add the include file MySheet.h to MainFrm.cpp; otherwise, the attempt to declare an object of type CMySheet will fail. The application is now ready to be recompiled and run.

Summary
In MFC, dialogs are represented by classes derived from CDialog. The steps of constructing a dialog that is part of an MFC application are as follows: 1. 2. 3. 4. 5. Create the dialog template resource. Invoke ClassWizard and create the dialog class corresponding to the resource. Through ClassWizard, add member variables corresponding to controls. Still using ClassWizard, add message handlers if necessary. Add code to your application that constructs a dialog object, invokes it (through the DoModal member function), and retrieves results.

Dialogs and Property Sheets CHAPTER 15

295

MFC applications can also have modeless dialogs. These dialogs are constructed differently. The constructor function in your dialog class should call the Create member function; it should also call the modeless version of the constructor of the CDialog base class. The modeless dialog must also explicitly be made visible by calling the ShowWindow member function. Classes that correspond to modeless dialogs should override the OnOK and OnCancel member functions and call the DestroyWindow member function from within them. They should also override PostNcDestroy and destroy the C++ object (using delete this, for example). Controls in a dialog are often represented by member variables in the corresponding dialog class. To facilitate the exchange of data between controls in the dialog box object and member variables in the dialog class, the Dialog Data Exchange mechanism can be used. This mechanism provides a simple method for matching member variables to controls. Member variables can be of simple value types or can represent control objects. It is possible to simultaneously use a member variable of a simple type to obtain the value of a control while using a control object to manage other aspects of the control. The Dialog Data Exchange mechanism also offers data validation capabilities. For frequently used nonstandard types, it is possible to extend the ClassWizards capability to handle Dialog Data Exchange. New data exchange and validation routines can be added either on a per project basis or to your overall Visual C++ configuration. Property sheets represent several overlapping dialogs, or property pages, which the user can choose by clicking on corresponding tabs in a tab control. Creating a property sheet is a two-phase process. First, property pages must be created; second, a property sheet object must be constructed, the property pages must be added to it, and the property sheet must be displayed. Construction of property pages involves the same steps as construction of a dialog: 1. Create the dialog template resource for every property page; ensure that the resources have the Child style, Thin border style, Titlebar style, and Disabled style, and that their caption is set to the text that is desired in the corresponding tab. 2. Invoke ClassWizard and create a class derived from CPropertyPage corresponding to every dialog template resource. 3. Through ClassWizard, add member variables corresponding to controls in each property page. 4. Still using ClassWizard, add message handlers if necessary. After the property pages have been constructed, you can proceed with the second phase: 1. Construct a CPropertySheet object or an object of a class derived from
CPropertySheet.

15
DIALOGS AND PROPERTY SHEETS

2. Construct a property page object for every property page you wish to add to the property sheet.

296

Microsoft Foundation Classes PART III

3. Add the property pages to the property sheet by calling AddPage. 4. Invoke the property sheet by calling DoModal. It is also possible to create modeless property sheets. To implement modeless property sheets, it is necessary to derive a class from CPropertySheet and override its PostNcDestroy member function to delete not only the property sheet object, but also all of its property pages. The modeless property sheet should be invoked via the Create member function instead of DoModal.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

297

16

MFC Support for Common Dialogs and Common Controls


IN THIS CHAPTER
s Common Dialogs 298 s Common Controls 305

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

298

Microsoft Foundation Classes PART III

The Microsoft Foundation Classes Library provides extensive support for common dialogs and common controls. Common dialogs have been a feature of Microsoft Windows since version 3.1. They are used for opening and closing files, selecting colors and fonts, performing text searches, and setting up and using the printer. Use of these dialogs has never been particularly difficult; unless extensive customization was required, it usually involved populating the members of a C structure, invoking the common dialog, and processing the users selection. The MFC Library makes it even easier by automating the construction of the necessary structures. The common dialog classes of MFC also make customization easier. Windows also provides support for an ever growing selection of standard controls, such as edit controls, static controls, or buttons. New controls were introduced with Windows 95 and an even newer, more powerful set has become available with Internet Explorer 4. MFC 4.2 now provides complete class support for most of these new controls and integrates them with other MFC framework features where applicable.

Common Dialogs
MFC support for common dialogs is provided through classes derived from the class CCommonDialog, which is itself derived from CDialog . You would never derive a class from CCommonDialog directly or create a CCommonDialog object. Instead, to use a common dialog you create an object of the appropriate dialog class; or, if you use a customized common dialog template, you derive a class from that dialog class. The set of classes in MFC that provide common dialog support is shown in Figure 16.1.

FIGURE 16.1.
Common dialog support in MFC.

CCommonDialog

CColorDialog

CFileDialog

CFindReplaceDialog

CFontDialog

COleDialog

CPageSetupDialog

CPrintDialog

MFC Support for Common Dialogs and Common Controls CHAPTER 16

299

In case of an error, many common dialogs provide extended error information. This information can be retrieved by calling CommDlgExtendedError. In the following sections, we take a brief look at each of the common dialog classes provided by the MFC Library. In order to do this, we are going to use an AppWizard-generated application skeleton. This application, CDLG, can be created as an SDI application with AppWizard defaults. We use its View menu for additional commands that invoke the various dialogs. However, to avoid being repetitive, I do not mention separately the creation of a menu command in CDLGs View menu for every common dialog.

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

CColorDialog
The CColorDialog class supports the Windows color selection common dialog. To display a color dialog, construct a CColorDialog object and call its DoModal function. Before calling DoModal, you can initialize the dialog by setting up the m_cc member of CColorDialog, which is a Win32 structure of type CHOOSECOLOR. After the dialog has been dismissed, the users color selection can be retrieved by calling the GetColor member function. Listing 16.1 shows a possible implementation of a color selection dialog using the CColorDialog class.

Listing 16.1. Use of CColorDialog in CDLG.


void CMainFrame::OnViewColordialog() { // TODO: Add your command handler code here CColorDialog dlg; if (dlg.DoModal() == IDOK) { TCHAR temp[100]; wsprintf(temp, _T(Color selected: %#8.8X), dlg.GetColor()); AfxMessageBox(temp); } }

You can also customize the color selection dialog. To do so, derive your own class from CColorDialog and supply a custom dialog template. The Windows color selection dialog template can be found in the file color.dlg in your Program Files\DevStudio\VC\include directory. If you add any new controls, you might want to add a message map to your new class to process notification messages from the new controls.

CFileDialog
The CFileDialog class provides support for the Windows Open and Save As dialogs. To use a file dialog in an MFC application, you should first construct a CFileDialog object. The constructor takes several parameters; the first parameter is a mandatory Boolean value that determines whether the dialog is an Open or a Save As dialog. After the dialog has been constructed, call its DoModal member function to display it.

300

Microsoft Foundation Classes PART III

Most common initialization settings for this dialog can be specified in the CFileDialog constructor. You may also set the values of the m_ofn structure; this is a Win32 OPENFILENAME structure. When the dialog terminates, the users filename selection can be retrieved by calling the GetPathName member function. If the selection of multiple filenames was allowed, an iterator to the filename list can be obtained by calling GetStartPosition; individual filenames can be retrieved by repeatedly calling GetNextPathName with the iterator. Listing 16.2 demonstrates the use of the Open dialog.

Listing 16.2. Use of CFileDialog in CDLG.


void CMainFrame::OnViewFiledialog() { // TODO: Add your command handler code here CFileDialog dlg(TRUE); if (dlg.DoModal() == IDOK) { CString temp = _T(File selected: ) + dlg.GetPathName(); AfxMessageBox(temp); } }

If you derive a class from CFileDialog, you can provide customized handling for conditions such as sharing violation, filename validation, and list box change notifications. The file dialog can also be customized. To do so, derive a class from CFileDialog, add a custom dialog template, and add your message map to handle notifications from any new controls. The original file dialog template can be found in the file fileopen.dlg in your Program Files\DevStudio\VC\include directory.

CFindReplaceDialog
The CFindReplaceDialog class supports the use of the Windows Find and Replace dialog in MFC applications. Use of the Find and Replace dialog is fundamentally different from the user of other common dialogs. While other common dialogs are modal, the Find and Replace dialog is modeless and must be constructed and used accordingly. To use the Find and Replace dialog, first construct a CFindReplaceDialog object. This object cannot be constructed on the stack; it must be created using the new operator to ensure its persistence after the function in which it is created returns. The Find and Replace dialog communicates with its owner window using special messages. To enable use of these messages, use the ::RegisterWindowMessage function. The value obtained upon registering Find and Replace dialog messages can be used in a windows message map using the ON_REGISTERED_MESSAGE macro.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

301

The actual dialog is created by calling the Create member function. Make sure to specify the window that should receive messages from the dialog in the call to CFindReplaceDialog::Create. Now youre ready to put all this into practice. Listing 16.3 shows how the use of the Find and Replace dialog is implemented in the CDLG application. All the code shown here is part of the MainFrm.cpp source file.

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

Listing 16.3. Use of CFindReplaceDialog in CDLG.


// MainFrm.cpp : implementation of the CMainFrame class // ... static UINT WM_FINDREPLACE = RegisterWindowMessage(FINDMSGSTRING); /////////////////////////////////////////////////////////////////// // CMainFrame IMPLEMENT_DYNCREATE(CMainFrame, CFrameWnd) BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd) //{{AFX_MSG_MAP(CMainFrame) ... //}}AFX_MSG_MAP ON_REGISTERED_MESSAGE(WM_FINDREPLACE, OnFindReplace) END_MESSAGE_MAP() ... void CMainFrame::OnViewFindreplacedialog() { // TODO: Add your command handler code here CFindReplaceDialog *pDlg = new CFindReplaceDialog; pDlg->Create(TRUE, _T(Findme), NULL, FR_DOWN, this); } LRESULT CMainFrame::OnFindReplace(WPARAM wParam, LPARAM lParam) { if (((LPFINDREPLACE)lParam)->Flags & FR_FINDNEXT) { CString temp = _T(Search string: ); temp = temp + ((LPFINDREPLACE)lParam)->lpstrFindWhat; AfxMessageBox(temp); } return 0; }

The function OnFindReplace must also be declared in MainFrm.h. Toward the end of the class declaration, after the ClassWizard-generated message map, but before the DECLARE_MESSAGE_MAP macro call, insert the following line:
afx_msg LRESULT OnFindReplace(WPARAM wParam, LPARAM lParam);

Customization of the Find and Replace dialog involves deriving a class from CFindReplaceDialog, providing a custom dialog template, and adding your own message map to process notification messages from any extra controls you added. The original dialog template can be found in the file findtext.dlg in your Program Files\DevStudio\VC\include directory.

302

Microsoft Foundation Classes PART III

CFontDialog
The CFontDialog class supports font selection in MFC applications through the Windows font selection common dialog. To use the font selection dialog, create a CFontDialog object and call its DoModal member function. Before calling DoModal, you can initialize the dialog by setting the values of the m_cf member of CFontDialog. This member is a structure of type CHOOSEFONT. When the dialog returns, you can use CFontDialog member functions such as GetFaceName, GetSize, or GetColor to obtain information about the font selected by the user. Alternatively, you can examine the values of the m_cf structure. Of specific interest is m_cf.lpLogFont; upon return of CFontDialog::DoModal, this pointer points to a LOGFONT structure, which can be used in subsequent calls to ::CreateFontIndirect or CFont::CreateFontIndirect to create a logical font. The use of CFontDialog is demonstrated in Listing 16.4.

Listing 16.4. Use of CFontDialog in CDLG.


void CMainFrame::OnViewFontdialog() { // TODO: Add your command handler code here CFontDialog dlg; if (dlg.DoModal() == IDOK) { CString temp = _T(Font selected: ) + dlg.GetFaceName(); AfxMessageBox(temp); } }

To use a customized font selection dialog, derive your own class from CFontDialog and create a custom dialog template. Add a message map to the new class to process notification messages from any new controls you might have created. As the basis of the dialog template, use the font selection dialog template font.dlg that can be found in your Program Files\DevStudio\ VC\include directory.

CPageSetupDialog
The Page Setup dialog is used under Windows 95 or later and Windows NT 3.51 or later to set up the printed page. This dialog replaces the Print Setup dialog of earlier Windows versions. To use the Page Setup dialog, create an object of type CPageSetupDialog and call its DoModal member function. Information about the users page setup preferences can be obtained through a variety of member functions, such as GetDeviceName, GetMargins, or GetPaperSize. Alternatively, you can examine the m_psd member variable (which is of type PAGESETUPDLG). It is also possible to modify members of this structure prior to calling DoModal in order to override default behavior.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

303

Listing 16.5 shows an example of using the Page Setup dialog.

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

Listing 16.5. Use of CPageSetupDialog in CDLG.


void CMainFrame::OnViewPagesetupdialog() { // TODO: Add your command handler code here CPageSetupDialog dlg; if (dlg.DoModal() == IDOK) { char temp[100]; sprintf(temp, Paper size: %g\ x %g\., (double)dlg.m_psd.ptPaperSize.x / 1000.0, (double)dlg.m_psd.ptPaperSize.y / 1000.0); AfxMessageBox(temp); } }

It is possible to create a customized page setup dialog. To do so, derive a class from CPageSetupDialog, supply your own dialog template, and supply a message map to process notifications from any controls you might have added. The original dialog template for the Page Setup dialog can be found in the file prnsetup.dlg in your Program Files\DevStudio\ VC\include directory.
OnDrawPage

If you derive your own class from CPageSetupDialog, you can override the member functions and PreDrawPage to draw a customized version of the image of the printed page in the dialog. You can utilize this capability to render an image of a printed page that closely resembles a page printed by your application.

CPrintDialog
Although AppWizard-generated application skeletons provide printing and page setup capabilities, sometimes it is not possible to rely on AppWizard-generated code for this purpose. In this case, applications must explicitly utilize the Print, Print Setup, and Page Setup dialogs. The CPrintDialog class supports the use of the Print and Print Setup dialogs in MFC applications. However, applications written for Windows 95 or later or Windows NT 3.51 or later should refrain from using the Print Setup dialog; they should instead use the new Page Setup dialog. To create a Print dialog, construct a CPrintDialog object and call its DoModal member function. The first parameter of the constructor, a Boolean value, determines whether the dialog is a Print or Print Setup dialog; set this parameter to TRUE if you want to use the Print Setup dialog. The dialog can be initialized by setting the values in the m_pd member structure, which is a structure of type PRINTDLG. The users printing preferences can be obtained through member functions such as GetDeviceName or GetPrinterDC.

304

Microsoft Foundation Classes PART III

The CPrintDialog class can also be used to obtain a printer device context corresponding to the default printer without displaying a dialog. Use the CreatePrinterDC function for this purpose. Note that this function overwrites any previously stored device context handles in m_pd.hDC without actually deleting the device context object. The device context object must be deleted not only when it has been created by a call to CreatePrinterDC, but also when CPrintDialog has been constructed for a Print dialog (with the first parameter of the constructor set to FALSE). Applications must also delete any DEVMODE and DEVNAMES structures supplied by CPrintDialog by calling the Windows function GlobalFree for the handles m_pd.hDevMode and m_pd.hDevNames. Listing 16.6 shows how a Print dialog is created in the CDLG application.

Listing 16.6. Use of CPrintDialog in CDLG.


void CMainFrame::OnViewPrintdialog() { // TODO: Add your command handler code here CPrintDialog dlg(FALSE); if (dlg.DoModal() == IDOK) { CString temp = Device selected: + dlg.GetDeviceName(); AfxMessageBox(temp); } GlobalFree(dlg.m_pd.hDevMode); GlobalFree(dlg.m_pd.hDevNames); DeleteDC(dlg.GetPrinterDC()); }

It is possible to customize the Print and Print Setup dialogs. To do so, derive a class from CPrintDialog, create a custom dialog template, and add a message map to process notification messages from any new controls. Use the dialog templates in the file prnsetup.dlg in your Program Files\DevStudio\VC\include directory as the basis for any new dialog template you create. Note that if you want different class behavior depending on whether the Print or Print Setup dialog is displayed, you might have to derive to separate classes from CPrintDialog.

COleDialog
OLE supports several additional common dialogs. These common dialogs are also supported by the MFC Library through a series of dialog classes. The class COleDialog serves as the base class for all OLE common dialog classes. Figure 16.2 shows the set of OLE common dialogs.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

305

FIGURE 16.2.
OLE common dialogs in MFC.

COleDialog

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

COleBusyDialog

COleChangeIconDialog

COleChangeSourceDialog

COleConvertDialog

COleInsertDialog

COleLinksDialog

COleUpdateDialog

COlePasteSpecialDialog

COlePropertiesDialog

Common Controls
The MFC provides a wrapper class for most of the new common controls, including those introduced with Windows 95 and Internet Explorer 4. Many of these controls can now be constructed using the dialog editor within the Developer Studio. Assigning an object of the appropriate control class to a newly created control is often simply a matter of creating a member variable through ClassWizard. In our tour of common controls, we are going to use a simple application to demonstrate control usage where applicable. This application, which I named CTRL, is based on an AppWizardgenerated application skeleton (dialog-based) with all the standard settings. This application demonstrates the use of sliders, progress bars, hotkeys, spin buttons, list controls, tree controls, animation controls, and tab controls. The tab control is used to implement a tabbed dialog by hand; something you would not normally do when you can use property sheets, but it serves as a perfect demonstration of the tab controls capabilities.

306

Microsoft Foundation Classes PART III

Animation Control
The animation control is a control that can display simple AVI format video files. You can create an animation control in a dialog using the dialog editor in Developer Studio. You can then use the ClassWizard to assign a member variable of type CAnimateCtrl. It is typically not necessary to derive your own class from CAnimateCtrl. To load a video clip into an animation control, you should call the CAnimateCtrl::Load member function. This function accepts either a filename or a resource identifier as its parameter. For example, in the CTRL application, I added an AVI file as a custom resource in an external file, and assigned to it the text identifier VIDEO. Afterward, in the CTRL dialogs OnInitDialog member function, I made the following function call:
m_cAnimate.Open(_T(VIDEO));

Unless the animation control has been created with the auto play style, you have to start playback by calling the CAnimateCtrl::Play member function. By setting the appropriate parameters of this function, you can specify the starting and ending frame, and the number of times the clip should be replayed. To play back the entire clip and repeat playback continuously, I used the following call to this function in CTRL:
m_cAnimate.Play(0, -1, -1);

Figure 16.3 illustrates the animation control in the CTRL application.

FIGURE 16.3.
Animation control in the CTRL application.

NOTE
The animation control is not designed as a general-purpose video clip playback facility. It serves the specific purpose of displaying simple animations, such as an animated icon during a lengthy operation.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

307

Date Time Picker Control


A date time picker control (see Figure 16.4) lets the user enter date and time values. It can optionally use a month calendar control where the user can pick a date.

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

FIGURE 16.4.
A date time picker control and an IP address control.

The time can be set in a date time picker control using SetTime and read using GetTime. This control is available only on Windows 98 or later, Windows NT 5 or later, or on systems with Internet Explorer 4 installed.

Header Control
The header control is used to create column headers for columnized lists. This control is most frequently encountered as part of list controls in Report View. For example, the list control in Figure 16.6 contains a header control. If you want to create a header control on its own, you can use the CHeaderCtrl class for this purpose. To create the actual control, call this classs Create member function. To add items to the header control, call CHeaderCtrl::InsertItem. If the header control is placed inside a view window, you would typically perform these initializations in the view objects OnInitialUpdate member function. A header control cannot be added to a dialog using the dialog editor in Developer Studio. If you want to use a header control in your dialog, add a member variable of type CHeaderCtrl to your dialog class manually, and initialize the header control in your dialogs OnInitDialog member function.

Hotkey Control
A hotkey control is a special edit control; it accepts single keystrokes, displays a symbolic representation of the keystroke (for example, Alt+Shift+F1), and returns the virtual key code and shift codes corresponding to the key. It is typically used in conjunction with the WM_SETHOTKEY message to associate a hotkey with a window.

308

Microsoft Foundation Classes PART III

The use of the hotkey control is demonstrated in the CTRL application (see Figure 16.5). When the user selects a hotkey and clicks the Set button, a WM_SETHOTKEY message is sent to the dialog window; subsequently, any time the user types the selected key, the CTRL dialog is activated.

FIGURE 16.5.
A hotkey control and a spin button in CTRL.

A hotkey control can be added to a dialog using the Developer Studio dialog editor, and a corresponding object of type CHotKeyCtrl can be created as a member variable of the dialog class using ClassWizard. The hotkeys current setting is usually retrieved when the user indicates that the selection is complete, for example, by clicking on a button. In the CTRL application, the Set button serves this purpose. The value of the hotkey can be retrieved by calling the member function CHotKey::GetHotKey; the DWORD value returned by this function can be used in a subsequent call to SendMessage as the WPARAM parameter of a WM_SETHOTKEY message. Listing 16.7 illustrates this usage in the CTRL application.

Listing 16.7. Processing a hotkey in CTRL.


void CCTRLDlg::OnButton() { // TODO: Add your control notification handler code here SendMessage(WM_SETHOTKEY, m_cHotKey.GetHotKey()); }

The hotkey control can also be used in conjunction with the Win32 function RegisterHotKey to set up thread-specific hotkeys.

IP Address Control
The IP address control (refer to Figure 16.4) simplifies the task of requesting IP numbers. IP numbers, the numeric addresses used on the Internet, consist of four groups of decimal digits separated by periods. Each group represents a decimal number between 0 and 255. The IP address control provides a text field that is divided into four regions, and also performs validation of the numbers entered.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

309

The address can be set and retrieved either as a DWORD value or as a set of four BYTE values, through the member functions GetAddress and SetAddress. This control is available only on Windows 98 or later, Windows NT 5 or later, or on systems with Internet Explorer 4 installed.

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

List Control
A list control presents a collection of items, each identified by an icon and a label. Items in a list control can be arranged in a variety of ways. In Icon view, items appear as large icons with text underneath and can be dragged anywhere in the control. In Small Icon view, items are displayed as small icons with text to the right and can be dragged to any position. In List view, items appear in columnar form as small icons with text to the right, and their position cannot be altered. In Report view, items are presented in multiple columns, with subitems occupying the columns right of the first columns. The MFC supports list controls in two ways. The CListView class can be used to create a view window with a list control occupying the entire client area. For other uses of list controls, including use of list controls in dialogs, you can use the CListCtrl class. If you want to add a list control to a dialog, you can do so using the Developer Studio dialog editor. Construction of a list control is a multiple step process. It involves the creation of the list control, the setting up of its columns by calling the InsertColumn member function, and the insertion of list control items by calling InsertItem. The icons associated with items must be supplied in the form of an image list. Image lists represent a collection of small graphics items of identical size. Support for image lists in MFC is provided in the form of the CImageList class. Figure 16.6 shows the use of a list control in the CTRL application. The list control in the dialog template has been configured for Report view and single selection.

FIGURE 16.6.
A list control in CTRL.

310

Microsoft Foundation Classes PART III

The bitmap that is used in the list control consists of four images, 16 by 16 pixels each. This bitmap is shown in Figure 16.7.

FIGURE 16.7.
Bitmap of list control icons.

To manage the list control in CTRL, I used the ClassWizard to create a member variable of type CListCtrl in my dialog. The list control is then initialized in OnInitDialog, as show in Listing 16.8.

Listing 16.8. Setting up a list control in CTRL.


m_cImageList.Create(IDB_IMAGE, 16, 10, 0); m_cList.InsertColumn(0, _T(Shape), LVCFMT_LEFT, 200); m_cList.SetImageList(&m_cImageList, LVSIL_SMALL); PSTR pszListItems[] = {_T(Square), _T(Rectangle), _T(Rounded Rectangle), _T(Circle), _T(Ellipse), _T(Equilateral Triangle), _T(Right Triangle), NULL}; int nListTypes[] = {0, 0, 0, 1, 1, 2, 2}; for (int i = 0; pszListItems[i] != NULL; i++) m_cList.InsertItem(i, pszListItems[i], nListTypes[i]);

When processing the results of the users selection in a list control, the state of an item can be obtained by calling the member function CListCtrl::GetItemState.

Month Calendar Control


The month calendar control presents a monthly calendar that can be used by the user to pick a date. This is the same calendar control that is used optionally by the date time picker control (refer to Figure 16.4). A calendar control can display one or more months and has several customizable features. It can also be used to select a single date or a range of dates. The currently selected date can be set and retrieved using the SetCurSel and GetCurSel member functions. This control is available only on Windows 98 or later, Windows NT 5 or later, or on systems with Internet Explorer 4 installed.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

311

Progress Bar
A progress bar is used to provide graphics feedback to the user about progress during a lengthy operation. It is different from other controls inasmuch as its sole purpose is to provide information to the user; it does not accept input of any kind. To use a progress bar in a dialog, insert a progress bar control using the Developer Studio and assign a member variable of type CProgressCtrl to it using ClassWizard. Initialization of the control consists of setting up the minimum and maximum value and, optionally, the controls current position. When using the control in a dialog, this can be done in the dialog classs OnInitDialog member function. Figure 16.8 shows the use of a progress bar in conjunction with a slider control in CTRL.

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

FIGURE 16.8.
A progress bar and a slider control in CTRL.

The initialization of the progress bar in CTRLs ing 16.9.

CCTRLDlg::OnInitDialog

is shown in List-

Listing 16.9. Initializing a progress bar in CTRL.


m_cProgress.SetRange(1, 100); m_cProgress.SetPos(1);

The position of the progress bar can be set by calling the SetPos member function.

Rich-Text Edit Control


The rich-text edit control, or RTF control (also called rich edit control), vastly enhances the capabilities of the standard Windows edit control. RTF controls display and manage formatted text.

312

Microsoft Foundation Classes PART III

Rich-text edit controls present a complex programming interface, which is reflected in the numerous member functions of the classes supporting this type of control. In fact, the MFC Library offers four classes in connection with this control. Simple support for rich-text edit controls is provided via the CRichEditCtrl class. However, the MFC Library also supports applications with documents based on rich-text control capabilities. Such support is provided through the classes CRichEditDoc, CRichEditView, and CRichEditCntrItem. Rich edit controls can be viewed as controls providing a superset of the capabilities of multiline edit controls. Text in a rich edit control can be assigned character and paragraph formatting. Formatting of text within a rich edit control represented by a CRichEditCntr object can be accomplished by calling the SetSelectionCharFormat, SetWordCharFormat, and SetParaFormat member functions of the control class. The rich edit control does not provide a user interface for text formatting; the application must implement the user interface in the form of menus, toolbars, and other user-interface controls. Rich edit controls can also support Clipboard, OLE, printing, and file operations. Rich edit controls can load and save text in both text (ASCII) and rich text format. Use of a rich edit control in an MFC application is demonstrated by the Windows 95 WordPad application. As a minor miracle, Microsoft decided to release the source code for this application in its entirety to developers. (We are talking about production quality source code here, not some Mickey Mouse implementation like my three-liner examples!) This application provides an excellent reference for anyone wanting to utilize a rich edit control in an MFC application.

Slider Control
Slider controls, also called trackbars, are often compared to the sliding volume controls on stereo equipment. In this facility, they provide a replacement for the dubious practice of using scrollbars for this purpose (for lack of anything better). Slider controls are supported in MFC through the CSliderCtrl class. If used in a dialog, a slider control can be inserted into the dialog template using the Developer Studio dialog editor. Next, create a corresponding member variable in your dialog class using ClassWizard. Through this member variable, the slider control can be initialized in your dialog classs OnInitDialog member function. Initialization includes setting up minimum and maximum values for the slider and optionally setting its initial position and changing its appearance. The use of a slider control in the CTRL application was shown in Figure 16.8. This slider control in CTRL is used to control the position of the progress bar. Initialization of the slider control consists of a single function call:
m_cSlider.SetRange(1, 100);

MFC Support for Common Dialogs and Common Controls CHAPTER 16

313

Slider controls send WM_HSCROLL messages to their parent window whenever the position of the slider changes. Handlers for these messages can be installed in an MFC application using ClassWizard. In CTRL, a handler function has been installed that takes the position of the slider control and uses it to update the position of the progress bar. This handler function is shown in Listing 16.10.

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

Listing 16.10. Handling WM_HSCROLL messages from a slider control.


void CCTRLDlg::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar) { // TODO: Add your message handler code here and/or call default if ((CSliderCtrl *)pScrollBar == &m_cSlider) { m_cProgress.SetPos(m_cSlider.GetPos()); } else CDialog::OnHScroll(nSBCode, nPos, pScrollBar); }

Spin Button
A spin button, also known as an up-down control, consists of a pair of arrows that is typically used to increment or decrement a value. Spin buttons are most often used in conjunction with edit controls that contain a numerical value. Spin buttons can be inserted into a dialog using the Developer Studio dialog editor. A spin button automatically adjusts the value of an edit control if such a control is set to be its buddy control. Alternatively, you can create a spin button with the auto-buddy style set; such a spin button automatically selects the control preceding it in the dialogs tab order as its buddy control. Often nothing else is needed; by setting up a spin control with the auto-buddy style, you might eliminate the need to write a single line of additional code in your application. However, should you need to manipulate the spin button, you can create a member variable of type CSpinButtonCtrl in your dialog class. The CTRL application demonstrates the use of a spin control (refer to Figure 16.5). A member variable for the spin control has been created for the sole purpose of showing or hiding the control.

Status Window
Anyone who is used to the standard appearance of MFC applications must be familiar with status windows, or status bars. MFC applications display a status bar in the bottom of their main frame windows.

314

Microsoft Foundation Classes PART III

The Windows 95-style status window control is supported in MFC by the CStatusBarCtrl class. Applications can utilize this class if there is a need to create status bars above and beyond what is offered by the MFC framework. In order to create a status bar object, you must first create a CStatusBarCtrl object and next use its Create member function to create the actual control. Status bar controls are not supported by the Developer Studio dialog editor (nor are they typically part of dialogs). The CStatusBar class is used to integrate status bars with MFC frame windows.

Tab Control
Tab controls are used to present a series of overlapping items using a more compact visual interface. The typical use of tab controls is in property sheets (tabbed dialogs). Tab controls can be created in a dialog template by using the Developer Studio dialog editor. Next, a dialog class member variable of type CTabCtrl can be created using ClassWizard. Through such a member variable, the labels and appearance of the tab control can be set. Tab controls send notification messages to their parent window. Handlers for these messages can be installed using ClassWizard. In the CTRL application, a tab control is used to simulate the appearance of a tabbed dialog (property sheet). Refer to Figure 16.3 and subsequent illustrations to see this tab control in action. The member variable assigned to the tab control, m _ c T a b , is initialized in the CCTRLDlg::OnInitDialog member function. This initialization is shown in Listing 16.11.

Listing 16.11. Tab control initialization in CTRL.


TC_ITEM tcItem; PSTR pszTabItems[] = {_T(Slider), _T(Spin), _T(List), _T(Tree), _T(Animate), NULL}; for (i = 0; pszTabItems[i] != NULL; i++) { tcItem.mask = TCIF_TEXT; tcItem.pszText = pszTabItems[i]; tcItem.cchTextMax = strlen(pszTabItems[i]); m_cTab.InsertItem(i, &tcItem); }

A handler for messages of type TCN_SELCHANGE has been installed (using ClassWizard) to respond to the users selection of a tab in the tab control. In this handler, shown in Listing 16.12, other controls in the dialog are enabled and disabled in accordance with the newly selected tab. The function also sets the focus to the appropriate control and restarts video playback in CTRLs animation control if that control is made visible as a result of selecting the Animate tab.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

315

Listing 16.12. Handling of TCN_SELCHANGE messages in CTRL.


void CCTRLDlg::OnSelchangeTab(NMHDR* pNMHDR, LRESULT* pResult) { // TODO: Add your control notification handler code here m_cSlider.ShowWindow(m_cTab.GetCurSel() == 0 ? SW_SHOW : SW_HIDE); m_cProgress.ShowWindow(m_cTab.GetCurSel() == 0 ? SW_SHOW : SW_HIDE); m_cEdit.ShowWindow(m_cTab.GetCurSel() == 1 ? SW_SHOW : SW_HIDE); m_cSpin.ShowWindow(m_cTab.GetCurSel() == 1 ? SW_SHOW : SW_HIDE); m_cHotKey.ShowWindow(m_cTab.GetCurSel() == 1 ? SW_SHOW : SW_HIDE); m_cButton.ShowWindow(m_cTab.GetCurSel() == 1 ? SW_SHOW : SW_HIDE); m_cList.ShowWindow(m_cTab.GetCurSel() == 2 ? SW_SHOW : SW_HIDE); m_cTree.ShowWindow(m_cTab.GetCurSel() == 3 ? SW_SHOW : SW_HIDE); switch (m_cTab.GetCurSel()) { case 0: m_cSlider.SetFocus(); break; case 1: m_cEdit.SetFocus(); break; case 2: m_cList.SetFocus(); break; case 3: m_cTree.SetFocus(); break; case 4: m_cAnimate.SetFocus(); break; } if (m_cTab.GetCurSel() == 4) { m_cAnimate.ShowWindow(SW_SHOW); m_cAnimate.Play(0, (UINT)-1, (UINT)-1); } else { m_cAnimate.Stop(); m_cAnimate.ShowWindow(SW_HIDE); } *pResult = 0; }

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

Toolbar
Toolbars are windows containing a set of buttons. If the user clicks on a button, the toolbar sends a command message to its parent window. Toolbars are used in MFC applications extensively. Support for toolbar controls is provided in MFC through the CToolBarCtrl class. A toolbar control is created by first creating an object of type CToolBarCtrl, and then calling its Create member function to create the control. The control supports buttons containing bitmaps, text strings, or both; these can be added by calling the AddBitmap or AddString member functions of CToolBarCtrl. Toolbar buttons can be created by calling the AddButtons member function. Toolbars work in conjunction with tooltips. If you create a toolbar with the TBSTYLE_TOOLTIPS style, the toolbar will send tooltip notification messages to its parent window. These notifications are requests for tooltip text, which is then used in tooltip controls that are created and managed by the toolbar control.

316

Microsoft Foundation Classes PART III

Toolbars support extensive customization through a system-defined dialog box that enables the user to add, delete, and rearrange toolbar buttons. During toolbar customization, the toolbar sends notification messages to its parent window. The state of a toolbar can be saved to the Registry using the CToolBarCtrl::SaveState member function. The saved state of the toolbar can be retrieved using CToolBarCtrl::RestoreState. MFC supports toolbars that are integrated with frame windows through the CToolBar class.

Tooltip Control
Tooltips are small pop-up windows that display a single line of text. Tooltips are typically used as a visual aid in identifying the function of a control, or tool, in an application. Tooltips are automatically created and managed by toolbar controls. Tooltip support is also provided in the CWnd class through the functions EnableToolTips , CancelToolTips , FilterToolTipMessage, and OnToolHitTest. To create a tooltip, first create an object of type CToolTipCtrl, then call its Create member function to create the control. Use the AddTool member function to register a tool with the tooltip. Registering consists of identifying a window and a rectangular area within it; if the mouse cursor rests within that area for more than one second, the tooltip will appear. You can register several tools with the same tooltip control. To activate a tooltip, call the member function CToolTipCtrl::Activate. Tooltips send notification messages to their parent windows. In particular, if a tooltip needs information on the text that is to be displayed, it will send TTN_NEEDTEXT notifications to its parent window.

Tree Control
A tree control displays a list of items in a hierarchical form. Three controls are supported in MFC by the CTreeView and CTreeCtrl classes. CTreeView is used by applications that use views that consist entirely of a single tree control. Tree controls can be added to a dialog template using the Developer Studio dialog editor. For every tree control, a corresponding object of type CTreeCtrl should be created using ClassWizard. It is through this object that the tree control is initialized. The items in a tree control are set using the CTreeCtrl::InsertItem member function. Tree controls, like their list control cousins, use image lists for displaying icons associated with items. To create an image list, use the class CImageList. Tree controls send notification messages to their parent window. For example, when an item in a tree control is selected, the tree control sends a TVN_SELCHANGED notification.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

317

Figure 16.9 demonstrates the use of a tree control in the CTRL application. This control was added to the applications dialog using the Developer Studio dialog editor. Three style settings (Has buttons, Has lines, Lines at root) were set to enable the display of lines and expand buttons within the control.

16
FOR

MFC SUPPORT DIALOGS AND CONTROLS

FIGURE 16.9.
A tree control in CTRL.

The control is initialized in the CCTRLDlg::OnInitDialog member function (see Listing 16.13).

Listing 16.13. Initialization of a tree control in CTRL.


PSTR pszTreeRoots[] = {_T(Rectangles), _T(Ellipses), _T(Triangles)}; m_cTree.SetImageList(&m_cImageList, TVSIL_NORMAL); HTREEITEM rootitems[3]; for (i = 0; i < 3; i++) rootitems[i] = m_cTree.InsertItem(TVIF_PARAM | TVIF_TEXT | TVIF_IMAGE | TVIF_SELECTEDIMAGE, pszTreeRoots[i], i, i, 0, 0, -1, TVI_ROOT, TVI_LAST); for(i = 0; pszListItems[i] != NULL; i++) m_cTree.InsertItem(TVIF_PARAM | TVIF_TEXT | TVIF_IMAGE | TVIF_SELECTEDIMAGE, pszListItems[i], 3, 3, 0, 0, i, rootitems[nListTypes[i]], TVI_LAST);

The CTRL application handles notification messages from this tree control. When an item in the tree control is selected, the handler for TVN_SELCHANGED messages sets the selection state of the corresponding item in the applications list control. This handler function is shown in Listing 16.14.

318

Microsoft Foundation Classes PART III

Listing 16.14. Handling of TVN_SELCHANGED messages in CTRL.


void CCTRLDlg::OnSelchangedTree(NMHDR* pNMHDR, LRESULT* pResult) { NM_TREEVIEW* pNMTreeView = (NM_TREEVIEW*)pNMHDR; // TODO: Add your control notification handler code here int i = m_cTree.GetItemData(m_cTree.GetSelectedItem()); if (i != -1) m_cList.SetItemState(i, LVIS_SELECTED, LVIS_SELECTED); *pResult = 0; }

Summary
The Microsoft Foundation Classes Library provides extensive support for the common dialogs and common controls that are available in Windows. Common dialog support is available through a series of classes that are derived from CCommonDialog. These classes encapsulate the functionality of the Windows common dialogs and provide easy ways to set up, display, and process such dialogs. Specifically, the following dialogs are supported: s s s s s s Color selection dialogs are supported through the CColorDialog class. File Open and File Save As dialogs are supported through the CFileDialog class. Find and Replace dialogs are supported through CFindReplaceDialog. Font selection dialogs are supported through the class CFontDialog. Page setup dialogs are supported through CPageSetupDialog. Print and Print Setup dialogs are supported through CPrintDialog.
COleDialog

In addition, a variety of classes derived from dialogs.

support OLE-related common

Common controls are a new feature of Windows, introduced with Windows 95. Common controls drastically enrich the set of controls available for application programmers for use in dialogs and views. The type of common controls and the form of support MFC and the Developer Studio dialog editor offer for these controls are summarized in Table 16.1.

MFC Support for Common Dialogs and Common Controls CHAPTER 16

319

16
FOR

Table 16.1. MFC and Developer Studio support for common controls. Control type Control class View class
Animation control Date Time Picker control Header control Hotkey control IP Address control List control Month Calendar Control Progress Bar Rich-text edit control Slider control Spin button Status window Tab control Toolbar Tooltip Tree control
CAnimateCtrl CDateTimeCtrl CHeaderCtrl CHotKeyCtrl CIPAddressCtrl CListCtrl CMonthCalCtrl CProgressCtrl CRichEditCtrl CSliderCtrl CSpinButtonCtrl CStatusBarCtrl CTabCtrl CToolBarCtrl CToolTipCtrl CTreeCtrl CTreeView CRichEditView CListView

MFC SUPPORT DIALOGS AND CONTROLS

In dialog editor? Yes Yes No Yes Yes Yes Yes Yes Yes Yes Yes No Yes No No Yes

Additional rich-text edit control support is provided in the form of the CRichEditDoc and CRichEditCntrItem classes. Through these classes and CRichEditView, applications can be constructed that manage rich-text documents. More integrated support for toolbars and status windows is provided in the form of the CToolBar and CStatusBar classes. These classes integrate toolbars and status bars with MFC frame windows.

320

Microsoft Foundation Classes PART III

Using ActiveX Controls CHAPTER 17

321

Using ActiveX Controls

17

IN THIS CHAPTER
s Adding ActiveX Controls to Your Application 323 s ActiveX Controls Supplied with Visual C++ 330

17
USING ACTIVEX CONTROLS

322

Microsoft Foundation Classes PART III

Several years ago, Microsofts Visual Basic programming environment introduced a new style of software development. Visual Basic custom controls, or VBXs, provided the basis for component-based programming. Although VBXs were not exactly revolutionary (they represented just another form of a dynamic link library), they fulfilled the promise of reusability to a degree not previously thought possible. Novice programmers who may not even have heard the acronym DLL manipulated VBXs with ease and created near-professional quality Visual Basic applications in days. Although VBX technology was developed as a technology specific to Visual Basic, its unprecedented success prompted Microsoft to add VBX support to the Visual C++ development system. Interfacing C programs and a VBX library is not as hard as it sounds; after all, the VBX is also written in the C language. The difficulty lies in the fact that the data types used were tailored towards the BASIC language, not C. Unfortunately, the good days of VBX support in C programs were too good to last. Shortly after VBX support was introduced, ongoing development of the 16-bit version of Visual C++ ceased. The 32-bit version, which represented the development environment of choice for Windows NT and now Windows 95, was fundamentally incompatible with 16-bit VBX libraries. Microsoft decided to address this issue and design a new custom control mechanism that also promised to resolve other VBX limitations. No longer a technology specific to a single programming environment, the new OLE custom controls used OLE technology for communicating with a control container application and promised the same ease of use as VBX technology. These OLE custom controls have since evolved into Microsofts flagship ActiveX control technology. This chapter examines ActiveX controls from the perspective of the control user; that is, the application programmer who takes an ActiveX control as some kind of a black box and incorporates it as a component into his or her application. This perspective markedly differs from that of the control developer, who creates the ActiveX control in the form of an OCX library; and that of the end user, who is the recipient of the application created by the control user. For the purposes of demonstration, I developed a very simple ActiveX control with the uninspired name MCTL. MCTL is a button that can have a rectangular, elliptical, or triangular shape; when the button is clicked with the mouse, it changes color from green to red or back to indicate its selected or deselected state (Figure 17.1).

FIGURE 17.1.
The MCTL control in action.

Using ActiveX Controls CHAPTER 17

323

What exactly is an ActiveX control? It can be viewed as an object that presents itself to the application programmer in the form of a series of properties, methods, and events. Control properties can be set or retrieved and are represented by values of various types, such as integers or character strings. Control methods are procedures that you can call in order for the control to perform a function. Events are messages that the control sends to its parent window indicating an occurrence of some kind, such as a mouse click. Although many controls present a visual interface, a control does not necessarily have to be visible in order to be functional. It is possible to create controls that offer properties, respond to methods, and send event notifications without ever presenting a visual interface to the end user.

17
USING ACTIVEX CONTROLS

Adding ActiveX Controls to Your Application


Follow these steps to add custom control support to an application: 1. Ensure that your application supports custom controls by calling AfxEnableControlContainer. 2. Add the custom control to a dialog template. 3. Set the controls initial properties. 4. Add member variables to represent the control or its properties as appropriate. 5. Add message handlers to handle messages from the control as appropriate. To demonstrate custom control usage, I built an application named MDLG, which uses a dialog with my MCTL control in it. MDLG is an AppWizard-generated, dialog-based application; with default ActiveX control support. Although I describe ActiveX control use in the context of dialogs, custom control use is no more restricted to dialog boxes than is the use of standard Windows controls. Internet Explorer 3 and later versions also support ActiveX controls in HTML (World Wide Web) pages as active content.

Creating a Control Container


In order to be capable of using ActiveX controls, an application must be a control container. This is accomplished by calling AfxEnableControlContainer during application initialization. Applications created by AppWizard and marked to support ActiveX controls have this call placed into the InitInstance member function of their application objects. If you created an application with AppWizard and did not include ActiveX control support, you can add this call manually any time.

Adding an ActiveX Control to a Dialog Template


ActiveX controls can be added to a dialog template just like other controls, using the dialog editor in Developer Studio. However, ActiveX controls, at least initially, may not appear in the

324

Microsoft Foundation Classes PART III

palette of controls. Therefore, it is necessary to use the right mouse button to invoke a pop-up menu with the option of inserting an ActiveX control (Figure 17.2).

FIGURE 17.2.
Inserting an ActiveX control.

When you select the Insert ActiveX Control option, the Developer Studio responds with a dialog listing all ActiveX controls that are installed on your system (Figure 17.3). Select a control and click on the OK button to have it placed in your dialog at a default location in the upper-left corner.

FIGURE 17.3.
Installed ActiveX controls.

Once the control has been placed in the dialog, you can move it around and resize it just as you would move and resize any other control. In the MDLG application, I added a button in addition to the MCTL control. This button will be used by the end user to retrieve and display the controls Shape property. This button is identified as IDC_BUTTON; the MCTL control itself is identified as IDC_MCTLCTRL.

Using ActiveX Controls CHAPTER 17

325

Setting Control Properties


Although it may not be obvious at first sight, as soon as you insert a control into your dialog template, the code comprising the control becomes active. The Developer Studio actually loads the library code in the OCX file and activates the control in design mode. The behavior of many controls varies according to whether the control has been activated in this mode or in user mode when an application using the control is running. The significance of this is that most ActiveX controls offer a property page interface where the controls properties can be set. Many control properties are persistent; settings configured at design time define the runtime appearance and behavior of the control. The MCTL control has exactly two properties. IsSelected, a Boolean property, tells whether the control is in a selected or deselected state. This property is read-only (and so obviously cannot be set at design time). The other property, Shape, determines whether the control acquires an elliptical, rectangular, or triangular shape, and is meant to be used as a design-time property. (In fact, attempting to set this property at runtime results in an error.) A controls properties can be set by right-clicking on the control in the dialog editor to invoke the controls property sheet interface (Figure 17.4).

17
USING ACTIVEX CONTROLS

FIGURE 17.4.
ActiveX control properties.

Of the three or more property pages that form part of this property sheet, the first and the last are supplied by Developer Studio; the ones in the middle are supplied by the control itself. For example, the MCTL control supplies a single property page where the value of its Shape property can be set (Figure 17.5).

FIGURE 17.5.
MCTL property page: Control.

326

Microsoft Foundation Classes PART III

The last property page (Figure 17.6) in this property sheet offers an interface to all control properties. This includes read-only properties as well, whose values cannot be modified.

FIGURE 17.6.
All MCTL control properties.

Adding Member Variables


The procedure for adding member variables for an ActiveX control is like the one for adding member variables for other types of controls. Select the control in the dialog editor and invoke the ClassWizard, select the Member Variables tab in ClassWizard, and double-click on the line containing the controls identifier to add a member variable. However, at this point, something strange takes place. The ClassWizard responds with the dialog shown in Figure 17.7. In order to create variables corresponding to the ActiveX control, it is necessary to first create a CWnd-derived class representing the control. This is done automatically by ClassWizard when you first attempt to add a member variable for the control.

FIGURE 17.7.
Creation of class for ActiveX control.

If you click OK, Developer Studio responds with yet another dialog (Figure 17.8) where you can modify any default class and filenames that the ClassWizard generated for you.

Using ActiveX Controls CHAPTER 17

327

FIGURE 17.8.
Confirm ActiveX control classes.

17
USING ACTIVEX CONTROLS

After adding the new classes, you can proceed adding a member variable (m_cMCTL) for the ActiveX control. This procedure occurs only once; subsequently, you can add member variables for the control normally. Now we take a brief look at the code generated by ClassWizard. When I added a member variable for the MCTL control to my MDLG application, ClassWizard created the CMCTL class; the header file was mctl.h, and the implementation file, mctl.cpp. The declaration of class CMCTL, shown in Listing 17.1, includes all member functions necessary to create the control, adjust its properties, and invoke its methods.

Listing 17.1. CMCTL class declaration.


class CMCTL : public CWnd { protected: DECLARE_DYNCREATE(CMCTL) public: CLSID const& GetClsid() { static CLSID const clsid = { 0xb4a04ecb, 0x7ef0, 0x11d0, { 0xae, 0xcd, 0x2, 0x7, 0x1, 0x1c, 0x43, 0x7c } }; return clsid; } virtual BOOL Create(LPCTSTR lpszClassName, LPCTSTR lpszWindowName, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, CCreateContext* pContext = NULL) { return CreateControl(GetClsid(), lpszWindowName, dwStyle, rect, pParentWnd, nID); } BOOL Create(LPCTSTR lpszWindowName, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID,

continues

328

Microsoft Foundation Classes PART III

Listing 17.1. continued


CFile* pPersist = NULL, BOOL bStorage = FALSE, BSTR bstrLicKey = NULL) { return CreateControl(GetClsid(), lpszWindowName, dwStyle, rect, pParentWnd, nID, pPersist, bStorage, bstrLicKey); } // Attributes public: BOOL GetSelected(); void SetSelected(BOOL); short GetShape(); void SetShape(short); // Operations public: void AboutBox(); };

There is nothing surprising in the implementation file (Listing 17.2), either. However, consider yourself strongly advised to heed the warnings at the top of these files. These pieces of code are machine-generated for the sole purpose of providing an interface to a control written by somebody else. It is not your job as the application programmer to attempt to modify the controls behavior (nor are you able to do that without having access to the controls source code).

Listing 17.2. CMCTL class implementation.


/////////////////////////////////////////////////////////////////// // CMCTL IMPLEMENT_DYNCREATE(CMCTL, CWnd) /////////////////////////////////////////////////////////////////// // CMCTL properties short CMCTL::GetShape() { short result; GetProperty(0x1, VT_I2, (void*)&result); return result; } void CMCTL::SetShape(short propVal) { SetProperty(0x1, VT_I2, propVal); } BOOL CMCTL::GetSelected() { BOOL result; GetProperty(0x2, VT_BOOL, (void*)&result); return result; }

Using ActiveX Controls CHAPTER 17


void CMCTL::SetSelected(BOOL propVal) { SetProperty(0x2, VT_BOOL, propVal); } /////////////////////////////////////////////////////////////////// // CMCTL operations void CMCTL::AboutBox() { InvokeHelper(0xfffffdd8, DISPATCH_METHOD, VT_EMPTY, NULL, NULL); }

329

Back to my MDLG sample application. I added a single member variable of type CMCTL, which I named m_cMCTL. To use this variable, I added a message handler function for the Describe button. In this function, shown in Listing 17.3, I retrieve the controls Shape property by calling the CMCTL::GetShape member function and display the result in a message box.

17
USING ACTIVEX CONTROLS

Listing 17.3. Accessing a controls properties.


void CMDLGDlg::OnButton() { // TODO: Add your control notification handler code here CString shape; switch (m_cMCTL.GetShape()) { case 0: shape = _T(Ellipse); break; case 1: shape = _T(Rectangle); break; case 2: shape = _T(Triangle); break; } AfxMessageBox(_T(The controls shape is ) + shape + _T(.)); }

Handling Messages
Handling messages from an ActiveX control is perhaps even simpler than handling properties. You can add handlers for any event the control supports via ClassWizard. For example, Figure 17.9 demonstrates adding a message handler for the single event that the MCTL control can generate, a Select event. Depending on the type of the event, the event handler may receive parameters. In the case of MCTL, a handler for the Select event received a Boolean parameter specifying whether the control has just been selected or deselected. The sample handler function in MDLG simply displays a message box to this effect, as shown in Listing 17.4.

330

Microsoft Foundation Classes PART III

FIGURE 17.9.
Adding a message map entry for an ActiveX control event.

Listing 17.4. Handling an ActiveX control event.


void CMDLGDlg::OnSelectMctlctrl(BOOL IsSelected) { // TODO: Add your control notification handler code here if (IsSelected) AfxMessageBox(_T(The control has been selected.)); else AfxMessageBox(_T(The control has been deselected.)); }

Thats all there is to it! All that remains is recompiling and running your application. As you can see, the procedure for inserting an ActiveX control is no more complex than the procedure for inserting other types of controls. Moreover, in order to efficiently use an ActiveX control, you need not know the details of its implementation (or even how ActiveX controls are implemented in the first place). All you need is documentation on the controls purpose and behavior, as well as its properties, methods, and events. I believe that component-based programming, as implemented by Microsofts ActiveX control technology, is indeed quickly becoming one of the foundations of state-of-the-art C++ programming.

ActiveX Controls Supplied with Visual C++


Microsoft supplies several ActiveX controls with Visual C++. Some of these controls are similar in function and appearance to VBX controls found in earlier versions (for example, Version 3.0) of Visual Basic. The set of supplied controls is constantly growing, and often new controls that you can distribute with your application can be downloaded from Microsofts Web site. Here, you might also find links to controls supplied by third parties.

Using ActiveX Controls CHAPTER 17

331

Summary
ActiveX controls represent a technology that is a 32-bit successor to the Visual Basic custom control (VBX) technology. In order for an application to act as an ActiveX control container, it is necessary to call the AfxEnableControlContainer function when the application is being initialized. ActiveX controls can be used in applications just like ordinary controls. They can be inserted into a dialog template using the Developer Studio dialog editor. The control can be configured through a set of property page interfaces that appear in the controls property sheet in the dialog editor. The controls properties can be accessed from within the application by using the ClassWizard to add member variables that correspond to the control. When adding a member variable for a certain type of control for the first time, ClassWizard creates a wrapper class for the control and adds it to the project. The ClassWizard can also be used to add handler functions for control events. Microsoft supplies several custom controls as part of the Visual C++ development environment.

17
USING ACTIVEX CONTROLS

Device Context and GDI Objects CHAPTER 18

333

Device Context and GDI Objects

18

IN THIS CHAPTER
s Device Contexts 334 s GDI Object Support in MFC 346

18
DEVICE CONTEXT AND GDI OBJECTS

334

Microsoft Foundation Classes PART III

Drawing and graphics have special significance in almost every Windows application; MFC applications are no exception to this rule. The Windows Graphics Device Interface (GDI) capabilities are encapsulated in two families of MFC classes. Device context classes provide an encapsulation of GDI device contexts and most drawing functions; GDI object classes encapsulate GDI objects such as pens, brushes, bitmaps, or fonts. As in non-MFC Windows applications, drawing to an output device consists of obtaining the appropriate device context, setting up GDI drawing objects, performing drawing operations, and cleaning up. The MFC framework greatly simplifies these steps by assuming many of the more mundane responsibilities that used to befall on the application developer. For example, you can construct a pen object by passing the appropriate parameters to the Cpen constructor function:
Cpen myPen(PS_SOLID, 0, RGB(255, 0, 0));

and never worry about it afterwards; the GDI pen object is destroyed automatically by the Cpen destructor when the Cpen object goes out of scope. As is the case with windows and CWnd objects, there is a distinction between the MFC (CDC- or CGdiObject-derived) object and the actual device context or GDI object in Windows. Constructing the MFC object does not automatically imply construction of an underlying Windows object. On the contrary, it is a legitimate practice to construct a blank MFC object first and later associate it with the Windows object as the need arises. In this chapter, we first focus our attention on device contexts, which serve as the canvas onto which drawing takes place. Actual drawing operations (functions such as Rectangle, Ellipse, or DrawText) are also encapsulated in the CDC class and are discussed here. In the second part of this chapter, we shift our focus to classes that encapsulate GDI objects, which represent drawing tools.

Device Contexts
Although constructing a device context using MFC is easy, often trivial, there are many situations where it is not even necessary. Typical drawing functions (such as the OnDraw member function in a view class) are called by the MFC framework with a pointer to a device context object representing a device context that is already created and configured for use. MFC classes that represent device contexts are all derived from the CDC base class. Figure 18.1 illustrates the hierarchy of device context classes. Although there are several classes derived from CDC, the CDC class itself is frequently used as a wrapper class for device contexts. The other CDC-derived classes differ from CDC primarily in their constructor function and offer no extra functionality. If you need to construct an MFC object that is attached to an existing device context handle, you should always use the base CDC class instead of any of the derived classes.

Device Context and GDI Objects CHAPTER 18

335

FIGURE 18.1
Hierarchy of device context classes.

Device contexts CDC

CPaintDC

CClientDC

CMetafileDC

CWindowDC

The Basic CDC Class


The CDC class encapsulates the functionality of the Windows device context. And there is a lot of functionality to encapsulate! The CDC class not only maps functions that are directly related to configuring and managing a device context, it also maps all GDI drawing functions. Last time I counted, there were approximately 180 documented member functions. With such a large and complex interface, where should we begin? With the simplest parts, of course. The following section reviews how a CDC object is created and attached to a GDI device context.

Creating a Device Context


When a CDC object is created through its constructor function, a GDI device context is not automatically created. Instead, it is necessary to create a device context through the CreateDC function or attach the CDC object to a device context that has been created earlier. The CreateDC member function takes several parameters that specify the device, the device driver software, and the port the device is attached to. These parameters correspond to the parameters of the GDI ::CreateDC function. A CDC object has not one, but two member variables that are GDI device context handles: m_hDC and m_hAttribDC. Usually, these two handles point to the same device context object. The m_hDC handle, or output device context handle, is used for all output operations; the m_hAttribDC handle, or attribute device context handle, is used, in turn, for operations that request information from the device context. To attach a CDC object to a device context handle that has been created earlier, use the Attach member function. To detach a CDC object from a device handle, use the Detach member function. Note that neither D e t a c h , nor the member functions R e l e a s e O u t p u t D C and ReleaseAttributeDC (which reset the values of m_hDC and m_hAttribDC to NULL), actually delete the GDI device context object. In this case, if the device context object was created using the GDI function ::CreateDC, it may be necessary to manually call ::DeleteDC. Calling ::DeleteDC

18
DEVICE CONTEXT AND GDI OBJECTS

336

Microsoft Foundation Classes PART III

is not required if you do not detach the CDC object from the device context; the CDC destructor function makes this call automatically. The CDC class also has a member function DeleteDC, which can be used to detach the CDC object from the GDI device context and delete the device context. This function should only be used if the device context was created using the CreateDC member function. Another function that creates a device context object is CreateCompatibleDC. This function creates a memory device context that is compatible with a given device context. For example, applications may use this function in conjunction with the CClientDC class to create a memory device context that is compatible with the device context representing the current windows client area:
CClientDC clientDC(&myWnd); CDC memDC; memDC.CreateCompatibleDC(&clientDC);

Subsequently, operations such as CDC::BitBlt can be used to transfer blocks of pixels between the two device contexts. Similar techniques are often used in programs that perform smooth animation; by constructing the next animation frame in a memory device context and transferring only completed frames to the display, you can create animations that are free of jerkiness. A static CDC member function is CDC::FromHandle. This function allows you to retrieve the address of a CDC object (if such an object exists) that corresponds to a device context handle. If no such CDC object exists, a temporary CDC object is created. This function may be called as follows:
CDC *pDC = CDC::FromHandle(hDC);

Be warned that the pointer returned by this function is not for any purpose beyond immediate use. As it points to a CDC object that may be under the control of another part of your application, you do not usually know when the CDC object may be destroyed, rendering the pointer returned by CDC::FromHandle invalid. Temporary CDC objects returned by CDC::FromHandle are also deleted by the CDC::DeleteTempMap function, which is typically called from by the idletime handler in your applications CWinApp object. One more function worth mentioning is the GetSafeHdc function. This function returns the m_hDC member of the CDC object. This is a safe function inasmuch as it can also be used with NULL pointers; that is, the following code would be valid and not cause an exception:
CDC *pDC = NULL; HDC hDC = pDC->GetSafeHdc();

Paint-Time Device Contexts


The CPaintDC class encapsulates the calls to the BeginPaint and EndPaint in its constructor and destructor. This class is designed to be used when responding to WM_PAINT messages.

Device Context and GDI Objects CHAPTER 18

337

Note that most applications do not need to create a CPaintDC object directly. The default implementation of the Cview::OnPaint member function creates such a device context and passes it to the classs OnDraw member function (which is usually overridden to provide an applicationspecific drawing of a view).

Client-Area Device Contexts


The CClientDC class is used to create a device context object corresponding to the client area of a given window. The constructor and destructor of CClientDC encapsulate calls to the GetDC and ReleaseDC functions. CClientDC objects are most often used when drawing into a device context is required outside an OnDraw function. A particular use of CClientDC objects concerns mapping modes. Sometimes, it is necessary for an application to translate logical coordinates into physical coordinates or vice versa even when no actual drawing is performed. In these situations, it is a frequent practice to create a CClientDC object for the sole purpose of being able to use one of its coordinate transformation functions, as in this example:
CClientDC dc(myView); myView->OnPrepareDC(&dc); dc.LPtoDP(&point);

Window Device Contexts


Similar to client-area device contexts are window device contexts, represented by the CWindowDC class. The constructor and destructor of CWindowDC encapsulate calls to GetWindowDC and ReleaseDC, respectively.

Metafile Device Contexts


The CMetaFileDC class represents metafile device contexts. Metafile device contexts provide a means to draw into Windows metafiles or the new enhanced metafiles. Metafile device contexts differ from other device contexts in a variety of ways. Most importantly, the m_hAttribDC member of a metafile device context, which would normally be set to refer to the same device as m_hDC, is set to NULL instead. Thus, calls that would retrieve information about the device context would typically fail for an object of type CMetaFileDC. It is possible to assign a value to the m_hAttribDC member. For example, you can assign it the value of another device context that you created, which represents the screen, the printer, or another output device. Constructing a metafile device context is a two-step process. First, the CMetaFileDC object is created; next, its Create or CreateEnhanced member functions are called. Depending on whether or not you supply a filename to the Create or CreateEnhanced member functions, the metafile will be either file-based or memory-based. A memory-based metafile exists only temporarily.

18
DEVICE CONTEXT AND GDI OBJECTS

338

Microsoft Foundation Classes PART III

When you are finished with drawing into the metafile, you close the metafile object by calling CMetaFileDC::Close or CMetaFileDC::CloseEnhanced (depending on the type of the metafile). These functions return a handle to a metafile object. This handle can be used, for example, in a call to CDC::PlayMetafile to play back the metafile into another device context. It can also be passed to the Windows function ::CopyMetaFile (or ::CopyEnhMetaFile) to copy the metafile to a disk file. As soon as you call its Close or CloseEnhanced member function, you can delete the CMetaFileDC object. When you are done with using the metafile handle obtained through calling Close or CloseEnhanced, you should delete the Windows metafile object by calling DeleteMetaFile or DeleteEnhMetaFile.

CDC Attributes
A device context object has many attributes that can be set or retrieved through CDC member functions. Of these, attributes that relate to mapping modes and coordinate transformations are reviewed in the next section. This section focuses on other attributes. The background color, used to fill the gaps in styled lines, hatched brushes, and in character cells, is set by the SetBkColor member function. The current background color can be retrieved by calling GetBkColor. The background mode, which determines whether the background is transparent or opaque, is set by calling SetBkMode; GetBkMode retrieves the current background mode. The SetROP2 member function can be used to set the drawing mode. The drawing mode determines how bits in the drawing tool and bits on the device surface are combined. The default drawing mode is R2_COPYPEN; in this mode, pixels from the drawing tool are copied over pixels in the device bitmap. This is what you would expect as normal behavior; as you draw with a specific pen or brush, the pen or brush will simply overwrite what may already be on the device context surface. There are several other commonly used drawing modes that can be set with SetROP2. These include, for example, R2_BLACK (the target pixels turn always black), R2_NOTCOPYPEN (the target pixel acquires a color that is the inverse of the drawing tools color), or R2_XORPEN (the target pixels color is formed by performing an exclusive OR operation between the target pixel and the pixel in the drawing tool). Drawing modes are not restricted to these preset values; the drawing mode setting can specify an arbitrary binary operation between pixels of the device surface and pixels of the drawing tool. The current drawing mode setting can be acquired by calling GetROP2. Note that drawing mode settings are specific to raster devices and have no effect on vector devices, like plotters. The SetPolyFillMode function determines the polygon filling mode. The difference between the ALTERNATE and WINDING filling modes is illustrated in Figure 18.2. The current filling mode can be retrieved by calling GetPolyFillMode.

Device Context and GDI Objects CHAPTER 18

339

FIGURE 18.2.
Filling modes.

Alternate

Winding

Coordinate Mapping and Views


The coordinates for most graphics operations are provided in the form of logical coordinates. Logical coordinates are translated into device coordinates through what is called coordinate mapping. Mapping defines a linear relationship between the logical and the physical coordinate space. Mapping matches the origin of the logical coordinate space to the origin of the physical coordinate space, and also matches logical and physical coordinate units. Mapping in the horizontal and vertical directions may be independent of each other. Windows defines a set of mapping modes. These mapping modes can be set using the SetMapMode member function. On a raster device such as the screen or printer, device coordinates represent pixel coordinates. The upper-left corner is assigned the coordinates [0,0]; the horizontal coordinate increases from left to right, the vertical coordinate increases from top to bottom. Of the many predefined mapping modes, MM_TEXT matches logical coordinates to physical coordinates. Other predefined mapping modes reverse the direction of the horizontal coordinate, so it grows from bottom to top. These mapping modes are listed in Table 18.1.

18
DEVICE CONTEXT AND GDI OBJECTS

Table 18.1. Mapping modes. Mapping mode Description


MM_LOENGLISH MM_HIENGLISH MM_LOMETRIC MM_HIMETRIC MM_TWIPS

100 logical units equal one inch on the device 1,000 logical units equal one inch on the device 100 logical units equal one centimeter on the device 1,000 logical units equal one centimeter on the device One logical unit is one twentieth of a point (1/1440")

340

Microsoft Foundation Classes PART III

In all of these mapping modes, applications can use the SetWindowOrg and SetViewportOrg functions to set the origin of the logical coordinate space (window) and physical coordinate space (viewport). The significance of these settings is that the two origins are mapped onto each other when coordinates are transformed. In addition to MM_TEXT and the mapping modes in Table 18.1, Windows also defines the MM_ISOTROPIC and MM_ANISOTROPIC mapping mode. In these mapping modes, applications can not only specify the origin, but also the extent of the window and viewport coordinate space. By specifying the extent, applications define how many logical units are mapped to how many physical units. The difference between MM_ISOTROPIC and MM_ANISOTROPIC is that in the former mode, applications only define extents in the horizontal direction, while Windows calculates the vertical extent preserving the device aspect ratio. In the latter mode, applications can freely define any extents in both directions. Figure 18.3 illustrates the effects of a typical mapping from logical to physical coordinates.

FIGURE 18.3.
Coordinate mapping.
Window Extent

Viewport Origin

Viewport Extent

Window Origin

For those who prefer to think in terms of formulae, here is how device coordinates (Dx and Dy) are derived from logical coordinates (Lx and Ly) and vice versa, using the window and viewport origin (xWo and yWO, xVO and yVO), and window and viewport extent (xWE and yWE, xVE and yVE) values:
Dx Dy Lx Ly = = = = (Lx (Ly (Dx (Dy xWO) yWO) xVO) yVO) * * * * xVE/xWE yVE/yWE xWE/xVE yWE/yVE + + + + xVO yVO xWO yWO

The CDC class does not directly support world coordinate transformations that are available in Windows NT. To use world coordinate transforms, applications may need to call the Windows function SetWorldTransform directly. The CDC class provides a set of coordinate transformation functions that can be used to obtain logical coordinates from physical coordinates or vice versa. These functions are DPtoLP and

Device Context and GDI Objects CHAPTER 18


LPtoDP; both of these functions have several overloaded versions that allow them to be used on points, rectangles, and SIZE objects, or MFC classes that encapsulate these objects (CPoint, CRect, CSize).

341

Additional transformation functions include DPtoHIMETRIC, HIMETRICtoDP, LPtoHIMETRIC, and HIMETRICtoLP. These functions are particularly useful for OLE applications. OLE objects are usually measured in HIMETRIC units; these functions provide a direct means of transforming those units directly into physical or logical coordinates or vice versa. Coordinate mapping is used extensively in views (that is, classes derived from CView). In these classes, the member function OnPrepareDC is used to set up coordinate mapping that appropriately reflects the view and its current state. For example, in scroll views, OnPrepareDC is used by the framework to displace the window and/or viewport origin to reflect the amount by which the view client area is scrolled. Applications that wish to implement features such as zooming can do so, for example, by overriding CView::OnPrepareDC and changing the window or viewport extent.

Simple Drawing Functions


The CDC class offers a series of member functions that correspond to low-level GDI drawing operations. Among these is the function FillRect (fills a rectangle with a specific brush), FillSolidRect (fills a rectangle with a specific color), FrameRect (draws the borders of a rectangle), and InvertRect (inverts the interior of a rectangle). Analogous functions that accept regions as their parameters are FillRgn, FrameRgn, and InvertRgn. Additional functions include DrawIcon (draws an icon) and DrawDragRect (erases and draws a dragging rectangle). Other simple drawing functions assist in drawing controls in various (selected, deselected) states and with various border settings.

18

Selecting GDI Objects


Many drawing functions that operate on device contexts require that you select a GDI object into the device context first. To select a GDI object into a device context, use the CDC::SelectObject member function. This member function has several overloaded versions that enable you to select an object of type CPen, CBrush, CFont, CBitmap, or CRgn into the device context. Select a pen into a device context for functions that draw lines. These include simple line drawing functions (such as Line, Arc) as well as functions that draw shapes, as the contour of shapes is drawn using the current pen. Select a brush when you are drawing a shape (for example, Ellipse, Rectangle). The brush will be used to fill the interior of the shape. Select an object of type CFont into a device context if you want to draw text using a specific font.

DEVICE CONTEXT AND GDI OBJECTS

342

Microsoft Foundation Classes PART III

To use memory device contexts, you must select a CBitmap object into them. The CBitmap object must represent either a monochrome bitmap or a bitmap that is compatible with the device context. Selecting a CRgn object into a device context sets the clipping region of the device context to the specified region. Doing this is equivalent to calling the CDC::SelectClipRgn member function. In many situations, it is expected that when you are finished using a device context, you restore its previous state, including any previous GDI object selections. There are two ways of doing this. You can save the return value of SelectObject (which is usually a pointer to a CPen, CBrush, CFont, or CBitmap object representing the previous selection) and use it in a subsequent call to SelectObject. Alternatively, you can use the SaveDC and RestoreDC member functions. In either case, it is your responsibility to delete any GDI objects you created after they are no longer in use. A variant of SelectObject is SelectStockObject; this CDC member function enables you to select a GDI stock object into the device context.

Basic Lines and Shapes


The CDC class provides a series of drawing functions that correspond to Windows GDI drawing functions. Basic drawing functions include those that draw various (straight and curved) lines and those that draw shapes. Many line drawing functions make use of the concept of the current position in the device context. The current position is a pair of coordinates that usually represents the endpoint of the last drawing operation. The current position can be set using the MoveTo member function and can be retrieved using the GetCurrentPosition member function. Line drawing functions use the current pen for drawing the line. To draw a straight line, use the MoveTo and LineTo member functions. To draw an elliptical arc, use the Arc or ArcTo functions. The direction of the arc can be controlled using the SetArcDirection member function. (Use GetArcDirection to retrieve the current setting.) The Polyline and PolylineTo functions can be used to draw a series of connected line segments; PolyPolyline is a function to draw several polylines in a single operation. Windows can also draw Bzier curves; use the PolyBezier or PolyBezierTo member functions. Finally, a series of line segments and Bzier curves can be drawn in a single operation using PolyDraw. Shape functions include Rectangle, RoundRect, Ellipse, Chord, Pie, and Polygon. The shapes generated by these functions are shown in Figure 18.4. One additional function, PolyPolygon, enables the drawing of multiple polygons in a single operation.

Device Context and GDI Objects CHAPTER 18

343

FIGURE 18.4.
Some basic shapes.

Rectangle

RoundRect

Ellipse

Chord

Pie

Polygon

The PaintRgn function draws a region using the current brush. The DrawFocusRect function is used to draw a rectangle around an object to indicate that the object has the focus. The focus rectangle is drawn using the exclusive OR logical function, so calling DrawFocusRect for the second time effectively removes the focus rectangle. Note that if you scroll an area containing a focus rectangle, you must remove the focus rectangle first, scroll the area, and then redraw the focus rectangle.

Bitmaps and Scrolling


Many member functions in the CDC class are used to perform bitwise operations on pixel maps, or bitmaps. Perhaps the simplest pixel operation is SetPixel, which sets a pixel, specified by its logical coordinates, to a specific color. The current color of a pixel can be retrieved by calling GetPixel. A somewhat faster variant of SetPixel is SetPixelV; this version of the function does not return the actual color of the pixel. The BitBlt member function can be used to transfer a rectangular area from one location to another. BitBlt can also be used to transfer blocks of pixels between device contexts. Thus, BitBlt is the operation of choice when, for example, you are transferring blocks of pixels from the screen to a compatible memory bitmap or vice versa. A variant of BitBlt, StretchBlt, also transfers blocks of pixels from one location to another, but it also compresses and stretches the pixel block to fit the destination rectangle. The stretching mode (the method used to eliminate and/or add pixels) is controlled by SetStretchMode (GetStretchMode) and SetColorAdjustment (GetColorAdjustment). The PatBlt member function combines the pixels on the device with the pixels of the selected brush in a bitwise logical operation. The MaskBlt operation combines the source and destination bitmaps and a mask bitmap in a bitwise logical operation. To fill an area in a bitmap using the current brush, call the FloodFill or ExtFloodFill member functions.

18
DEVICE CONTEXT AND GDI OBJECTS

344

Microsoft Foundation Classes PART III

To scroll an area within a device context, use the ScrollDC member function. This function also provides information about the areas uncovered by the scrolling operation, which you can use for repainting purposes. However, if you wish to scroll the entire client area of a window, you should instead utilize the CWnd::ScrollWindow function.

Text and Font Functions


In order to perform text output, applications can use any one of a wide selection of text output and font manipulation functions. The simplest text output function is CDC::TextOut. This function places a character string at a specified location using the currently selected font. A variant, CDC::ExtTextOut, outputs a character string into a specified rectangle. Yet another variant is TabbedTextOut; this function expands tabs in the text that is to be outputted in accordance with an array specifying tab stop positions. The color of the text is determined by SetTextColor (use GetTextColor to retrieve the current setting). The horizontal and vertical text alignment are determined by SetTextAlign (GetTextAlign). This function can also be used to specify that text output functions use the current position (as specified by functions such as MoveTo) rather than any coordinates specified in the function call as the location of the text. It is possible to obtain the size of a block of text without actually drawing the text. The function GetTextExtent calculates the width and height of a line of text using the attribute device context. To perform the same calculation using the output device context, use GetOutputTextExtent. The functions GetTabbedTextExtent and GetOutputTabbedTextExtent perform the same calculations for text that contains tab characters that are to be expanded. The function SetTextJustification can be used in conjunction with the function GetTextExtent to create justified text. SetTextJustification evenly distributes an amount of space among the break characters (usually spaces) in the text. A related function is SetTextCharacterSpacing, which can be used to set the amount of intercharacter spacing. (Use GetTextCharacterSpacing to retrieve the current setting.) A more sophisticated text output function is DrawText. This function can be used to output multiline text. Note that the DrawText function is not recorded in standard Windows metafiles. (It is recorded in enhanced metafiles.) The GrayString function can be used to create grayed (dimmed) text. Information about the current font can be obtained using GetTextFace (retrieves the name of the font), GetTextMetrics (retrieves a TEXTMETRICS structure containing information about the font currently selected in the attribute device context), and GetOutputTextMetrics (same, for the output device context).

Device Context and GDI Objects CHAPTER 18

345

Several other CDC member functions deal with scaleable (TrueType) fonts and information that can be retrieved from such font files.

Clipping Operations
A particularly important GDI feature is the capability to clip output to a specified rectangle or region. This capability is used by Windows throughout; for example, clipping is used to only repaint portions of a window that are not covered by other windows. Applications can make explicit use of clipping through a series of CDC member functions that act as wrappers for similar GDI functions. These include SelectClipRgn, ExcludeClipRect, ExcludeUpdateRgn, IntersectClipRect, and OffsetClipRgn. To obtain the smallest rectangle that encloses the entire clipping region, call GetClipBox. To determine whether a point or any parts of a rectangle are inside the clipping region, use the PtVisible or RectVisible member functions. Windows can also maintain a bounding rectangle in which bounding information about the bounds of subsequent drawing operations is accumulated. To access the bounding rectangle, use the SetBoundsRect and GetBoundsRect member functions.

Printing
Although printing is essentially no different from drawing into any other kind of device contexts, the CDC class offers a series of printer escape member functions that control specific aspects of printing. Printing a document and individual pages is controlled by the StartDoc, StartPage, EndPage, and EndDoc member functions. To abort the printing process (and effectively erase everything that has been sent to the printer device context since the last call to StartDoc), call the AbortDoc member function. Note that if printing is canceled or the printer device driver returns any other error, your application should not call the EndDoc or AbortDoc member functions. Use the SetAbortDoc member function to create a callback function that Windows calls when the print job is canceled or must be terminated. The QueryAbort member function can be used to query this callback function to determine whether printing should be aborted. Device driver-specific features can be accessed through the Escape member function. However, because Win32 provides many more printer control functions, the utility of this function relative to earlier Windows implementations has greatly diminished.

18
DEVICE CONTEXT AND GDI OBJECTS

Path Functions
The CDC class also offers member functions that encapsulate path functionality. A patha complex shape create by a series of GDI function callsis created by calling the BeginPath member function, calling the appropriate drawing functions, and calling EndPath. Calling EndPath automatically selects the path into the device context for subsequent manipulation.

346

Microsoft Foundation Classes PART III

Functions that manipulate paths include CloseFigure, FlattenPath, StrokePath, and WidenPath. The path can be rendered into the device context using StrokePath. To render the paths interior, use FillPath; or you can use StrokeAndFillPath to render both the paths contours and its interior at the same time using the current pen and brush. A path can be turned into a clipping region by calling SelectClipPath.

GDI Object Support in MFC


Many GDI drawing operations are accomplished using a series of GDI objects, such as pens, brushes, or fonts. The Microsoft Foundation Classes Library provides a series of wrapper classes that encapsulate the functionality of these GDI objects. All GDI object classes are derived from the class CGdiObject. Figure 18.5 illustrates GDI object classes in MFC.

FIGURE 18.5.
GDI object classes.

GDI objects CGdiObject

CBitmap

CBrush

CFont

CPalette

CPen

CRgn

The CGdiObject class provides generic support for GDI objects in the form of a series of member functions. The Attach and Detach member functions can be used to attach a CGdiObjectderived MFC object to a GDI object or detach it from the GDI object. The handle of the object, stored in the m_hObject member variable, can be retrieved through the safe function GetSafeHandle. (This function can also be used with null CGdiObject pointers.) A pointer to a CGdiObject that corresponds to a Windows GDI object handle can be obtained by calling the static member function FromHandle. To obtain a GDI objects type, use the GetObjectType member function.

Device Context and GDI Objects CHAPTER 18

347

The CreateStockObject member function can be used to crate a stock pen, brush, font, or palette. Note that this function should be called with a CGdiObject-derived object that is of the appropriate class (CPen, CBrush, CFont, or CPalette). The UnrealizeObject member function can be used to reset the origin of a brush or reset a palette. Do not use this member function for objects of any other type. The DeleteObject member function deletes the GDI object that the CGdiObject-derived MFC object is attached to. The DeleteTempMap function, called usually from the idle-time handler of your applications CWinApp object, is used to delete any temporary CGdiObject objects that were created by the FromHandle member function.

Pens
Support for GDI pen objects in MFC is provided through the CPen class. A pen can be created either in a single step or in two steps. If you wish to create a pen in a single step, you can utilize overloaded versions the CPen constructor for this purpose. For example, to create a dashed black pen, you can declare the pen object as follows:
CPen myPen(PS_DASH, 0, RGB(0, 0, 0));

Alternatively, pens can be created in a two-step operation by creating first the MFC CPen object using a parameterless constructor, and then calling the CreatePen or CreatePenIndirect member functions to create a corresponding GDI pen object, as in this example:
CPen *pPen; pPen = new CPen; pPen->CreatePen(PS_SOLID, 3, RGB(255, 0, 0));

To obtain a LOGPEN structure from a CPen object, use the GetLogPen function. You can also use a CPen object in any GDI function calls that require a pen handle of type HPEN because the CPen class defines the operator HPEN operator function.

18
DEVICE CONTEXT AND GDI OBJECTS

Brushes
The MFC supports GDI brushes through the CBrush class. Like pens, brushes can also be created in either a single step or a two-step process. To create a GDI brush while constructing the CBrush object, use one of the overloaded versions of the CBrush constructor. For example, to create a solid yellow brush, you could construct the CBrush object as follows:
CBrush *pBrush; pBrush = new CBrush(RGB(255, 255, 0));

Alternatively, you can first create the CBrush MFC object through the parameterless constructor and then create the GDI brush object by calling the CreateSolidBrush, CreateHatchBrush,

348

Microsoft Foundation Classes PART III


CreatePatternBrush, CreateDIBPatternBrush, CreateSysColorBrush,

or CreateBrushIndirect

member functions. For example,


CBrush cyanBrush; cyanBrush.CreateSolidBrush(RGB(0, 255, 255));

To obtain a LOGBRUSH structure from a CBrush object, use the GetLogBrush member function. You can also use CBrush objects in place of handles of type HBRUSH in GDI function calls because the CBrush class defines the operator HBRUSH operator function.

Bitmaps
GDI bitmaps are supported in MFC by the CBitmap class. Constructing a bitmap is a two-step process. First, the MFC CBitmap object must be created; next, one of the initialization functions must be called to create the GDI bitmap object. The initialization functions include LoadBitmap, LoadOEMBitmap, LoadMappedBitmap, CreateBitmap, CreateBitmapIndirect, CreateCompatibleBitmap, and CreateDiscardableBitmap. To obtain a pointer to a BITMAP structure representing the GDI bitmap the CBitmap object is attached to, call the GetBitmap member function. You can also use CBitmap objects in place of handles of type HBITMAP in GDI function calls, thanks to the presence of the operator HBITMAP operator function. The bits in a bitmap can be set or read using the SetBitmapBits and GetBitmapBits member functions. You can also assign a width and a height to the bitmap (in LOMETRIC units) using S e t B i t m a p D i m e n s i o n . However, these values are only used as return values with GetBitmapDimension and serve no other purpose.

Fonts
Support for GDI logical fonts is encapsulated in MFC in the CFont class. Creation of a font in MFC is a two-step process: first, the CFont MFC object is created; next, an initialization function is called to create the underlying GDI logical font. The initialization functions include CreatePointFontIndirect.
CreateFont, CreateFontIndirect, CreatePointFont,

and

A pointer to a LOGFONT structure can be obtained by calling the GetLogFont member function. CFont objects can also be used in place of HFONT handles in GDI function calls, thanks to the presence of the operator HFONT operator function.

Palettes
Logical palette support in MFC is provided through the CPalette class. Palettes, like fonts and bitmaps, are constructed in a two-step operation. First, the CPalette object is created; next, an initialization function is called to create the underlying GDI palette object.

Device Context and GDI Objects CHAPTER 18

349

The two palette initialization functions are CreatePalette and CreateHalftonePalette. Palette operations include
AnimatePalette , GetNearestPaletteIndex , GetEntryCount , GetPaletteEntries, SetPaletteEntries,

and ResizePalette.
HPALETTE.

CPalette objects can be used in GDI function calls that require a palette handle of type HPALETTE

because the CPalette class defines the operator function operator

Regions
Support for GDI regions is provided in MFC through the CRgn class. A region is created by first creating the CRgn object, and then calling an initialization function. There are a large number of CRgn initialization functions. These are summarized in Table 18.2.

Table 18.2. CRgn initialization functions. Member function Creates


CreateRectRgn CreateRectRgnIndirect CreateEllipticRgn CreateEllipticRgnIndirect CreatePolygonRgn CreatePolyPolygonRgn CreateRoundRectRgn CombineRgn CopyRgn CreateFromPath CreateFromData

A rectangular region A region from a RECT structure An elliptical region An elliptical region from a RECT structure A polygonal region A region of several (possibly disjoint) polygons A region in the shape of a rounded rectangle A region that is the union of two existing regions A region that is the copy of another region A region from a path A region from a RGNDATA structure and an XFORM matrix

18
DEVICE CONTEXT AND GDI OBJECTS

To compare two CRgn objects and check if they are equivalent, use the EqualRgn member function. To obtain a RGNDATA structure for a CRgn object, call GetRegionData. To obtain a regions bounding rectangle (that is, the tightest rectangle that encloses the region), call GetRgnBox. To set a region to a specific rectangle, call SetRectRgn. To move a region by a specific offset, use OffsetRgn. You can determine whether a given point or parts of a given rectangle fall within the region; use the PtInRegion or RectInRegion member functions. A CRgn object can be used in place of an HRGN handle in GDI function calls, thanks to the presence of the operator function operator HRGN.

350

Microsoft Foundation Classes PART III

Summary
Windows Graphics Device Interface (GDI) functionality is encapsulated by the CDC class (representing device contexts), the CGdiObject class (representing GDI objects), and classes derived from both.
CDC-derived classes include CClientDC (representing a windows interior), CWindowDC (representing a window), and CPaintDC (representing a device context while processing a WM_PAINT message). These derived classes only differ from the base class inasmuch as their constructor and destructor encapsulate calls that create and destroy the appropriate GDI device context (for example, by calling GetDC and ReleaseDC, or BeginPaint/EndPaint). In contrast, a base CDC object is not automatically attached to a GDI device context; the device context must be explicitly created through the CreateDC member function (or attached to through the Attach member function).

Another class derived from CDC is CMetafileDC, which represents metafile device contexts. A CDC object maintains two GDI device context handles. The output device context handle is used for drawing operations; the attribute device context handle is used for operations obtaining information about the device context. The two handles are usually identical, except for the case of CMetafileDC objects, in which case the attribute device context is set to NULL. The CDC class encapsulates most GDI drawing functionality. This includes basic drawing functions, simple lines and shapes, text and font functions, clipping functions, bitmap and scrolling functions, and region and path-related functions. The CDC class also provides functionality related to mapping modes. Note that encapsulation of the Windows NT world coordinate transformation functions is not provided as part of the CDC class. Most drawing operations utilize drawing tools that are selected into a device context using the SelectObject or SelectStockObject functions. These tools are GDI objects such as pens, brushes, palettes, bitmaps, fonts, and regions. Support for these GDI objects in MFC is provided through a series of classes derived from CGdiObject. The CPen, CBrush, CFont, CBitmap, CPalette, and CRgn classes encapsulate the functionality of pens, brushes, fonts, bitmaps, palettes, and regions, respectively.

Serialization: File and Archive Objects CHAPTER 19

351

Serialization: File and Archive Objects

19

IN THIS CHAPTER
s The CFile Class 352 s CArchive 358 s Serialization in MFC Framework Applications 362

19
SERIALIZATION: FILE AND ARCHIVE OBJECTS

352

Microsoft Foundation Classes PART III

A concept of central importance in the Microsoft Foundation Class Library is serialization. Through serialization, objects derived from CObject obtain persistence. Before you begin wondering why such glorified terminology is used for what is essentially saving and loading data to or from a file, let me point out that serialization can take place with a target other than a disk file. It is also through serialization that a CObject-derived object is copied to or from the clipboard or passed to other applications through OLE. Serialization represents a relationship between objects derived from the CObject class, the CArchive class representing an archive, and the CFile class that represents physical storage (see Figure 19.1).

FIGURE 19.1.
Relationship of CObject, CArchive, and CFile.

CObject:: Serialize

CArchive:: operator<< operator>> ReadObject WriteObject

CFile

This relationship notwithstanding, the utility of the CFile class transcends CObject serialization. The next section presents an examination of the CFile class and shows how it can be utilized in simple scenarios.

The CFile Class


CFile

is the base class for MFC file services. As is, CFile supports unbuffered binary input/ output for disk files. Through derived classes, it supports text files, memory files, Windows sockets, and Internet file transfer. The hierarchy of CFile and its derived classes is shown in Figure 19.2.

A recurring theme with MFC classes that act as wrapper classes for Windows objects is the duality of the C++ object versus the Windows object itself. Briefly, a CFile object is not identical to a file object in Windows; it merely represents one. Construction of the CFile object does not necessarily ensure construction of a file object (that is, constructing a CFile object may or may not mean that a file is actually opened). In a CFile object, the member variable m_hFile contains (usually) the handle of the file that the object represents. This handle may be initialized in the CFile constructor or through an explicitly called initialization function.
CFile

Serialization: File and Archive Objects CHAPTER 19

353

FIGURE 19.2.
CFile

class hierarchy.

File objects CFile CMemFile

CSharedFile

COleStreamFile

CMonikerFile

CAsyncMonikerFile

CDataPathProperty

CCachedDataPathProperty

CSocketFile

CStdioFile

CInternetFile

CGopherFile

CHttpFile

19
SERIALIZATION: FILE AND ARCHIVE OBJECTS

CRecentFileList

NOTE
The file handle m_hFile is a Win32 file handle; this is not to be confused with the file handles or file descriptors used in the C/C++ low-level file I/O libraries.

354

Microsoft Foundation Classes PART III

CFile Initialization
Construction of a CFile object may be done in either one or two steps. To construct a CFile object in a single step, use the form of the constructor that accepts a handle to an already open file or the name of the file that is to be opened with the CFile object. Alternatively, you can use a parameterless constructor and call the Open member function. When you are opening a file through the CFile constructor or the Open member function, several flags can be specified. Files can be opened for reading or writing, in text or binary mode. Both the constructor and the Open member function can also create files. Additional mode flags specify file sharing and other attributes. An open file can be closed by calling the Close member function. The Abort member function can also be used for this purpose; unlike Close, Abort will close the file under all circumstances, ignoring any errors.

Reading from and Writing to a CFile Object


Quite unsurprisingly, reading and writing to/from a CFile object can be accomplished by calling the Read or Write member functions. Needless to say, the file must be opened with the appropriate mode in order for the reading or writing operation to be successful. The Flush member function can be used to force any buffered data to be written to the file. Random access to files is provided through the Seek member function. This function is used to set the position within the file for the next read or write operation. Two variants, SeekToBegin and SeekToEnd, set the position to the beginning or the end of the file, respectively. The current position can be obtained by calling GetPosition. The length of the file can be obtained through GetLength. The SetLength function can be used to set the length of the file; the file will be extended with uninitialized data or truncated as applicable.

File Management
Two static
CFile CFile::Rename

member functions can be used without constructing a CFile object. can be used to rename a file; CFile::Remove can be used to delete a file.

The status of a file can be obtained by calling the GetStatus member function. This function sets the values of a CFileStatus object. GetStatus also has a static variant that can be used to obtain the status of a file that was not opened previously. To set the status of a file from a CFileStatus object, call the SetStatus member function.

Serialization: File and Archive Objects CHAPTER 19

355

Error Handling
Many file operations can fail. Although some CFile member functions (for example, Open) indicate such failures in their return values, many other member functions throw an exception to indicate such conditions. The exception is always of type CFileException. To handle error conditions, you would write code similar to the following:
CFile myFile(filename.txt, CFile::modeWrite); try { CFile.Write(Data, 4); CFile.Close(); } catch (CFileException *e) { if (e->m_cause == CFileException::diskFull) cout << The disk is full!; else { // Handle other errors } e->Delete(); }

Locking
The CFile class also supports locking. A region of a file, as determined by the starting position and the number of bytes that are part of the region, can be locked using the LockRange member function. To unlock the region, use the UnlockRange member function. Simultaneous locking of several regions is allowed; however, locking of overlapping regions is not. Calls to UnlockRange must match exactly earlier calls to LockRange; for example, if you lock two regions of the file using LockRange, you must use two separate calls to UnlockRange even if the regions are adjacent. An attempt to lock a region of a file that is already locked results in an error.

19
SERIALIZATION: FILE AND ARCHIVE OBJECTS

Using a CFile in a Simple Application


The CFile class can be used in many situations, including console applications. Such a simple application is demonstrated in Listing 19.1. You can compile this program from the command line by typing MD-D AFXDLL hello.cpp.

Listing 19.1. Using CFile in a console application.


#include <afx.h> #define HELLOSTR Hello, World!\n #define HELLOLEN (sizeof(HELLOSTR)-1) void main(void)

continues

356

Microsoft Foundation Classes PART III

Listing 19.1. continued


{ CFile file((int)GetStdHandle(STD_OUTPUT_HANDLE)); file.Write(HELLOSTR, HELLOLEN); }

As this example also illustrates, there is little advantage to using CFile in this fashion. The real advantages of the CFile class come to light when it is used in conjunction with CArchive for MFC object serialization.

The CStdioFile Class


The CStdioFile class is used to associate a CFile-derived object with a standard C stream (that is, a FILE pointer). Its use is demonstrated with yet another simple program in Listing 19.2.

Listing 19.2. Using CStdioFile.


#include <afx.h> #define HELLOSTR Hello, World!\n #define HELLOLEN (sizeof(HELLOSTR)-1) void main(void) { CStdioFile file(stdout); file.Write(HELLOSTR, HELLOLEN); }

The stream pointer that a CStdioFile object is associated with is stored in the m_pStream member variable.
CStdioFile ReadString

objects are intended primarily for text I/O. Two additional member functions, and WriteString , support the input and output of CString objects and nullterminated text strings. The CStdioFile class does not support the CFile member functions Duplicate, LockRange, and

UnlockRange. Attempts to use these functions result in a CNotSupportedException being thrown.

The CInternetFile Class


The CInternetFile class is derived from CStdioFile and serves as the base class for CHttpFile and CGopherFile. These two classes simplify the task of requesting and reading files from Internet HTTP and Gopher servers.

The CMemFile Class


The CMemFile class supports CFile functionality in memory. One possible use of CMemFile objects

Serialization: File and Archive Objects CHAPTER 19

357

is to provide fast temporary storage. When a CMemFile object is created, you can specify a parameter that defines the amount by which the CMemFile object grows its storage at every subsequent allocation. The CMemFile class uses the standard C library functions malloc, realloc, free, and memcpy to allocate and deallocate memory and to transfer data to or from allocated memory blocks. It is possible to derive a class from CMemFile and override the default memory allocation behavior. Overridable member functions include Alloc, Free, Realloc, Memcpy, and GrowFile. A CMemFile object can also be attached to a previously allocated memory block. Use the Attach member function or the three-parameter version of the CMemFile constructor for this purpose. Note that in order to make the CMemFile object use the contents of the attached memory block, you must set the parameter controlling the growth of memory allocation to zero; in other words, a memory block attached in this fashion cannot be grown. Use the Detach member function to detach the memory block from a CMemFile object and obtain a pointer to it. To determine the size of the memory block, use the GetLength member function prior to calling Detach.
CMemFile does not support the CFile member functions Duplicate, LockRange, and UnlockRange. Attempts to use these functions result in a CNotSupportedException being thrown.

NOTE
The CSharedFile class, derived from CMemFile, can be used to create shared memory files. These use memory allocated via the Windows function GlobalAlloc and can be utilized with the clipboard, DDE, and OLE. Note, however, that neither CMemFile nor CSharedFile use memory-mapped files, and their contents cannot be directly shared between applications.

19
SERIALIZATION: FILE AND ARCHIVE OBJECTS

The COleStreamFile Class


The COleStreamFile class is associated with the OLE IStream interface. It provides CFile-like functionality on OLE streams. To construct a COleStreamFile object, pass to its constructor the pointer to an IStream interface. Alternatively, you can create a COleStreamFile object using the default constructor, and then call one of the initialization member functions. Initialization member functions include Attach (attaches the COleStreamFile object to an IStream interface), CreateMemoryStream, CreateStream, and OpenStream. To detach the COleStreamFile object from the IStream interface and obtain a pointer to that interface, call the Detach member function. Classes derived from COleStreamFile are used with ActiveX controls.

358

Microsoft Foundation Classes PART III

The CSocketFile Class


The CSocketFile class provides a CFile-like interface on Windows socket (CSocket) objects. A CSocketFile object can be attached to a CArchive object to support serialization through a socket; it can also be used as a standalone file object. Note that CSocketFile does not support several CFile member functions (such as Seek and related functions), and thus any use that assumes availability of these functions will fail. In particular, this renders the CEditView member function SerializeRaw unusable with CSocketFile objects.

CArchive
What is a CArchive object? What is its significance? Why can CObject objects not be written directly to CFile objects? Although the CFile class is a generic wrapper class for Win32 file objects, CArchive creates the link between permanent storage and the serialization functions in a CObject. In other words, CArchive allows the objects to serialize themselves. While in some cases (for example, when you are serializing an array of integers) it is enough to simply write out the memory image of the objects to permanent storage, in many other cases (for example, when the objects contain pointers) this is clearly not sufficient. By delegating the actual task of creating a persistent image to the objects themselves, the CArchive class provides an elegant solution to this problem. A CArchive object must be thought of as a one-shot or single pass entity. A CArchive is used for the sole purpose of either writing or reading a series of objects to/from permanent storage. You cannot perform random reads or writes, nor can you use the same CArchive object for both reading and writing. For example, if you wish to read back a series of objects after they have been written to permanent storage, you need to create a separate CArchive object for this purpose. Furthermore, you will have to read back the objects in the same order in which they were written to the archive originally.

Creating a CArchive
Creating and using a CArchive object is a multistep process. Before the CArchive can be created, you must have a CFile object representing a file that was opened with permissions appropriate for what you are planning to do. Once the CFile object has been created, the CArchive object can be created by passing a pointer to the CFile object to its constructor. In the constructor, you also specify whether the archive is used for reading or writing. Every CArchive object has a member variable m_pDocument that is a pointer to a CDocument object. Common use of this pointer is to refer to the document that is being serialized in MFC

Serialization: File and Archive Objects CHAPTER 19

359

framework applications. However, it is not necessary to use this member variable for this purpose (or indeed, for any purpose at all) if the objects you serialize do not depend on the presence of a valid CDocument-derived object. If you wish to obtain a pointer to the CFile object that a CArchive is associated with, call the GetFile member function. The CFile can be closed and the archive disconnected from it by calling the Close member function. Calling this function is usually not necessary, as the CFile is closed automatically when the archive is destroyed. If you do call Close, note that no further operations on the archive are permitted.

Reading and Writing Objects


The CArchive class can be used to read and write simple data types as well as CObject-derived objects. You can determine whether a CArchive object has been created for reading or writing by calling the IsLoading or IsStoring member functions. To read or write raw binary data, use the Read or Write member functions. To read or write null-terminated strings, use the ReadString or WriteString member functions. To write a CObject-derived object to the archive, call the WriteObject function. The ReadObject function creates and reads a CObject-derived object from the archive. This function uses runtime type information when creating the CObject; therefore, it is necessary that the CObject-derived class be declared and implemented using the DECLARE_DYNCREATE and IMPLEMENT_DYNCREATE macros.
CArchive SetObjectSchema

supports the concept of a schema number through the GetObjectSchema and member functions. Schema numbers enable an application to distinguish between different versions of the same archive. Using schema numbers, you can implement upward compatibility.

19
SERIALIZATION: FILE AND ARCHIVE OBJECTS

The Overloaded >> and << Operators


In many situations, applications do not call CArchive member functions directly in order to read or write an object to/from an archive. Instead, they rely on the overloaded input and output operators for this purpose. These overloaded operators have been defined for many simple types as well as the CObject type. The simple types include BYTE, WORD, LONG, DWORD, float, and double. An obvious question is why havent these operators been defined for basic C types, such as int, short, or long? The answer is that the size of these types is implementation-dependent; using them in CArchive operations would render the resulting storage object also dependent on the operating system version under which it was created. For example, in a 16-bit Windows application, the size of an int variable is two bytes; in contrast, the size of an int in 32-bit Windows is 4 bytes.

360

Microsoft Foundation Classes PART III

NOTE
Do not define your own versions of the operators << and >> to archive basic C types in a CArchive. Use type casting instead and rely on the existing operators to avoid a dependence on the operating system version.

When a simple type is being archived, the data is simply copied to or from the archive. The situation is very different when a CObject is being archived. The operators << and >> refer to the CObjects Serialize member function, passing to it a reference to the archive object. Thus, the object is responsible for serializing itself.

The CObject::Serialize Member Function


The Serialize member function in objects of type CObject is used to write an object to, or read an object from, a CArchive. This function is called with a reference to the CArchive object. The implementation of Serialize should use the CArchive::IsLoading or CArchive::IsStoring member function to determine whether the archive is used for reading or writing. A typical Serialize member function implementation looks like this skeleton:
void CMyClass::Serialize(CArchive &ar) { if (ar.IsLoading()) { // Load the data } else { // Save the data } }

In the Serialize member function, calls are often made to the >> or << operators or to the Serialize member functions of other objects. For example, if your class CMyClass contains a member variable m_other of type COtherClass (and this is also a class derived from CObject), your Serialize member function may look like this:
void CMyClass::Serialize(CArchive &ar) { m_other.Serialize(ar); if (ar.IsLoading()) { // Load the data } else { // Save the data } }

Serialization: File and Archive Objects CHAPTER 19

361

Error Handling
During the course of archive operations, many types of errors can occur. There can be a file operation error, there can be an inconsistency in the archive, and there can be memory allocation problems. Most CArchive member functions use exceptions to communicate the fact that an error occurred. member functions can throw three types of exceptions. CFileException exceptions are thrown in case of file errors. CArchiveException exceptions are thrown in case of archive problems (for example, when an object of the wrong type is being read). Finally, CMemoryException exceptions indicate memory allocation problems (for example, when the CArchive is attempting to allocate memory for an object it is about to read).
CArchive

Using CArchive in Simple Applications


Before we explore the use of CArchive in MFC framework applications, an example that demonstrates the use of CArchive in simple situations is probably in order. The program shown in Listing 19.3 uses a CArchive object to write the contents of a list to permanent storage. The list is built using the template class CList. Because CList is derived from CObject, it provides support for a Serialize member function. However, it does not support the operators << and >>. We can add this support, though, by explicitly declaring an operator<< function. This is exactly what we do for objects of type CList<WORD, WORD>.

Listing 19.3. Saving a list using a CArchive.


#include <afx.h> #include <afxtempl.h> #include <iostream.h> CArchive& operator<<(CArchive& ar, CList<WORD, WORD> &lst) { lst.Serialize(ar); return ar; } void main(void) { CList<WORD, WORD> myList; cout << Creating list: ; for (int i = 0; i < 10; i++) { int n = rand(); cout << n << ; myList.AddTail(n); } CFile myFile(mylist.dat, CFile::modeCreate | CFile::modeWrite); CArchive ar(&myFile, CArchive::store); ar << myList; }

19
SERIALIZATION: FILE AND ARCHIVE OBJECTS

362

Microsoft Foundation Classes PART III

The real power of CArchive becomes apparent when you consider that most of this code is about building a sample list; two lines are used to construct the archive object, and the entire list is written out using a single line of code. Similarly, the entire list can be read in a single line, as demonstrated by the reading program shown in Listing 19.4.

Listing 19.4. Loading a list from a CArchive.


#include <afx.h> #include <afxtempl.h> #include <iostream.h> CArchive& operator>>(CArchive& ar, CList<WORD, WORD> &lst) { lst.Serialize(ar); return ar; } void main(void) { CList<WORD, WORD> myList; CFile myFile(mylist.dat, CFile::modeRead); CArchive ar(&myFile, CArchive::load); ar >> myList; POSITION pos = myList.GetHeadPosition(); cout << Reading list: ; while (pos) { int n = myList.GetNext(pos); cout << n << ; } }

Both these programs can be compiled from the command line (for example, type MD-D readlst.cpp).

AFXDLL

Note that this simple example may be a little misleading. When using a collection template such as CList, it may be necessary to implement the SerializeElements helper function. The default implementation simply performs a bitwise read or write on elements of the collection. Although this is adequate when the elements are of type WORD, it falls short of what is required in case of more complex types (such as types derived from CObject). (Why dont the collection templates relay the Serialize member function of objects that comprise the collection? For the simple reason that these collection classes are not restricted to CObject-derived objects only, and other objects may not have a Serialize member function.)

Serialization in MFC Framework Applications


CFile

and CArchive are the building blocks; CObject::Serialize is the glue that connects objects and archives. But it is in MFC Framework applications that the concepts behind archives and serialization realize their full potential.

Serialization: File and Archive Objects CHAPTER 19

363

Serialization in Documents
In an MFC framework application, classes derived from CDocument play a central role. These classes represent the entities your applications manipulate. CDocument-derived objects achieve persistence through the serialization mechanism that we reviewed in this chapter. When AppWizard creates a skeleton framework application, it already supplies implementations for the File Open and File Save (and Save As) menu commands. These implementations create a CArchive object and call the document classs Serialize member function. It is your responsibility to supply an implementation of this member function that serializes all persistent data members of your document class.

Helper Macros
The MFC provides several helper macros that make serialization CObject-derived classes possible. When the CArchive reads data for a new object from a file, it is necessary for it to have a mechanism whereby an object of the given type can be created. This is accomplished by adding a static member function named CreateObject to the class in question. However, you do not need to declare or implement this function by hand; instead, you can use the DECLARE_DYNCREATE and IMPLEMENT_DYNCREATE macros for this purpose. How does the CArchive know the type of the object that is about to be created? Simple: together with the object, runtime type information is also saved. In order for a CObject-derived class to support runtime type information (through CRuntimeClass ), you can use the DECLARE_DYNAMIC and IMPLEMENT_DYNAMIC macros; however, as the functionality of these macros is implied by DECLARE_DYNCREATE and IMPLEMENT_DYNCREATE, it is not necessary to explicitly add these to the class declaration. Yet another pair of macros is DECLARE_SERIAL and IMPLEMENT_SERIAL. Although the documentation states that these macros are required to enable serialization, you may find that in an AppWizard-generated skeleton, the obviously serializable CDocument-derived class of your application does not use these macros. The reason for this is that DECLARE_SERIAL and IMPLEMENT_SERIAL are really only needed if you intend to use the << and >> operators with your class and a CArchive. (DECLARE_SERIAL and IMPLEMENT_SERIAL declare and implement the overloaded >> operator for your class).
DECLARE_SERIAL and IMPLEMENT_SERIAL encompass the functionality of DECLARE_DYNCREATE and IMPLEMENT_DYNCREATE ,

19
SERIALIZATION: FILE AND ARCHIVE OBJECTS

so you do not need to use those macros if IMPLEMENT_SERIAL are used.

DECLARE_SERIAL

and

364

Microsoft Foundation Classes PART III

Serialization, OLE, and the Clipboard


So far, we have discussed serialization in the context of file load and save operations. However, MFC applications also use serialization for OLE-related operations. MFC framework applications that act as OLE servers use the COleServerItem-derived class to provide a server interface. The Serialize member function of this class provides the mechanism, whereas application-specific data is stored for embedded or linked OLE objects. In the simplest implementation, this Serialize function delegates the task of serializing the document to the Serialize member function of the CDocument-derived document class. However, if the application supports serializing only portions of a document, a separate implementation may be required. Serialization of portions of a document can happen under two circumstances. First, it may happen for linked items. Second, the COleServerItem-derived class is also used for clipboard operations. If the application supports copying the users selection to the clipboard (as opposed to the entire document), the Serialize member function of the COleServerItem-derived class must provide an implementation where only the users selection is serialized.

Summary
Serialization in MFC applications represents a relationship of CObject-derived objects (those that need to be serialized), CFile-derived objects that represent persistent storage such as a disk file, and CArchive objects that provide the serialization interface. The CFile class encapsulates the functionality of a Win32 file object. Its member functions provide the means to open, read, write, and otherwise manipulate disk files. Variants of the CFile class include CStdioFile, CMemFile, COleStreamFile, and CSocketFile. These classes provide I/O functionality through C-style stream objects (FILE pointers), memory blocks, OLE IStream interfaces, and Windows sockets. The CArchive class provides the basic interface for serialization. Serialization is a mechanism that enables CObject-derived classes to assume responsibility for writing or reading their own data to/from persistent storage. CArchive accomplishes this by calling the Serialize member function for CObject-derived objects whenever data transfer takes place. The
CObject::Serialize

member function must be implemented for classes derived from

CObject. In this function, data is written to or read from a CArchive object, a reference to which

is passed to the function as its sole parameter. The direction of the operation, namely whether it is a save to or load from the archive, can be determined by calling the CArchive objects IsLoading member function.

Serialization: File and Archive Objects CHAPTER 19

365

Inside Serialize, member variables of the class are transferred to or from the archive. This can be accomplished by using the << or >> operators, by calling the member variables Serialize member function (if the member variable is of a type derived from CObject), or by calling the CArchive::Read or CArchive::Write functions for bitwise transfer of data. Serialization is used throughout in MFC framework applications. The framework provides a default implementation for the File Open and File Save commands. These default implementations call your document classs Serialize member function. This function, which you must implement yourself, should serialize all your documents persistent data. In order for a CObject-derived class to be serializable, it must be declared using the DECLARE_SERIAL macro and implemented using IMPLEMENT_SERIAL. If you do not plan to use the overloaded >> operator with your class, you may declare it using DECLARE_DYNCREATE and IMPLEMENT_DYNCREATE. For an example of a class with this behavior, take a look at any document class created by AppWizard.

19
SERIALIZATION: FILE AND ARCHIVE OBJECTS

366

Microsoft Foundation Classes PART III

Collection Classes CHAPTER 20

367

Collection Classes
IN THIS CHAPTER

20

s CObject Collections 368 s Other List Collections 373 s Other Array Collections s Mappings 375 374

s Template-Based Object Collections 379

20
COLLECTION CLASSES

368

Microsoft Foundation Classes PART III

The Microsoft Foundation Classes Library implements a number of collection classes. These collection classes include a variety of lists, arrays, and mappings. Collections have been supported by the MFC Library since its early versions. In the days of Visual C++ 1.5 and earlier versions, template support was not yet available in the compiler; correspondingly, there are several non-templatebased collection classes in the Library. Newer, 32-bit versions of the compiler obviously provide full template support. Accordingly, starting with MFC 3.0, a series of new type-safe collection classes was introduced. However, they do not render the old classes obsolete; if you have been using the non-templatebased collection classes in your code, there is no reason why you should not continue to do so. In new code, it is recommended that you try to use the new, template-based classes, as their typesafe nature provides for safer, more robust code. Throughout the MFC Library, the most commonly used collections are collections of CObject items. This fact, plus the relative simplicity of the CObject collection classes, provides a good reason for starting our review of collection classes with CObList and CObArray.

CObject Collections
The MFC Library provides two ordered collections of CObject items. The CObList collection is a list of CObject items; the CObArray collection is an array of CObject items. But, you may ask, what exactly is the difference between the two? What are the advantages of using one or the other? The CObList class organizes CObject pointers in a linked list. Due to the nature of such a list, insertion and removal of elements are very fast operations. On the other hand, the list is not indexed; retrieving an element by a numerical index is a slow process. List collections are meant to be used in situations where the list is walked with the help of an iterator. Consider, for example, a typical use of a CObList collection, namely to organize all items in a document. When the items are accessed in order to serialize or to draw the document, elements in the list are accessed sequentially. The CObArray class, in contrast, indexes elements by an integer index value. Inserting or removing elements are slow operations, as they involve moving potentially large blocks of data. However, retrieving data by the integer index is fast. Although both classes are nominally collections of CObject pointers, you will probably never use them with the CObject class. While technically not an abstract class, the CObject class is pretty useless by itself. Instead, CObList and CObArray are used as collections of CObjectderived classes, such as collections of items of type CWnd, CView, or CDocItem. In the following sections, I present several small examples of code in which objects of type CObject are declared; note that these code fragments will not compile as is, as creation of a CObject is prevented by the declaration of a protected constructor in the MFC Library. When applying these examples to concrete situations, just substitute a CObject-derived class in these examples and assume any required typecasts.

Collection Classes CHAPTER 20

369

The CObList Class and the POSITION Type


Using a CObList collection involves creating the collection, adding items to the collection, and accessing items through iterator functions. A CObList can be created either by declaring a variable of type CObList or using the new operator to obtain a pointer to a new CObList collection. Items can be added to a CObList by calling the AddHead or AddTail functions. As their names imply, these functions add an element at the beginning and at the end of the list, respectively. Items do not need to be unique; the same CObject pointer can occur any number of times in the list. To obtain the first or the last element on the list, use the GetHead or GetTail member functions. It is also possible to add a new element to the list at a specific position. The InsertBefore and InsertAfter member functions can be used to insert a new element before or after a specific element. The element position is identified by a variable of type POSITION. This type is used throughout collection classes as a general-purpose iterator type. can be obtained by calling either the GetHeadPosition or the member functions. The returned POSITION value can be used in subsequent calls to GetNext or GetHead to access elements of the list sequentially. For example, to walk the elements in a CObList from the beginning to the end of the list, you would use code similar to the following: A value of type
POSITION GetTailPosition CObList myList; ... // Populate the list ... POSITION pos = myList.GetHeadPosition(); while (pos != NULL) { CObject *pObject = myList.GetNext(pos); ... // Do something nasty to *pObject ... }

Similarly, you can walk the elements of a CObList backwards as follows:


CObList myList; ... // Populate the list ... POSITION pos = myList.GetTailPosition(); while (pos != NULL) { CObject *pObject = myList.GetPrev(pos); ... // Do something nasty to *pObject ... }

20
COLLECTION CLASSES

370

Microsoft Foundation Classes PART III

As you can see from these two examples, the names of the GetNext and GetPrev functions can be slightly misleading. What these functions do is return the current element pointed to by the POSITION parameter, while at the same time advancing this parameter to refer to the next (or previous) element. A POSITION value can also be used with the GetAt, SetAt, and RemoveAt member functions. All three of these functions take a parameter of type POSITION; GetAt retrieves a CObject pointer corresponding to that position, SetAt sets the element at the given position to a new value, and RemoveAt removes the element from the list altogether. Removing an element during an iteration may cause problems. In order to ensure that you always maintain a valid POSITION value for GetNext, you should use a method similar to the following:
POSITION pos = myList.GetHeadPosition(); while (pos != NULL) { POSITION pos2 = pos; CObject *pObject = GetNext(pos); if ( /* pObject is to be removed */ ) { myList.RemoveAt(pos2); } }

Additional functions that can be used to remove elements are RemoveHead and RemoveTail. The RemoveAll function can be used to remove all elements (empty the list). Removing an element does not destroy the element; the program that created the element is responsible for its destruction. For example,
CObject *pObject; CObList myList; ... pObject = new CObject; myList.AddTail(pObject); ... // some time later ... pObject = myList.RemoveTail(); delete pObject;

To determine if a list is empty, use the IsEmpty member function. To obtain the number of elements in the list, call GetCount. You can also search for a specific element in the list. The Find member function determines whether a particular CObject pointer is in the list; if so, it returns a value of type POSITION indicating its first occurrence. The FindIndex member function returns the POSITION value that corresponds to a given numerical index. Note that as the CObList class does not maintain an index of any kind, these operations can be slow if the list is large.

Collection Classes CHAPTER 20

371

The CObList type is itself derived from CObject. As such, the CObList class supports serialization. If its Serialize member function is called, it in turn serializes every CObject element in the list using the << and >> operators. In order to ensure that the list is serialized correctly, elements added to it should be of a CObject-derived type that is declared and implemented using the DECLARE_SERIAL and IMPLEMENT_SERIAL macros.
CObList

objects can be used in conjunction with the CArchive class and the << and >> operators, as in the following example:

class CMyDocument : public CDocument { CObList m_List; // rest of the class declaration follows ... } void CMyDocument::Serialize(CArchive &ar) { if (ar.IsStoring()) { ar << m_List; } else { ar >> m_List; } }

Because CObList is also CObject-derived, it is possible to use a CObList collection to refer to items of type CObList, in effect creating a list of lists.

The CObArray Class


CObArray objects are arrays of CObject pointers. These arrays are similar in function and behav-

ior to C arrays, with one crucial difference: a CObArray can grow or shrink dynamically. Using a CObArray involves constructing the CObArray object, populating it with CObject pointer elements, and retrieving array elements (possibly by using the overloaded [] operator). You can create a CObArray like you would create any other variable, either on the stack as an automatic variable or through the new operator. At the heart of the CObArray class are the SetAt, GetAt, and SetAtGrow member functions. SetAt and GetAt behave as you would expect, setting and retrieving an element at the specified location. SetAtGrow also sets an element at the specified location; however, this function causes the array to grow if the location is past the current array bounds. Neither SetAt nor GetAt report an error if an invalid index is specified. However, they do cause an assertion in the debug version of the MFC Library. Elements can also be added to the array using the Add member function. This member function appends a new element to the array, growing the array as necessary.

20
COLLECTION CLASSES

372

Microsoft Foundation Classes PART III

The current number of elements in the array can be obtained by calling GetSize. The largest valid array index (which is equal to the number of array elements minus one, as array indexes are zero based) can be obtained by calling GetUpperBound. The SetSize function can be used to set the number of elements in the array, allocating additional memory if necessary. If SetSize is used to shrink the array, unused memory will be freed. Whenever the array is grown as a result of a call to SetAtGrow, Add, or SetSize, a memory allocation error may occur. These errors are indicated by an exception of type CMemoryException being thrown. The SetSize function can also be used to specify the amount by which memory allocated by CObArray is grown. Whenever new memory needs to be allocated, the amount allocated will hold as many CObject pointers as specified in the second parameter of SetSize. If this parameter is not set, the CObArray class attempts to determine the optimum amount of memory that it should allocate to avoid heap fragmentation. Any extra memory allocated when the array was grown can be released by calling FreeExtra. The entire array can be emptied and all memory released by calling RemoveAll. The CObArray class provides an override version of the [] operator. Through this operator, array elements can be accessed. The operator can be used in situations where an lvalue is needed (for example, on the left side of an assignment operation). This behavior is implemented with the help of the ElementAt member function, which returns a reference to the CObject pointer at the specified location. Thus, the following line:
myArray[10] = &myObject;

is equivalent to
myArray.ElementAt(10) = &myObject;

Two member functions, InsertAt and RemoveAt, can be used to insert elements into the array or to remove an element at a specific index. Note that these operations are slow; in the case of large arrays, they potentially require the moving of large blocks of data. As with CObList, the CObArray class does not destroy any elements when they are removed from the array. You are responsible for freeing such items yourself. For example,
CObject *pObject; CObArray myList; ... pObject = new CObject; myArray.Add(pObject); ... // some time later ... pObject = myArray[myArray.GetUpperBound()]; myArray.SetSize(myArray.GetUpperBound()); delete pObject;

Collection Classes CHAPTER 20

373

The CObArray type is derived from CObject. One advantage of this fact is that CObArray collections can also be serialized. When the CObArray::Serialize member function is called, it in turn serializes array elements using the << and >> operators. To support serialization, elements added to the array must be of a CObject-derived type that is declared and implemented using the DECLARE_SERIAL and IMPLEMENT_SERIAL macros.
CObArray

collections can be used with the CArchive class and the << and >> operators.

Other List Collections


There are several other list collections with features and behavior very similar to that of CObList.

The CPtrList Class


The CPtrList class implements identical behavior to that of CObList, but for elements that are void pointers. The features and member functions of this class are otherwise identical to the features and member functions of CObList. Note that the CPtrList class does not support serialization.

The CStringList Class


The CStringList class is also similar in behavior and implementation to CObList. However, there is one crucial difference; instead of storing pointers to items of type CString, a CStringList stores copies of the actual CString objects themselves. As a consequence, the application is no longer required to explicitly delete a CString object after removing it from the CStringList. Nor is it necessary to allocate an element using new or malloc to ensure that an object remains valid after the function in which it was declared terminates. For example, consider the following (incorrect) function implementation:
void MyAddElement(CObList *pList) { CObject myObject; pList->AddTail(&myObject); // WRONG! }

This is obviously wrong, as the address &myObject becomes meaningless when the function returns. Instead, the following implementation should have been used:
void MyAddElement(CObList *pList) { CObject *pObject; pObject = new CObject; pList->AddTail(pObject); }

20
COLLECTION CLASSES

The same problem does not present itself when using a CStringList. The following implementation is correct:

374

Microsoft Foundation Classes PART III


void MyAddElement(CStringList *pList) { CString myString; pList->AddTail(myString); }

Serialization of CStringList collections is supported.

Other Array Collections


In addition to CObArray, the MFC Library provides a series of additional ready-to-use array collections. Analogous to the list collections CPtrList and CStringList are the array collections CPtrArray and CStringArray; however, there are also a series of additional array collections that hold integral types.

The CPtrArray Class


The CPtrArray class implements the same behavior as CObArray but for void pointer elements. The behavior of member functions and the features of the class are identical to the features and member function behavior of CObArray. Serialization is not supported by CPtrArray.

Integral Array Classes


There are several array classes in MFC that store elements of integral types. These are summarized in Table 20.1.

Table 20.1. Integral array collection classes. Class name Element type
CByteArray CWordArray CDWordArray CUIntArray BYTE WORD DWORD UINT

The type CUIntArray differs from the other three types in that the size of a UINT is implementationdependent. Under 16-bit Windows, a UINT is 16 bits wide; under Windows NT or Windows 95/98, it is a 32-bit type. Consequently, unlike the other three types, CUIntArray does not support serialization. The other three collection classes use element types that are guaranteed to be of the same size on different implementations. A BYTE is always 8 bits wide; a WORD, 16 bits, and a DWORD is always 32 bits.

Collection Classes CHAPTER 20

375

With the exception of the difference in serialization support by CUIntArray, the features and behavior of these classes is identical to the features and behavior of CObArray.

The CStringArray Class


The CStringArray class represents an array of CString objects. As in the case of CStringList, CStringArray stores copies of the CString objects themselves, not just pointers to them. Consequently, application programmers are not responsible for destroying CString objects that are removed from the array. The features and behavior of CStringArray are otherwise similar to the features and behavior of CObArray. In particular, CStringArray supports serialization.

Mappings
Mappings represent a type of a collection that is markedly different from lists or arrays. Lists and arrays are ordered collections; in contrast, mappings represent unordered mappings of key objects to value objects. Because of the obvious similarity, mappings are sometimes also referred to as dictionaries (and indeed, implementing the functionality of a dictionary of words is a trivial task using, for example, the mapping collection CMapStringToString). Mappings are tailored towards fast searches by key value. In all the mapping classes, key values are expected to be unique. An attempt to set a value with an existing key will overwrite the current entry in the mapping as opposed to creating an entry with a duplicate key value. The MFC Library offers several map collections. Keys that are pointers, strings, or 16-bit words are used to index items that are pointers to CObject, pointer to void, strings, or 16-bit words. Because not all combinations of these types are implemented in the form of a mapping collection, and because there are minor variations and differences in the behavior of these classes, I review them individually.

The CMapStringToString Class


The CMapStringToString class maps keys of type CString to values of the same type. To construct a CMapStringToString collection, simply declare an object of this type or use the operator. Once the collection has been constructed, key-value pairs can be added to it using the SetAt member function. A convenient shorthand for using SetAt is the overloaded [] operator. Curiously, this operator can only be used in the place of an lvalue; it cannot be used for looking up keys for the simple reason that key values are often not found in the collection.
new

20
COLLECTION CLASSES

To look up data by key value, use instead the Lookup member function. The Boolean return value of this function indicates whether the key was found or not.

376

Microsoft Foundation Classes PART III

I suppose it would be possible to implement an overloaded form of the [] operator that can be used on the right-hand side of assignments and use an exception to communicate a lookup failure. However, an unhandled exception would cause your application to terminate even though the failure to find a key in an index is normal behavior. To remove a key-value pair from the collection, use the RemoveKey member function. You can also remove all key-value pairs and thus empty the collection using RemoveAll. You can find out if a collection is empty by calling the IsEmpty member function. The GetCount member function returns the number of key-value pairs in the collection. It is also possible to iterate through a mapping. The GetStartPosition member function yields an iterator of type POSITION that can be used in subsequent calls to GetNextAssoc to obtain keyvalue pairs. Note that the order in which elements are returned is arbitrary and has no significance. In particular, these functions are not guaranteed to yield elements in ascending key order.
CMapStringToString CArchive

collections can be serialized. They can be used in conjunction with the class and the << and >> operators.

Listing 20.1 shows a simple, yet functional program that implements a word vocabulary. This command-line application can be compiled from the command line (cl /MT vocab.cpp).

Listing 20.1. Using CMapStringToString in a console application.


#include <afx.h> #include <iostream.h> #include <stdio.h> void main(void) { CString wrd, def; CMapStringToString map; CStdioFile inf(stdin); cout << Populating the dictionary\n; while (TRUE) { cout << Enter word: ; cout.flush(); inf.ReadString(wrd); if (wrd == Q) break; cout << Definition: ; cout.flush(); inf.ReadString(def); map[wrd] = def; } if (map.IsEmpty()) { cout << Empty vocabulary!\n; exit(0); } cout << Vocabulary populated with ; cout << map.GetCount() << elements.\n;

Collection Classes CHAPTER 20


cout << Looking up words in the dictionary\n; while (TRUE) { cout << Enter word: ; cout.flush(); inf.ReadString(wrd); if (wrd == Q) break; if (map.Lookup(wrd, def)) cout << def << \n; else cout << not found!\n; } }

377

This program allocates a CMapStringToString collection and then enters an input loop. In this loop, corresponding word-definition pairs are entered by the user and added to the collection. The loop terminates when the user enters a capital Q for the word. At this time, after displaying the size of the collection, the program enters a second loop. In this loop, the user enters words that are to be looked up in the vocabulary. Here is a sample session with this program:
Populating the dictionary Enter word: mouse Definition: small, nocturnal rodent Enter word: cat Definition: small, domesticated carnivore Enter word: dog Definition: large, supposedly domesticated ugly animal that barks Enter word: Q Vocabulary populated with 3 elements. Looking up words in the dictionary Enter word: cat small, domesticated carnivore Enter word: mouse small, nocturnal rodent Enter word: rat not found! Enter word:

As I said, implementing a dictionary with these dictionary collections is indeed a trivially simple task.

The CMapStringToOb Class


The CMapStringToOb class maps objects of type CString to CObject pointers. That is, it uses string indexes to maintain a collection of CObject items. The obvious use of this class is to create a named set of CObject items. The features and behavior of this class are almost identical to the features and behavior of CMapStringToString, with one crucial difference. As this class stores CObject pointers, it is the programmers responsibility that any CObject items that are removed from the collection are destroyed. For example,

20
COLLECTION CLASSES

378

Microsoft Foundation Classes PART III


CObject *pObject; CMapStringToOb myMap; ... pObject = new CObject; myMap[myObject] = pObject; ... // some time later ... if (myMap.Lookup(myObject, pObject)) { myMap.RemoveKey(myObject); delete pObject; }; CMapStringToOb

can also be serialized and used with the

CArchive

class, and the

<<

and

>>

operators.

The CMapStringToPtr Class


The CMapStringToPtr class maps objects of type CString to pointers to void. This class can be used to provide a collection of named items of arbitrary type. Like the CMapStringToOb class, this class also stores pointers to items and does not free the items when the pointers are removed from the collection. It is the application programmers responsibility to destroy the items the pointers point to. Unlike the CMapStringToOb class, CMapStringToPtr collections cannot be serialized. In all other respects, the features and behavior of CMapStringToPtr are identical to the features and behavior of CMapStringToOb.

The CMapPtrToPtr Class


The CMapPtrToPtr class maps void pointers to void pointers. Note that it is the pointer value that serves as a key to this collection, not the entities that these pointers refer. Thus, two pointers that refer to two identical but distinct objects will be treated as unequal keys by CMapPtrToPtr. For example, consider the following code:
CMapPtrToPtr myMap; int a, b, x, y; a = b = 123; myMap[&a] = &x; myMap[&b] = &y;

Although a and b are equal, &a and &b are not; consequently, this code adds two distinct keyvalue pairs to the collection. When a key-value pair is removed from a CMapPtrToPtr collection, the application is responsible for destroying both entities that the two pointers (the key and the value) refer to.

Collection Classes CHAPTER 20

379

The CMapPtrToWord Class


The
CMapPtrToWord CMapPtrToPtr,

class maps void pointers to values of type WORD . Note that as with it is the pointer value, not the entity it points to, that serves as the key to this

collection. When removing a key-value pair from this collection, applications should ensure that the entities the keys point to are appropriately destroyed.

The CMapWordToOb Class


The CMapWordToOb class maps an index of type WORD to items that are CObject pointers. What is the difference between this class and a CObArray? In an array collection, indexes are assumed to start at zero and be consecutive. In contrast, the WORD indexes in a CMapWordToOb collection can be arbitrary. For example, to use the indexes 1 and 100 in a CObArray collection requires allocating memory for 101 elements; the same two indexes in a CMapWordToOb only occupy two slots in the collection. Collections of type CMapWordToOb support serialization and work in conjunction with the CArchive class, and the << and >> operators.

The CMapWordToPtr Class


The CMapWordToPtr class maps an index of type WORD to items that are void pointers. The features and behavior of this class are identical to the features and behavior of CMapWordToOb with one exception: CMapWordToPtr does not support serialization.

Template-Based Object Collections


The collection classes that we have reviewed thus far are not type safe. Allow me to elaborate on this point. Consider, for example, how a collection of CWnd objects would be implemented using CObList. Items that are CWnd pointers would be added to the list in a fashion similar to the following:
CWnd *pWnd; CObList myList; ... myList.AddTail((CObject *)pWnd); ... pWnd = (CWnd *)(myList.GetHead());

20
COLLECTION CLASSES

Because of the typecast in the call to AddTail, the collection has no way of verifying that the object passed to it is indeed of the correct type. Similarly, when the item is retrieved from the collection, it is always a pointer to the CObject type. If, due to a programming error, a pointer of another CObject-derived type is passed to the collection, there will be no errors, no compiler

380

Microsoft Foundation Classes PART III

warnings, but the application will silently fail. For example, you can add a pointer of type CDocument to the collection:
CDocument *pDoc; ... myList.AddTail((CObject *)pDoc);

and not notice a thing; only later, when you retrieve this pointer assuming it is a pointer to a CWnd object, will your program show hard-to-analyze signs of misbehavior. Type-safe collection templates provide a solution to this problem. By declaring the collection as follows:
CTypedPtrList<CObList, CWnd *> myList;

one can eliminate the need for typecasts and thus ensure that if anything other than a CWnd pointer is added to the collection, the compiler will indicate an error.

NOTE
It is also possible to derive type-safe versions from non-template collection classes by adding properly typed wrapper functions. However, the templates discussed here represent a more general approach.

There are two types of template collections. The first category consists of simple arrays, lists, and mappings; the second category consists of arrays, lists, and maps of typed pointers. Members of the first category are the CList, CArray, and CMap templates; members of the second category include CTypedPtrList, CTypedPtrArray, and CTypedPtrMap.

Collection Class Helper Functions


The simple collection templates CList, CArray, and CMap use seven collection class helper functions. Implementing these functions may be necessary in order for these classes to provide expected behavior. For construction of elements, the collection classes use the ConstructElements helper function. ConstructElements is called after memory has been allocated for the new elements. The default implementation uses the constructor of type TYPE to create the elements. This function is used by all three simple collection templates when memory for new elements is allocated. The function DestructElements is called before memory allocated for elements in the collection is deallocated. The default implementation of this function uses the destructor of type TYPE to deinitialize collection elements. This function is also used by all three simple collection templates. The CompareElements function compares two elements for equality. The default implementation uses the == operator for this purpose. This function is used by the function CList::Find and by CMap-based collections.

Collection Classes CHAPTER 20

381

The CopyElements function copies elements. The default implementation performs a bitwise copy (hardly adequate in many situations). This function is used by the CArray::Append and CArray::Copy member functions. The SerializeElements helper function serializes elements in the collection. The default implementation performs bitwise serialization (again, this is hardly adequate in many cases). Override this function, for example, when you wish to call the Serialize member function of your collection elements instead. The HashKey helper function is used by CMap-based collections to create a hash key. The default implementation creates a hash key by right-shifting the key value by four bit positions. Override this member function if you wish to use a hash key that is more appropriate for your application. Finally, the DumpElements member function is used to create a diagnostic dump of collection elements. The default implementation of this function does nothing. Override this function, for example, if you wish to call the Dump member function of the collection elements instead.

The CList Template


The CList template is used to create lists of a given element type. A list is an ordered collection of items; it supports access to these items using an iterator. The CList template takes two parameters. It is declared as follows:
template<class TYPE, class ARG_TYPE> class CList : public CObject { ... };

Of the two parameters, TYPE represents the type of elements that the list consists of; ARG_TYPE represents the type used in function arguments. ARG_TYPE is often a reference to TYPE. For example, a list of CString objects could be declared as follows:
CList<CString, CString&> myList;

Although the behavior of a CList and a CObList are similar, note one fundamental difference: a CList stores objects of type TYPE, not pointers to those objects. In the previous example, for every CString that is added to the list, a copy of the item is created. A CList collection is constructed when it is declared. Elements of type TYPE are added to the collection using the AddHead or AddTail member functions. You can also add elements at a given position, identified by a POSITION value using InsertBefore and InsertAfter. An iterator of type POSITION can be obtained by calling GetHeadPosition or GetTailPosition. Iterating through elements in the collection can be done by repeatedly calling GetNext or GetPrev, as in the following example:

20
COLLECTION CLASSES

382

Microsoft Foundation Classes PART III


CList<CString, CString&> myList; ... // Populate the list ... POSITION pos = myList.GetHeadPosition(); while (pos != NULL) { CString str = GetNext(pos); ... // Do something ugly with str ... }

The head and tail element of the list can be obtained using the GetHead and GetTail member functions. A POSITION value can also be used in calls to GetAt, SetAt, and RemoveAt. These member functions obtain an element at a given position, set the element at a given position to a new value, and remove an element at a given position. The head or tail of the list can be removed by calling RemoveHead or RemoveTail. The entire list can be emptied by calling RemoveAll. To find out if the list is empty, call the IsEmpty member function; GetCount can be used to obtain the number of elements in the list. Elements can be searched for by numerical index using the FindIndex function, and by value using the Find function. Note that you may need to provide an override version of CompareElements in order for the Find member function to work correctly. The CList template supports serialization. In order for serialization to work properly, it may be necessary to provide an override version of the SerializeElements helper function.

The CArray Template


The CArray template is used to create a dynamically allocated array of a given element type. An array is a collection of elements accessed through a zero-based integer index. The function and behavior of CArray are identical to the function and behavior of C arrays, with the important exception that a CArray can dynamically grow and shrink. The CArray template takes two parameters. It is declared as follows:
template<class TYPE, class ARG_TYPE> class CArray : public CObject { ... };

The TYPE parameter represents the type of items that this collection consists of; the ARG_TYPE represents the argument type passed to functions. Often, ARG_TYPE is a reference to type. For example,
CArray<CString, CString&> myArray;

Collection Classes CHAPTER 20

383

Despite the many similarities, there is a fundamental difference between the behavior of CArray and the non-templatebased array collection CObArray. CArray stores copies the items themselves as opposed to pointers to items, as is the case with CObArray. After declaring and thus constructing a CArray object, you can use the SetSize member function to set its size. To set an element at a given index, use the SetAt member function; to obtain an element at a given index, use GetAt. SetAt will not grow the array if an index is specified that is out of bounds. However, you can use SetAtGrow for this purpose. You can also add elements to the array and grow the array as necessary by calling the Add member function. The [] operator is a shortcut for the SetAt and GetAt member functions. It can be used on both sides of an assignment operation. When used as an lvalue, it utilizes the ElementAt member function that retrieves a reference to the specified element. The SetSize function can also be used to define the amount by which memory allocated for the array grows when additional memory is allocated. The default implementation attempts to use an optimal value to minimize heap fragmentation. Any extra memory thus allocated can be freed by calling the FreeExtra member function. The current size of the array can be obtained by calling GetSize. The GetUpperBound function returns the maximum allowable index in the array (which is one less than the arrays size). It is possible to insert elements at a given location or remove an element at a given location using the InsertAt and RemoveAt functions. However, these operations may involve moving large chunks of data and thus tend to be slow. Elements from another array (of the same type) can be copied into the array at a specified index position using the Copy member function, or appended to the end of the array using the Append member function. Proper operation of these functions may require that you provide an overloaded version of the CopyElements helper function. The CArray class supports serialization. Proper serialization behavior may require that you provide an overloaded implementation of SerializeElements.

The CMap Template


The CMap collection template provides an indexed collection of key-value pairs. CMap is declared as follows:
template<class KEY, class ARG_KEY, class VALUE, class ARG_VALUE> class CMap : public CObject { ... }; KEY and VALUE represent the types of keys and values; ARG_KEY and ARG_VALUE represent types passed as function arguments. Often, ARG_KEY is a reference to KEY and ARG_TYPE is a reference to TYPE, as in the following example: CMap<CString, CString&, CString, CString&> myMap;

20
COLLECTION CLASSES

384

Microsoft Foundation Classes PART III

An efficient implementation of a CMap-based collection may require that you provide a version of the HashKey function overloaded for your KEY type. To use a CMap-based collection, construct it by declaring it. Key-value pairs can be added to the collection by calling the SetAt member function. The [] operator is a shortcut for this function. It can only be used in this situation; because not every key value is expected to be found in the collection, the [] operator cannot be used on the right-hand side of assignment expressions (in other words, as something other than an lvalue). Elements in the collection can be found using the LookUp member function. An element identified by a given key can be removed using the RemoveKey member function; to remove all elements (empty the collection), call RemoveAll. It is possible to iterate through the collection. An iterator of type POSITION can be obtained by calling GetStartPosition ; elements can be obtained one by one by repeatedly calling GetNextAssoc. The order in which the elements are returned is arbitrary and is not expected to match the key order. To obtain the number of elements, call GetCount. Call IsEmpty to determine whether the collection has any elements. Two additional functions, InitHashTable and GetHashTable, can be used to initialize the collections hashing table to a given size and to retrieve the hashing tables size.

The CTypedPtrList Template


The CTypedPtrList template provides a type-safe list of pointers by implementing a template wrapper for the non-templatebased classes CObList and CPtrList. CTypedPtrList is declared as follows:
template<class BASE_CLASS, class TYPE> CTypedPtrList : public BASE_CLASS { ... };

The type BASE_CLASS should be either CObList or CPtrList. If CObList is used, TYPE must represent a pointer to CObject-derived class; if CPtrList is used, TYPE can be any kind of a pointer.
CTypedPtrList works by providing wrapper functions for all CObList or CPtrList member func-

tions that refer to the collection elements by type. The wrapper functions perform any necessary type casting. Otherwise, the behavior of CTypedPtrList is identical to that of CObList or CPtrList. In particular, CTypedPtrList supports serialization of it is used in conjunction with CObList; however, serialization is not supported when CPtrList is used.

Collection Classes CHAPTER 20

385

The CTypedPtrArray Template


The CTypedPtrArray collection template provides a type-safe array of pointers. This template is a wrapper for the non-templatebased collections CObArray and CPtrArray. It is declared as follows:
template<class BASE_CLASS, class TYPE> CTypedPtrArray : public BASE_CLASS { ... };

The BASE_CLASS type should be either CObArray or CPtrArray. TYPE represents a pointer type; this must be a pointer to a CObject-derived type if CObArray is used as the BASE_CLASS but can be any pointer type if CPtrArray is used.
CTypedPtrArray works

by providing a wrapper function for every CObArray or CPtrArray function that refers to collection elements by type. The wrapper functions also perform all necessary type casting. Serialization is supported by CTypedPtrArray-derived classes if they are based on CObArray.

The CTypedPtrMap Template


The CTypedPtrMap template provides type-safe mappings. It is a wrapper template for the mapping collection classes CMapPtrToPtr, CMapPtrToWord, CMapWordToPtr, and CMapStringToPtr. Type-safe behavior is provided by implementing wrapper functions for base class member functions that reference the type of elements.
CTypedPtrMap-based

classes do not support serialization.

Summary
The Microsoft Foundation Classes Library provides a series of collection classes. There are several non-templatebased collections, and also several type-safe collection templates. Perhaps the most widely used collection classes are collections of CObject pointers. The CObList collection represents an ordered list (linked list) of CObject pointers and is used frequently for storing, for example, lists of windows in the MFC. Elements in the collection are accessed through an iterator of the special type POSITION. The other CObject pointer collection is CObArray; the function and behavior of this type of collection are similar to the function and behavior of C arrays with one crucial difference: a CObArray can dynamically grow and shrink. Both CObArray and CObList are serializable collections. Other list collections include CPtrList (a list of void pointers) and CStringList (a list of CString items). Of these two, CPtrList does not support serialization.

20
COLLECTION CLASSES

386

Microsoft Foundation Classes PART III

Other array collections include CPtrArray, CStringArray, and a variety of integral array types. The CPtrArray class does not support serialization. In addition to lists and arrays, the MFC Library also supports mappings. Mappings are unordered collections of key-value pairs indexed by the key. A variety of mapping classes provides support for keys of type CString, WORD, and pointers to void; values can be of type CString, WORD, pointers to void, and CObject pointers (not all combinations are supported). With the exception of mappings where either the key or the value (or both) are void pointers, mapping classes also support serialization. Ever since template support was introduced in Visual C++, the MFC Library supports typesafe collection templates. Two types of collection templates exist: Simple templates support collections of a specific type, and typed pointer templates support type-safe collections of pointers. The simple collection templates rely on several overridable helper functions to work correctly. These include SerializeElements, which is used when collection items are serialized, and CompareElements, which is used when collection items are searched for by value. Other helper functions are used for construction and destruction, element copy, diagnostic dumping, and hashing. The simple collection templates include CList (linked list collection), CArray (dynamically allocated array), and CMap (key-value mappings). The pointer-based collection templates include CTypedPtrList, CTypedPtrArray, and CTypedPtrMap. The simple collection templates can be used with any type. They support serialization through the helper function SerializeElements. The pointer-based collection templates rely on non-template collections for their behavior. Specifically, they build upon the behavior of CObList and CPtrList, CObArray and CPtrArray, and any of the pointer-based mapping classes (CMapPtrToPtr, CMapPtrToWord, CMapWordToPtr, and CMapStringToPtr). Serialization is supported by CTypedPtrList and CTypedPtrArray if they are used in conjunction with CObList and CObArray, but not when they are used in conjunction with CPtrList or CPtrArray. Serialization of pointer-based mapping templates is not supported.

Internet Support Classes CHAPTER 21

387

Internet Support Classes

21
INTERNET SUPPORT CLASSES

21

IN THIS CHAPTER
s The MFC Internet Class Architecture 388 s Using MFC Internet Classes in Applications 392

388

Microsoft Foundation Classes PART III

Microsofts turnaround since the days when the Internet appeared to be an exotic foreign concept to the companyits firm commitment to support Internet technologies in all its productsis reflected in recent changes and additions to the MFC library. For some time, MFC has offered support for low-level WinSock operations. However, with MFC 4.2 Microsoft introduced a series of Internet support classes that significantly simplify some common Internetrelated programming tasks. These classes include CInternetSession, CInternetConnection, and derived classes; the CStdioFile-derived class CInternetFile and its descendants; and several other support classes.

The MFC Internet Class Architecture


The MFC Internet support classes utilize the new Microsoft Win32 Internet (WinInet) library for Internet support. Built on top of the venerable Windows Sockets TCP/IP implementation, this library provides conceptually higher-level support for common Internet programming tasks. Specifically, the WinInet library provides extensive support for FTP, Gopher, and HTTP connections. These three types of connections are supported by corresponding MFC wrapper classes: CFtpConnection, CGopherConnection, and CHttpConnection. Additional classes implement the concept of an Internet file, through which transfers to and from a server can be accomplished using standard methods. Figure 21.1 illustrates Internet classes.

FIGURE 21.1.
Internet support classes.

CInternetSession

CInternetFile

CInternetConnection

CGopherFile

CFtpConnection

CHttpFile

CGopherConnection

CInternetException

CHttpConnection

CFileFind

CFtpFileFind

CGopherFileFind

CGopherLocator

Internet Support Classes CHAPTER 21

389

Building an Internet connection typically requires three MFC objects. An object of type CInternetSession is used to initialize the WinInet library. A connection object represents the connection between the client computer and the Internet server. Finally, a file object is used to represent the actual data transfer.

21
INTERNET SUPPORT CLASSES

Internet Sessions
You establish an Internet session by creating a CInternetSession object. Creating an object of this type initializes the WinInet library and sets up data structures internal to MFC. The CInternetSession::CInternetSession constructor function takes several parameters. Two of these, the third and the fifth, are of particular interest. The third parameter, dwAccessType, is used to control the use of proxy servers. Connections can be direct, through a preconfigured proxy server as specified in the Registry, or through a proxy server specified in the call to the constructor function. The fifth parameter to the constructor function, dwFlags, controls cache usage and asynchronous execution. The default cache behavior is to use the cache as appropriate and connect to remote servers if necessary. This can be modified in two ways. If the INTERNET_FLAG_DONT_CACHE flag is used, no downloaded data will be cached. The flag INTERNET_FLAG_OFFLINE instructs the library to never connect to remote servers but to attempt to satisfy requests only from the local cache. Internet calls are normally executed in a synchronous (blocking) fashion. When a request is made through a library function, the function does not return until the request is satisfied unless an error occurs. This behavior can be modified through the flag INTERNET_FLAG_ASYNC. When this flag is used, functions return immediately, even if operations are not finished. When a request is completed the application receives notification through a callback function. To use this capability you must derive your own class from CInternetSession and override the function CInternetSession::OnStatusCallback. Furthermore, to enable callbacks you must call CInternetSession::EnableStatusCallback with a value of TRUE. A complex Internet application might require several sessions, each with different settings, to offer a flexible range of services to its users. This is easily accomplished because an application can maintain multiple CInternetSession objects.

Internet Connections
The second step in establishing an Internet data transfer is to construct a connection object. MFC offers three types of connection objects: CFtpConnection, CGopherConnection, and CHttpConnection. These classes are derived from the class CInternetConnection, a class that encapsulates higher-level concepts of an Internet connection. As their names imply, they can be used to establish connections to FTP, Gopher, and HTTP servers.

390

Microsoft Foundation Classes PART III

Objects of these types are created through a call to a corresponding member function in CInternetSession. When you are finished with them, they can be deleted using the C++ delete operator.

FTP Connections
FTP connections are represented by objects of type CFtpConnection. These objects are created through a call to CInternetSession::GetFtpConnection. A manually executed FTP session requires that you specify a username (often anonymous or ftp) and a password (by convention set to your e-mail address when using an anonymous login). These two are among the parameters you supply to CInternetSession::GetFtpConnection when establishing a connection to an FTP server. You can use FTP connection objects to accomplish many of the same things you would accomplish during a manual session. You can manipulate directories through the functions SetCurrentDirectory, GetCurrentDirectory, GetCurrentDirectoryAsURL, RemoveDirectory, and CreateDirectory. Files can be opened (OpenFile), renamed (Rename), removed (Remove), downloaded (GetFile), or uploaded (PutFile). They can also be opened as a CInternetFile object using the OpenFile member function. An important FTP capability is to list the contents of directories. Unfortunately, FTP directories are returned by servers in a human-readable form, and the format is not consistent across various operating platforms and server implementations. This would require applications to have complex parsing algorithms built in, were it not for the fact that the class CFtpFileFind already implements these, providing a reliable way to parse FTP directories. After an FTP connection has been established, all you must do to enumerate the contents of a directory is switch to the desired directory, create a CFtpFileFind object, and use its FindFile and FindNextFile member functions. When you are finished using a CFtpConnection object, close it by calling its Close member function. Although the destructor also closes the connection, calling this member function helps avoid a diagnostic message.

Gopher Connections
CGopherConnection

objects manage connections to Internet Gopher servers. To create a Gopher connection, use CInternetSession::GetGopherConnection.

Note that although the specifications of CInternetSession::GetGopherConnection call for a server name, I found that it is not necessary to supply a valid server name; simply specifying the server name localhost will do nicely. This is because Gopher locators themselves contain a server name that supersedes any server name you might specify here. However, because this behavior is not fully documented it might change in future MFC versions.

Internet Support Classes CHAPTER 21

391

A Gopher server can provide a client application with two types of data: Gopher menus and Gopher files. Gopher menus consist of a series of Gopher locators, which in turn reference other Gopher menus (perhaps on other servers) and other Gopher files. Gopher files are provided to clients as isno capability to upload or manipulate files is offered.
CGopherConnection objects use a helper class for Gopher locators. This class, CGopherLocator, has three overloaded versions of its constructor, providing the ability to construct locator objects based on various pieces of information available to the client.

21
INTERNET SUPPORT CLASSES

Gopher menus can be parsed with the help of the CGopherFileFind class. The FindFile and FindNextFile member functions of this class set a CGopherLocator object to reference the selected Gopher menu entry, so you can avoid having to manually construct Gopher locators.

HTTP Connections
HTTP connection objects of type CHttpConnection represent connections to remote World Wide Web (HTTP) servers. Apart from the constructor, this class has one member function: CHttpConnection::OpenRequest. This member function returns a pointer to an object of type CHttpFile, which can be used to accomplish the transfer of an HTTP file item. In addition to downloading World Wide Web content, CHttpConnection supports Web authoring. CHttpConnection objects can also be used to establish secure connections. Objects of type CHttpConnection are typically created through a call to CInternetSession:: GetHttpConnection.

Internet Files
Internet file transfers are supported in MFC by the CInternetFile class and derived classes. CInternetFile itself is derived from CStdioFile, making it possible to use many standard MFC file manipulation operations on Internet file objects. FTP files are represented by objects of type CInternetFile. HTTP and Gopher files are, on the other hand, represented by classes derived from CInternetFile: CHttpFile and CGopherFile, respectively.

Other Support Classes


Several other support classes exist that facilitate efficient Internet programming with MFC. Some of these have already been mentioned in the preceding paragraphs, but allow me to briefly list them again for the sake of completeness. Finding specific files on an FTP or Gopher server is greatly simplified using two classes derived from CFindFile: CFtpFindFile and CGopherFindFile. These classes free the programmer from the burden of having to manually analyze the contents of FTP or Gopher directories; this is especially important in the case of FTP, where directory formats differ wildly.

392

Microsoft Foundation Classes PART III

Items on Gopher servers are referenced through Gopher locators; these are represented by objects of type CGopherLocator. Internet classes often use MFC exceptions for error reporting. Internet-specific errors are represented by the class CInternetException. Because connection problems on the Internet are frequent, it is imperative that applications catch and analyze these exceptions in order to avoid unexpected termination and to provide accurate feedback to users.

Using MFC Internet Classes in Applications


This section presents three very simple examples that demonstrate the use of the MFC Internet classes in FTP, Gopher, and HTTP sessions.

Communicating with an FTP Server


The first example in Listing 21.1 is a simple MFC program that retrieves a file from Microsofts FTP server. As are all the examples in this chapter, this is a console application that can be compiled and executed from the command line. To compile, use the following command:
cl /MT /GX ftpget.cpp

Listing 21.1. Retrieving a file from an FTP server.


#include <afxinet.h> #include <iostream.h> void main(void) { CInternetSession is(_T(FTPGET)); CFtpConnection *pFC = NULL; CInternetFile *pIF = NULL; try { pFC = is.GetFtpConnection(_T(ftp.microsoft.com)); pIF = pFC->OpenFile(_T(disclaimer.txt), GENERIC_READ, FTP_TRANSFER_TYPE_ASCII); char c; while (pIF->Read(&c, 1) == 1) cout << c; pIF->Close(); pFC->Close(); } catch (CInternetException *pIE) { cout << Internet error << pIE->m_dwError << . << endl; } delete pIF; delete pFC; }

Execution begins with the creation of a CInternetSession object on the stack. The single parameter supplied to the constructor is a string representing the applications name.

Internet Support Classes CHAPTER 21

393

Most of the body of the function main is protected by a try..catch block. This is to ensure that Internet errors are properly captured and reported to the user. After the Internet session has been established, the first step is to establish a connection to the desired FTP server. This is accomplished by calling CInternetSession::GetFtpConnection which, if successful, returns a pointer to a CFtpConnection object. The next step is to open the desired file for reading; the function CFtpConnection::OpenFile does this and returns a pointer to an object of type CInternetFile. The actual file is read using well-known CStdioFile functions, and its contents are written to standard output. When reading the file is finished, both the CInternetFile object and the CInternetConnection object are explicitly closed using their respective Close member functions before they are deleted.

21
INTERNET SUPPORT CLASSES

Communicating with a Gopher Server


Listing 21.2 retrieves a file from the University of Minnesota Gopher server (which can be called the mother of all Gopher servers because this is the place where Gopher originates). Its structure is very similar to the FTP example in the previous section. This program, too, can be compiled (cl /MT /GX gophrget.cpp) and executed from the command line.

Listing 21.2. Retrieving a file from a Gopher server.


#include <afxinet.h> #include <iostream.h> void main(void) { CInternetSession is(GOPHRGET); CGopherConnection *pGC = NULL; CGopherFile *pGF = NULL; try { pGC = is.GetGopherConnection(_T(localhost)); pGF = pGC->OpenFile( pGC->CreateLocator(0\t\tgopher2.tc.umn.edu\t70)); char c; while (pGF->Read(&c, 1) == 1) cout << c; pGF->Close(); pGC->Close(); } catch (CInternetException *pIE) { cout << Internet error << pIE->m_dwError << . << endl; } delete pGF; delete pGC; }

394

Microsoft Foundation Classes PART III

The basic order of execution is the same as in the FTP example: First, a CInternetSession object is created, followed by the creation of a CGopherConnection object and finally, a CGopherFile object. The CGopherFile object is created using CGopherConnection::OpenFile. The parameter passed to this file is a Gopher locator. This locator is created, for the same of simplicity, by directly calling CGopherConnection::CreateLocator with a locator string. In production applications, you might want to use the CGopherFindFile class to enumerate the contents of Gopher menus and establish locators for selected menu items.

Communicating with an HTTP Server


The third example (Listing 21.3) establishes a connection to Microsofts World Wide Web server and retrieves its main Web page, default.asp. This is also a console application that can be compiled from the command line with cl /MT /GX httpget.cpp.

Listing 21.3. Retrieving a file from an HTTP server.


#include <afxinet.h> #include <iostream.h> void main(void) { CInternetSession is(HTTPGET); CHttpConnection *pHC = NULL; CHttpFile *pHF = NULL; try { pHC = is.GetHttpConnection(_T(www.microsoft.com)); pHF = pHC->OpenRequest(_T(), _T(/default.asp), NULL, 0, NULL, NULL, 0); pHF->SendRequest(); char c; while (pHF->Read(&c, 1) == 1) cout << c; pHF->Close(); pHC->Close(); } catch (CInternetException *pIE) { cout << Internet error << pIE->m_dwError << . << endl; } delete pHF; delete pHC; }

Again, program execution begins with creating a CInternetSession object and then establishing an HTTP connection. Retrieval of the actual file is done in two steps. An object of type CHttpFile is created by calling CHttpConnection::OpenRequest. The actual request, however, is sent only when CHttpFile::SendRequest is called.

Internet Support Classes CHAPTER 21

395

The MFC Internet classes do not directly support parsing of HTML documents. For this reason no functionality is offered that would, for example, facilitate walking through URLs that appear in a retrieved document. If you want to follow links, you must parse HTML documents yourself and extract URLs from them.

21
INTERNET SUPPORT CLASSES

Summary
New to Windows programming is the Microsoft Win32 Internet (WinInet) library. This library provides a series of functions that implement FTP, Gopher, and HTTP connections. MFC 4.2 wraps the WinInet library in the form of a series of MFC Internet support classes, thus complementing lower-level support that already existed for the WinSock library. Transferring data between a client application and an Internet host is a three-step process: the creation of an Internet session, creation of an Internet connection, and the actual opening and transfer of the file. Internet sessions are represented by the CInternetSession class. An application can create several objects of this type, representing connections with different characteristics. Connections to FTP, Gopher, and HTTP servers are represented by objects of type CFtpConnection, CGopherConnection, and CHttpConnection. These objects are never created directly; instead, helper member functions of CInternetSession are used for this purpose. A connection object is closed when it is deleted, but an explicit call to the Close member function can help avoid a diagnostic message. The transferring of data between the client and Internet servers is accomplished through the CInternetFile class and the derived classes CGopherFile and CHttpFile. These classes make it possible to use the standard file support functionality of the MFC library to manage Internet data transfers. Other support classes can assist in the parsing of FTP and Gopher directories, managing Gopher locators, and handling Internet errors.

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22

397

22

Exceptions, Multithreading, and Other MFC Classes


IN THIS CHAPTER
s Using Exceptions in MFC Applications 398 s MFC and Multithreading 406 s Miscellaneous MFC Classes 410

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

398

Microsoft Foundation Classes PART III

Many MFC functions use C++ exceptions for reporting errors. This chapter begins with a look at exception handling in MFC, with particular emphasis on C++ style exception handling and the CException class. The MFC Library fully supports multithreading. Specific support for Win32 multithreading is available in the form of synchronization classes that wrap Win32 synchronization objects. MFC multithreading support and the CSyncObject class are the subject of the second half of this chapter.

NOTE
Just because MFC is thread-safe doesnt mean you can use all features freely in multithreaded applications. Underlying libraries that provide services used by MFC may not be thread-safe. Consequently, the corresponding MFC classes can only be used in the applications primary thread, not because of a limitation of MFC, but because of a limitation of the service for which they implement a wrapping.

Finally, I present a brief look at a series of miscellaneous MFC classes including simple data types, support classes for OLE and databases, and other classes and structures.

Using Exceptions in MFC Applications


The Microsoft Foundation Classes Library provides two different forms of exception handling. It supports C++ style typed expressions, and it also supports exception handling through oldstyle MFC macros.

Exception Handling with Macros


New applications should not use MFC exception processing macros. That said, as there are probably many applications out there that rely on the old, macro-based exception handling mechanism, it is probably helpful to have a brief summary of how those macros can be converted into code following the C++ exception syntax. The first and most obvious step is to replace the macro names with C++ keywords. The macros TRY, CATCH and AND_CATCH, and THROW and THROW_LAST should be replaced with the C++ keywords try, catch, and throw. The END_CATCH macro has no C++ equivalent and should simply be deleted. The syntax of the CATCH macro and the C++ catch keyword are different. What used to look like this:
CATCH(CException, e)

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22

399

should be replaced with this:


catch(CException *e)

An important difference between the two forms of exception handling is that when you are using the C++ exception handling mechanism, you are supposed to delete the exception object yourself. You can delete objects of type CException by calling their Delete member function. For example, if you used to catch an exception like this:
TRY { // Do something nasty here } CATCH(CException, e) { // Process the exception here } END_CATCH

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

you should translate the code into something similar to this:


try { // Do something nasty here } catch (CException *e) { // Process the exception here e->Delete(); }

NOTE
Do not attempt to delete an exception object in a catch block using the delete operator. The delete operator will fail if the exception object was not allocated on the heap.

C++ Exceptions and the CException Class


The C++ language provides support for the reporting and detection of abnormal conditions through typed exceptions. The MFC Library utilizes this support through a series of exception types that are derived from the class CException.

NOTE
The MFC Library does not directly support Win32 structured exceptions. If you want to process structured exceptions, you may have to use the Cstyle structured exception handling mechanism or write your own translator function that translates structured exceptions into C++ exceptions.

400

Microsoft Foundation Classes PART III

The primary function of the CException class is to provide a distinct type for MFC Library exceptions. It could fulfill that function even as an empty class. It does, however, provide several member functions that can be utilized when processing an exception. The member function GetErrorMessage can be used to retrieve a textual description of the error. The member function ReportError can be used to retrieve this error message and display it in a standard message box. Note that not all exceptions caught by a CException handler have a valid error message associated with them. The third member function that is important to know about is the Delete function. Use this function to delete an exception in a catch block if you process the exception. (Do not delete the exception if you use throw to pass it to another exception handler.) There are several classes derived from CException (see Figure 22.1). These classes are used to indicate errors and abnormal conditions relating to memory, file management, serialization, resource management, data access objects (DAO), database functions, Internet classes, OLE, and other categories. The following sections review these exception classes individually.

FIGURE 22.1.
Exception classes in MFC.

Exceptions CException CArchiveException

CDaoException

CDBException

CFileException

CInternetException

CMemoryException

CNotSupportedException

COleException

COleDispatchException

CResourceException

CUserException

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22

401

The CMemoryException Class


Exceptions of type CMemoryException are thrown to indicate a memory allocation failure. These exceptions are thrown automatically in MFC applications by the new operator. If you write your own memory allocation functions, you are responsible for throwing such exceptions yourself, for example,
char *p = malloc(1000); if (p == NULL) AfxThrowMemoryException(); else { // Populate p with data }

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

The CFileException Class


Exceptions of type CFileException indicate one of many file-related failures. To determine the cause of the exception, examine the m_cause member variable. For example,
try { CFile myFile(myfile.txt, CFile::modeRead); // Read the contents of the file } catch(CFileException *e) { if (e->m_cause == CFileException::fileNotFound) cout << File not found!\n; else cout << A disk error has occurred.\n; e->Delete(); }

Table 22.1 lists the possible values of m_cause.

Table 22.1. CFileException::m_cause values. Value Description


none generic fileNotFound badPath tooManyOpenFiles accessDenied invalidFile removeCurrentDir directoryFull

No error. Unspecified error. File could not be located. Part of the path name is invalid. Maximum number of open files exceeded. Attempt to open file with insufficient privileges. Attempt to use an invalid file handle. Attempt to remove current directory. Maximum number of directory entries reached.
continues

402

Microsoft Foundation Classes PART III

Table 22.1. continued Value


badSeek hardIO sharingViolation lockViolation diskFull endOfFile

Description Could not set file pointer to specified location. Hardware error. Attempt to access a locked region. Attempt to lock a previously locked region. The disk is full. The end of the file was reached.

These m_cause values are operating system independent. If you want to retrieve an operating system-specific error code, examine the member variable m_IOsError. Two member functions, OsErrorToException and ErrnoToException, can be used to translate operating system specific error codes and C runtime library error numbers into exception codes. Two helper member functions, ThrowOsError and ThrowErrno, can be used to throw exceptions using these error codes.

The CArchiveException Class


The CArchiveException class is used in exceptions indicating serialization errors. These exceptions are thrown by member functions of the CArchive class. To determine the cause of the exception, examine the m_cause member variable like this, for example:
CMyDocument::Serialize(CArchive &ar) { try { if (ar.IsLoading()) { // Load from the archive here } else { // Store in the archive here } } catch (CArchiveException *e) { if (e->m_cause == CArchiveException::badSchema) { AfxMessageBox(Invalid file version); e->Delete(); } else throw; } }

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22

403

Table 22.2 lists the possible values of m_cause.

Table 22.2. CArchiveException::m_cause values. Value Description


none generic readOnly endOfFile writeOnly badIndex badClass badSchema

No error. Unspecified error. Attempt to store into an archive opened for loading. The end of the file was reached. Attempt to load from an archive opened for storing. Invalid file format. Attempt to read an object of the wrong type. Incompatible schema number in class.

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

The CNotSupportedException Class


Exceptions of type CNotSupportedException are thrown when a feature is requested that is not supported. No further information is available on this error. This exception is frequently used in overridden versions of member functions in derived classes when the derived class does not support a base class feature. For example, the class CStdioFile does not support the base class feature LockRange:
try { CStdioFile myFile(stdin); myFile.LockRange(0, 1024); ... } catch (CNotSupportedException *e) { cout << Unsupported feature requested.\n; e->Delete(); }

The CResourceException Class


Exceptions of type CResourceException are thrown when a resource allocation fails or when a resource is not found. No further information is available on this error. Exceptions of this type are used in conjunction with GDI resources. For example:
try { CPen myPen(PS_SOLID, 0, RGB(255, 0, 0)); }

404

Microsoft Foundation Classes PART III


catch (CResourceException *e) { AfxMessageBox(Failed to create GDI pen resource\n); e->Delete(); }

The CDaoException Class


CDaoException exceptions are used to indicate errors that occur when MFC database classes are

used in conjunction with data access objects (DAO). All DAO errors are expressed in the form of DAO exceptions of the type CDaoException. To obtain detailed information about the error, examine members of the CDaoErrorInfo structure pointed to by m_pErrorInfo. Further OLE and extended MFC error codes can be obtained by examining the member variables m_scode and m_nAfxDaoError. To obtain information about a specific DAO error code, use the GetErrorInfo member function. To find out the number of error codes for which error information can be obtained, call GetErrorCount.

The CDBException Class


Exceptions of type CDBException are used to indicate errors that occur when using MFC ODBC database classes. To obtain information about the error, examine the m_nRetCode member variable that contains an ODBC error code. To obtain a textual description of the error, examine the m_strError member variable. More detailed information is available in the member variable m_strStateNativeOrigin, which provides a textual description of the error in the following format:
State: %s,Native: %ld,Origin: %s

In this string, the format codes are replaced as follows. The first code (%s) is replaced by a five-character ODBC error code corresponding to the szSqlState parameter of the ::SQLError function. The second code corresponds to the pfNativeError parameter of ::SQLError and represents a native error code specific to the data source. Finally, the third code corresponds to error message text returned in the szErrorMsg parameter of ::SQLError.

The CInternetException Class


The CInternetException class is used by the MFC Internet classes to indicate a failure. Failures include connection time-outs, DNS lookup failures, and other Internet problems. The class has two member variables: m_dwContext and m_dwError. The context value is a value passed to the constructor of MFC classes such as CInternetSession, and can be used to identify specific operations in an application with multiple simultaneous open Internet connections.

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22

405

The error value indicates the nature of the error. This can be a general Windows error (see the GetLastError function) or an Internet-specific error code. The latter are defined in the file wininet.h. Some of the most commonly occurring error codes include ERROR_INTERNET_TIMEOUT, ERROR_INTERNET_NAME_NOT_RESOLVED, and ERROR_INTERNET_CANNOT_CONNECT.

The COleException Class


The COleException class is used in exceptions indicating general OLE errors. To obtain information about the error, examine the m_sc member variable, which contains an OLE status code. The static member function Process can be used to turn any caught exception into an OLE status code. For example, this function, when passed an object of type CMemoryException, returns the OLE status code E_OUTOFMEMORY.

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

The COleDispatchException Class


Exceptions of type COleDispatchException are used to indicate OLE errors related to the OLE IDispatch interface. This interface is used in conjunction with OLE automation and OLE controls. An error code specific to IDispatch can be obtained by examining the m_wCode member variable. The member variable m_strDescription contains a textual description of the error. Additional member variables identify a help context (m_dwHelpContext), the name of the applicable help file (m_strHelpFile), and the name of the application that threw the exception (m_strSource).

The CUserException Class


Exceptions of type CUserException are meant to be used by application programs to indicate an error caused by the user. Typically, these exceptions are thrown after the user has been notified of the error condition through a message box.

Throwing an MFC Exception


If you wish to throw an MFC exception from your own code, you can do so by using one of the helper functions available in the MFC Library for this purpose. These helper functions are summarized in Table 22.3.

Table 22.3. Exception helper functions. Function name


AfxThrowArchiveException AfxThrowDaoException

Action Throws a CArchiveException Throws a CDaoException


continues

406

Microsoft Foundation Classes PART III

Table 22.3. continued Function name


AfxThrowDBException AfxThrowFileException AfxThrowMemoryException AfxThrowNotSupportedException AfxThrowOleDispatchException AfxThrowOleException AfxThrowResourceException AfxThrowUserException

Action Throws a CDBException Throws a CFileException Throws a CMemoryException Throws a CNotSupportedException Throws a COleDispatchException Throws a COleException Throws a CResourceException Throws a CUserException

These functions take a varying number of parameters in accordance with the type of the exception being thrown. These functions construct an exception object of the specified type, initialize it using the supplied parameters, and then throw the exception. Naturally, you can also elect to construct an exception object and throw the exception manually. This may be necessary if you derive a class from CException yourself.

MFC and Multithreading


MFC support for multithreading has two aspects. First, the MFC Library is thread-safe; it can be used in the context of multithreading applications. Second, the library provides a series of classes that provide explicit support for multithreading-related synchronization objects in Win32.

Thread-Safe MFC
A curious, frequently seen phrase in the MFC documentation states that MFC objects are not thread safe at the object level, only at the class level. I believe that this sentence requires more elaboration than what is provided in the pages of various MFC manuals. If read at face value, this would mean that it is patently unsafe to use separate threads in your application to manipulate two member variables in, say, a CDocument-derived class of your application. If this were indeed the case, it would mean a very severe restriction on multithreading usage, almost to the point where it would render multithreading and the MFC fundamentally incompatible in most real-life situations. Fortunately, this is not so. When you define member variables in an MFC-derived class of your own, you are responsible for making them thread-safe if necessary. This can be accomplished, for example, by providing

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22

407

wrapper functions that restrict access to these variables and by the judicious use of synchronization techniques inside those wrapper functions. In view of this, what the sentence I quoted at the beginning of this section really means is that for reasons of performance and simplicity, this was not done in the MFC Library. For this reason, accessing the same object from two different threads may fail because no synchronization mechanism is used. Consider, for example, the case of a CString. When you assign a value to a CString object, it frees any memory previously allocated for it, allocates the necessary memory for the new string data, and copies the data to this memory block. These operations are not protected by synchronization techniques, which means that if another thread attempts to obtain the value of the CString while it is being updated, the object will be in an inconsistent, transitory state. When the attempt is made to retrieve its value, only parts of the string may be returned, garbage information may be returned, or worse yet, an access violation may occur. On the other hand, if you add a CString member variable to your own CDocument-derived class, you can make this a protected or private member variable and restrict access to it through wrapper functions. In the wrapper functions, you can use, for example, mutex objects to provide exclusive access to the CString. This way, different threads in your application will be able to safely access the same object. In fact, notwithstanding the blanket statement made in the documentation, many MFC objects are actually safely accessible from separate threads assuming that you know what you are doing. For example, you can access a CString object through the operator LPCSTR operator from as many threads as you wish; but do not try to modify the same CString object from two different threads simultaneously! The rule of thumb: Unless you know that what you are doing is safe, do not do it. By default, accessing an object from two different threads should be considered unsafe, unless you know explicitly that the particular access mechanism you intend to use has no harmful consequences.

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

Creating Threads in MFC


The MFC Library differentiates between two types of threads. User-interface threads are threads that have a message loop and process messages; worker threads are everything else. The typical use of a user-interface thread is to create a message loop for a window that runs independently of your applications main message loop. The typical use of a worker thread is to perform some background processing (for example, background printing). The creation of a user-interface thread involves deriving a class from CWinThread and calling AfxBeginThread to create the thread object. In the class derived from CWinThread, you must override several CWinThread member functions. As a minimum, you must provide an override version of InitInstance, the member function which is called when the thread is initialized.

408

Microsoft Foundation Classes PART III

The creation of worker threads is simpler. Worker threads do not require a separate CWinThreadderived class; instead, AfxBeginThread is called with the address of the thread function. Note that MFC objects should not be used in conjunction with threads not created using AfxBeginThread. As an example for creating a worker thread, consider the following code:
UINT MyWorkerThread(LPVOID pParam) { // Do something lengthy with pParam return 0; // Terminate the thread } ... // elsewhere AfxBeginThread(MyWorkerThread, myParam);

Thread Synchronization
The Win32 API provides a series of synchronization objects that support the synchronization of concurrently executing threads. Of these, events, mutexes, critical sections, and semaphores are supported by MFC class wrappers. The base class for all MFC synchronization classes is CSyncObject, which is a pure virtual base class. The hierarchy of MFC synchronization classes is shown in Figure 22.2.

FIGURE 22.2.
Synchronization classes.

Synchronization CSyncObject CCriticalSection CMultiLock

CEvent

CSingleLock

CMutex

CSemaphore

The CSyncObject class supports creation of a synchronization object by name through its constructor. Subsequently, the Lock and Unlock member functions can be used to gain access to, and release, the synchronization object. The specific meaning of these functions depends on the synchronization class being used.
CSyncObject-derived

objects can be used in conjunction with the CSingleLock or CMultiLock classes. These classes provide an access control mechanism to the objects. After examining the synchronization classes, we return our attention to these synchronization access classes.

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22

409

The CEvent Class


The CEvent class encapsulates the functionality of a Win32 event. An events state is set to signaled by calling the CEvent::SetEvent function. An event can be either a manual-reset event or an automatic event. A manual-reset event must be explicitly reset to its nonsignaled state; an automatic event is reset to nonsignaled when a waiting thread is released. An event can be waited upon by calling the Lock member function. The Unlock member function is not used for CEvent objects. The PulseEvent member function sets the events state to signaled, releases waiting threads (if any), and resets the events state to nonsignaled. In case of a manual-reset event, all waiting threads are released; in case of an automatic event, only a single thread is released.

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

The CMutex Class


Mutex objects, represented by the CMutex class, are used to gain mutually exclusive access to a resource. While one thread owns a mutex, other threads cannot gain access to it. When you construct a CMutex object, you can specify in the call to the constructor whether you wish to initially own the mutex object. If yes, the constructor will not return until it gains ownership of the mutex object. To otherwise gain ownership to a mutex object, call the Lock member function. To release the mutex object, call Unlock.

The CCriticalSection Class


Critical section objects have functionality similar to that of mutexes; however, critical sections are slightly more efficient but cannot be used across process boundaries. A critical section object is typically used to prevent multiple threads from executing the same piece of code simultaneously. The critical section object is initialized by the CCriticalSection constructor. Subsequently, you can use the Lock and Unlock member functions to access the critical section. To access the underlying critical section object, you can use the operator CRITICAL_SECTION* operator. Note that objects of type CCriticalSection cannot be used in conjunction with the classes CSingleLock and CMultiLock.

The CSemaphore Class


Semaphores are used to limit the number of accesses to a resource. Semaphore objects are represented by the CSemaphore class.

410

Microsoft Foundation Classes PART III

When a semaphore is object created through the CSemaphore constructor, you can specify the initial and maximum usage count. The usage count can be increased by calling CSemaphore::Lock; if the usage count exceeds the maximum usage count, this function will wait until the semaphore object becomes available. The usage count can be decreased by calling Unlock.

Synchronization with CSingleLock and CMultiLock


Synchronization objects of type CEvent, CMutex, and CSemaphore can be accessed through the synchronization access classes CSingleLock and CMultiLock. To create an access object of type CSingleLock, create the synchronization object first, then pass a pointer to this object to the CSingleLock constructor. Subsequently, you can gain access to the object by calling CSingleLock::Lock and release the object using CSingleLock::Unlock. To determine if an object has been locked, use the CSingleLock::IsLocked member function. The functionality of the CMultiLock class is similar to that of CSingleLock; however, CMultiLock makes it possible for you to wait on several synchronization objects at the same time. To construct a CMultiLock object, pass an array of CSyncObject-derived objects to its constructor. Later, you can wait for any or all of these objects to become signaled by calling the Lock member function. The return value of this function identifies the object that was signaled. You can release that object by calling CMultiLock::Unlock. The CMultilock::IsLocked function can be used to determine the locked state of a specific synchronization object. Note that objects of type CCriticalSection cannot be used in conjunction with CSingleLock and CMultiLock.

Miscellaneous MFC Classes


The Microsoft Foundation Classes Library provides a series of miscellaneous classes. Some of these are general purpose (for example, CString) while others are used in specific contexts. The remainder of this chapter presents a brief tour exploring these classes.

Simple Value Types


The MFC Library provides a series of classes that represent simple data types. The CPoint class is an MFC wrapper for the Win32 POINT structure. A pointer to a CPoint object can be used every time when a pointer to a POINT structure is expected. The CPoint class supports a series of operators including addition and subtraction, testing for equality and inequality, and the += and -= operators. The Offset member function can be used to offset a CPoint by a given pair of values in the horizontal and vertical direction. The CRect class wraps the functionality of the Win32 RECT structure. Pointers to objects of this class and pointers to structures of type RECT can be used interchangeably.

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22


CRect supports a variety of member functions and overloaded operators to compare, copy, offset, inflate, or deflate rectangles, calculate the union and intersection of two rectangles and perform other operations.

411

The CSize class is an MFC wrapper for the Win32 SIZE structure. Pointers to type CSize and pointers to SIZE structures can be used interchangeably. The CSize class defines a series of operators for comparing, adding, and subtracting CSize objects.
CPoint

The addition and subtraction operators can also be used with mixed types. Objects of type and CRect can be offset by an object of type CSize or CPoint using the addition or subtraction operator.

The CString type represents a variable-length string. Memory for the string in a CString object is dynamically allocated and released as appropriate. Objects of type CString can be used to store ANSI and OEM strings, and on systems supporting Unicode (such as Windows NT), Unicode strings as well. The CString class defines a large variety of functions and operations that can be used for manipulating the string. In particular, CString objects can be concatenated using the addition operator. They can be compared using the equal to, less than, and greater than operators. The Mid, Left, and Right member functions can be used to perform operations similar to those available in the BASIC language. Other functions can be used to extract parts of a string, change the case of a string, find substrings in the string, and collate and compare strings. The CString class supports loading a string value directly from a Windows resource file via the LoadString function. The CString class also supports serialization and the use of the << and >> operators with CString objects in conjunction with the CArchive class.
CString

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

objects can be used in many situations in place of pointers to type char, thanks to the existence of the operator LPCSTR operator.

The CTime class represents an absolute time; the CTimeSpan class represents the difference between two time values. Both of these classes support a variety of member functions to set, compare, and manipulate time values, and to extract various elements (for example, seconds, minutes, hours) from time values. The CTime class also supports time zones and the conversion of a CTime value into a formatted string representing date and time. Both CTime and CTimeSpan support serialization and the use of the << and >> operators in conjunction with a CArchive. Much of the functionality of these two classes has been superseded by the COleDateTime class.

Structures and Support Classes


There is a series of miscellaneous structures and classes in MFC that support specific areas of functionality.

412

Microsoft Foundation Classes PART III

The CCommandLineInfo encapsulates command-line information in an MFC application. An object of type CCommandLineInfo or an object of a derived class can be used in conjunction with CWinApp::ParseCommandLine to parse the command line. The default implementation of CCommandLineInfo supports a filename on the command line and a variety of flags that specify printing, DDE, OLE Automation, and editing an OLE-embedded item. If other commandline flags or parameters are needed, derive a class from CCommandLineInfo and override its ParseParam member function. The CCreateContext class is used when the framework creates frame windows and views associated with a document in an MFC framework application. Member variables of CCreateContext are pointers identifying the view class, the document, and the view and the frame windows. The CFileStatus structure is used by the functions CFile::GetStatus and CFile::SetStatus to retrieve and set a files attributes (such as creation date, permissions, or the filename). The CMemoryState class is used for detecting memory leaks. By creating CMemoryState objects and calling their Checkpoint member functions at various stages during program execution, you can verify that all allocated memory has been correctly released and dump the contents of unreleased objects. The CPrintInfo class is used to store information about a print job. Objects of this type are used by the framework when calling printing-related member functions of the CView class. class is used in ON_UPDATE_COMMAND_UI handler functions of classes derived from Through objects of this type, applications can enable, disable, or otherwise manipulate user-interface items. Update handler functions are typically added via the ClassWizard. For instance, if your application has a Copy command in its Edit menu, you can open the ClassWizard, add a handler for the UPDATE_COMMAND_UI message for the item ID_EDIT_COPY, and write a function similar to the following:
CCmdUI CCmdTarget. void CMyView::OnUpdateEditCopy(CCmdUI* pCmdUI) { BOOL bEnable = AreThereCopiableObjects(); pCmdUI->Enable(bEnable); }

The

The CDataExchange class is used to support dialog data exchange. Objects of this type store context information that is used by dialog data exchange (DDX) and dialog data validation (DDV) functions. Classes of similar functionality include CPropExchange (used to exchange data on persistent properties of OLE controls), CFieldExchange (used for exchanging data between ODBC records and dialog controls), and CDaoFieldExchange (used for exchanging data between DAO records and dialog controls). The CRectTracker class implements a tracking rectangle for onscreen objects. It is used by the framework in conjunction with embedded OLE objects, but can also be used by applications for application-specific objects.

Exceptions, Multithreading, and Other MFC Classes CHAPTER 22

413

The CWaitCursor class provides a one-line mechanism for displaying an hourglass cursor. When the object is constructed, the hourglass cursor is displayed; when the object is destroyed, the original cursor is restored. Declare an object of this class in functions that perform lengthy operations. Additional support classes and structures provide support for OLE, OLE automation, ODBC and DAO, and the Microsoft Internet Information Server.

Summary
The MFC Library uses C++ style exceptions to communicate error conditions. Exceptions that are of a type derived from CException are thrown using a variety of helper functions and caught by your application. Older MFC applications that predate C++ exception support in Visual C++ used a series of macros for this purpose. These macros can be easily translated into the C++ keywords try, throw, and catch. The CException-derived classes that are used by MFC functions are summarized in Table 22.4.

22
EXCEPTIONS, MULTITHREADING, AND OTHER MFC CLASSES

Table 22.4. MFC exception classes. Function name Action


CArchiveException CDaoException CDBException CFileException CInternetException CMemoryException CNotSupportedException COleDispatchException COleException CResourceException CUserException

Serialization errors Errors occurring with data access objects Errors occurring during ODBC usage File system errors Internet errors Memory allocation failure Notification of unsupported feature request OLE IDispatch errors (automation, controls) Generic OLE errors Resource allocation failure (GDI) Errors caused by the user

For most of these exception types, there is a corresponding helper function (for example,
AfxThrowArchiveException). You can also construct a CException-derived object and throw an

exception manually.

414

Microsoft Foundation Classes PART III

In the exception handler, you are responsible for deleting the CException-derived object by calling its Delete member function. You can also derive your own exception class from CException. Multithreading support in MFC can be considered from two aspects. First, the MFC Library is thread-safe at the class level. Second, it provides multithreading support in the form of CWinThread and a series of synchronization classes derived from CSyncObject. The MFC distinguishes between threads that maintain a message loop (user-interface threads) and threads that do not (worker threads). User-interface threads are created by deriving a type from CWinThread, while worker threads only require a worker thread function. Both types of threads are created by calling AfxBeginThread. synchronization classes include CEvent, CMutex, CCriticalSection, and All of these classes, with the exception of CCriticalSection, can be used in conjunction with the classes CSingleLock and CMultiLock.
CSemaphore. CSyncObject-derived

The MFC Library defines a series of support classes that encapsulate various Win32 structures or provide support for various operations. Simple data types include CPoint, CSize, CRect, CString, CTime, and CTimeSpan. Other support classes and structures include CCommandLineInfo, CCreateContext, CFileStatus, CMemoryState, CPrintInfo, and CCmdUI. Dialog data exchange is supported by CDataExchange, and the specific classes CPropExchange, CFieldExchange, and CDaoFieldExchange. The CRectTracker class implements a tracking rectangle; the CWaitCursor class can be used to easily display an hourglass cursor. Additional support classes and structures exist for OLE, ODBC and DAO, and Internet Server.

IN THIS PART
s OLE, ActiveX, and the Component Object Model 417 s OLE Servers 439

IV
PART

s OLE Containers 455 s OLE Drag and Drop s Automation 493 511 537 477

s Building ActiveX Controls with MFC s Using the ActiveX Template Library s ActiveX Documents 559 s Distributed COM 571

OLE, COM, and MFC Applications

OLE, ActiveX, and the Component Object Model CHAPTER 23

417

23

OLE, ActiveX, and the Component Object Model


IN THIS CHAPTER
s OLE Basics and the Component Object Model 418 s COM and Compound Documents s Applications of COM and OLE s A Simple Example 426 422

425

23
OLE, ACTIVEX, AND COM

418

OLE, COM, and MFC Applications PART IV

Object linking and embedding is at the core of most modern Windows applications. It is also a very complex technology that would be difficult to master without the help of MFC. However, in order to use the MFC Library efficiently for OLE applications, it is helpful to have a solid understanding of OLE fundamentals. While not strictly needed if you are satisfied with the stock implementation of basic OLE features in your application (provided by AppWizard), this understanding becomes essential if you want to implement more advanced features, such as OLE drag and drop, linked OLE objects, or OLE-based Clipboard operations. Microsoft recently introduced a new term for some of its OLE-related technologies. What used to be called OLE custom controls are now ActiveX controls; ActiveX also describes other technologies built on the same foundations as OLE.

OLE Basics and the Component Object Model


At the heart of OLE and ActiveX is a technology known as the Component Object Model (COM). COM is a binary standard that specifies how OLE and ActiveX components, or objects, interact with each other. It is important to note that COM is a language-independent standard; the only requirement is that the language used must support the concept of pointers and calling functions through pointers. Naturally, it is easier to develop COM applications in objectoriented language environments.

Interfaces and Methods


A COM object is accessed exclusively through one or more sets of interfaces. An interface is a set of functions, also referred to as methods. The COM standard not only specifies the binary object standard, it also defines a series of standard interfaces that provide common functionality. Let me attempt to translate this into different terms. A COM interface should best be thought of as a table of function pointers and information relating to those function pointers that define the parameters and return values of those functions. For example, methods of an automation object are exposed through a METHODDATA structure, which is defined as follows:
typedef struct FARSTRUCT tagMETHODDATA { OLECHAR FAR* szName; PARAMDATA FAR* ppdata; DISPID dispid; UINT iMeth; CALLCONV cc; UINT cArgs; WORD wFlags; VARTYPE vtReturn; } METHODDATA, FAR* LPMETHODDATA;

Of particular interest in this structure is the iMeth member. This member is an index into a table of function pointers. In C++ implementations, it is used in conjunction with the virtual function table of a C++ class.

OLE, ActiveX, and the Component Object Model CHAPTER 23

419

Virtual function tables are not often on the C++ programmers mind. While we accept and use the benefits of virtual functions, we rarely think of the specifics of their implementation. The next paragraphs present a quick reminder as to the whys and hows of virtual function implementation. Virtual functions in C++ have been introduced to answer a common problemnamely, how to refer to member functions in a derived class when all you have is a pointer of base class type; that is, how to avoid using the base class implementation of the function instead. By referring to derived functions through a table of function pointers associated with the object in question, the compiler ensures that the function appropriate to the object is called, even when it lacks type information on the object otherwise. A recommendation found in Stroustrups The Annotated C++ Reference Manual suggests a table of function pointers preceding the object data. It is this table of functions that is also utilized through the indexes in the COM METHODDATA structure. Naturally, if you implement COM in C or another language that does not automatically build virtual function tables, it may become necessary to construct those tables by hand. An obvious consequence of this is that all COM methods, if declared as C++ member functions, must be declared with the virtual keyword. This is accomplished by using a standard set of macros (more about these macros later in this chapter). It is important to realize that a COM interface is not the same as the C++ class, object, or C structure that is used to implement the interface. The COM standard specifies how interfaces are exposed; it does not specify the implementation of methods. In other words, we know through the COM standard how to interpret tables that reference function addresses; however, the standard says nothing about how those functions implement the expected behavior. Interfaces are strongly typed. Furthermore, an interface cannot be changed or altered. It is not possible to add methods to, or remove methods from, an interface; doing so creates a new interface. (Of course, COM objects can implement multiple interfaces.)

23
OLE, ACTIVEX, AND COM

Methods and Memory Allocation


Of particular interest when implementing methods is the issue of memory allocation. COM defines specific rules for cases when it becomes necessary for the caller to pass a pointer to the called method, or for the method to return data in the form of a pointer to the caller. When memory is allocated by the caller, it must be freed by the caller. When memory is allocated by the method, it must be freed by the caller. The exception is error conditions: in such cases, the method must ensure that memory allocated by it will be reliably freed and that all returned pointers are explicitly set to NULL as appropriate. When a pointer is passed to the method, the method may free the memory associated with it and reallocate. The caller is responsible for freeing the final value. However, if an error occurs, the method is responsible for releasing any memory allocations it made.

420

OLE, COM, and MFC Applications PART IV

COM provides a memory allocation interface (the IMalloc interface) that provides thread-safe memory allocation methods. A pointer to this allocator can be obtained by calling the CoGetMalloc COM function.

Inheritance and Reusing Objects


Inheritance and reusability are terms with specific meanings for the developer of objectoriented code. These terms imply the capability of deriving your own classes from base classes, replacing methods with customized versions, and adding methods of your own. None of this is available with respect to COM objects. Although it is possible to inherit an interface, that does not mean inheritance of functionality; the interface contains no implementation. Instead, COM objects are treated as black boxes. Nothing is known about the details of the implementation of the interface, only the specifications of the interface itself. In other words, what we know is the objects behavior, not how that behavior is implemented. COM offers two specific mechanisms for reusability. Containment/delegation is a mechanism wherein outer objects act as clients to inner objects acting as servers. Sound familiar? Think of an OLE drawing embedded in a Word document. The other mechanism, aggregation, enables outer objects to expose interfaces from inner objects, making it appear as if the outer object implemented those interfaces itself.

Interface Identifiers
Interfaces are identified through globally unique identifiers, or GUIDs. GUIDs are 128-bit identifiers that are unique (hopefully) throughout the world. Thus, a programmer assigning a GUID to an interface can reasonably expect that no other interface, no matter who develops it, would conflict with this one with an identical identifier. The Visual C++ development system provides two utilities that help you generate globally unique identifiers. Both the command-line utility uuidgen.exe and the Windows utility guidgen.exe can be used for this purpose. The guidgen.exe utility can create identifiers in a form suitable for pasting into source code. These programs rely on the COM API function CoCreateGuid, which in turn uses the RPC function UuidCreate to create an identifier that is globally unique to a high degree of certainty.

Interface Definition Through IUnknown


All COM objects must implement the interface known as IUnknown. This interface defines three essential methods: QueryInterface, AddRef, and Release. The AddRef and Release methods are used to manage the lifetime of an object. They are typically implemented as functions that increase or decrease a reference count. When the reference count in Release reaches zero, the object should be destroyed.

OLE, ActiveX, and the Component Object Model CHAPTER 23


QueryInterface

421

is used to query the object about specific interfaces. This method receives a unique identifier for the interface requested; upon return, it should supply an indirect pointer to the interface, or an error if the requested interface is not supported by the object.

To add support for the IUnknown interface in a C++ class, you can derive the class from the class which is declared in unknwn.h. The member functions QueryInterface, AddRef, and Release are declared as pure virtual functions; you must supply your own implementation.
IUnknown,

Class Objects and Registration


A class object should not be confused with the concept of a class in object-oriented languages. It is merely another COM object, one that specifically implements the IClassFactory interface. This interface is the key to a fundamental COM capability. Through the IClassFactory interface, applications that were written without any knowledge as to the particular class can still create objects for that class. This is accomplished in part by registering the class, and in part by specific methods within the class. A class is identified through a CLSID, which is just another GUID. The operating system maintains a database of all registered CLSIDs in the system. What this means in the Windows environment is a set of entries in the Windows Registry. Registration entries are made in the form of a subkey under HKEY_CLASSES_ROOT\CLSID, identified by the CLSID in string form. Applications that are aware of a CLSID can use the COM API functions CoGetClassObject and CoCreateInstance to create a class object or create an uninitialized object of the kind associated with the CLSID.

23
OLE, ACTIVEX, AND COM

Inter-Object Communication
After you have obtained a pointer to an interface, you can call the methods in that interface. If the interface is within the same process as the caller, the call is made directly to the functions implementing the methods of the interface, with no intervening operating system code. However, if the interface is outside the boundaries of the current process, an intervening infrastructure becomes essential. (Remember that Win32 processes run in their own private memory space and do not see each others processes or data.) In order for a call to reach a server across process boundaries, it is necessary for a mechanism to exist that packages the calls parameters in the client side and unpackages them on the server side. The process of packaging parameters for transmission across process boundaries is called marshaling; the process of unpackaging them on the server side is called unmarshaling. COM provides a useful and efficient marshaling mechanism (standard marshaling) but also enables developers to implement customized marshaling techniques (custom marshaling). Marshaling is always performed by a proxy object, an object which, from the clients point of view, looks, feels, and smells like the real thing. The only difference is that the table of function pointers representing the objects methods point to stub implementations instead of actual ones.

422

OLE, COM, and MFC Applications PART IV

The stub implementations translate a call on the client side into a call on the server side using a communication mechanism such as Remote Procedure Calls (RPCs). Neither the client nor the server sees any difference between calls made in-process and calls made across process boundaries.

Monikers
Monikers are COM objects that implement the IMoniker interface. Through this interface, applications can obtain a pointer to an object identified by the moniker. They can do so by calling the IMoniker method BindToObject. COM identifies several types of monikers. File monikers identify objects stored in their own files. Item monikers identify objects contained within another object; for example, an embedded object in an OLE container document or a user selection such as a range of cells in a spreadsheet. Composite monikers are concatenations of monikers; you can think of them, for example, as concatenated partial pathnames that form a complete pathname. Finally, antimonikers serve the same role in composite monikers as the .. symbol does in pathnames, effectively removing parts of a composite moniker just like using .. removes parts of a path from a pathname. Monikers are used as a method of naming COM objects. Monikers can be saved so the naming is persistent. A special, rarely used moniker type, the pointer moniker, provides a monikerlike wrapping of interface pointers that can be passed whenever a moniker is expected. However, pointer monikers are not persistent; they cannot be saved.

COM and Threads


COM defines a specific approach for thread-safe implementation. This apartment model defines a set of rules applications must follow if they are to be able to create and access objects from within separate threads of the same process. The apartment model groups objects by owner thread. Objects can only live in a single thread (an apartment). Within the same thread, methods can be called directly; however, when calls are made across thread boundaries, the same marshaling technique must be used as with calls across process boundaries. The COM libraries provide a set of helper functions for this purpose.

COM and Compound Documents


The most commonly known use of OLE technology is in the form of OLE containers and servers. Together, container and server applications enable users to manipulate, within a single application, data coming from different sources and several applications. Compound document technology is based, in addition to the basics of the Component Object Model, on Structured Storage and Uniform Data Transfer.

OLE, ActiveX, and the Component Object Model CHAPTER 23

423

Structured Storage
Structured Storage provides two interfaces that closely mimic traditional functions found in most disk-based file systems. The IStorage interface provides functionality analogous to that of file systems (directories). Just like disk-based directories, a storage object can contain hierarchical references to other storage objects. It also tracks the locations and sizes of other objects stored within. The IStream interface is analogous to a file. As its name implies, a stream object contains data as a contiguous sequence of bytes. Compound files consist of a root storage object with at least one stream object representing native data. Additional storage objects can represent linked or embedded items. File-based storage is implemented with the help of the IRootStorage interface. Objects that can be embedded within container application documents must implement the IPersistStorage interface, which enables the object to be saved in a storage object. Other persistent storage-related interfaces include IPersistStream and IPersistFile. Structured storage has many benefits other than providing the means to treat a hierarchical set of objects as a single file. As in the case of real file systems, replacing a single object does not require that the entire compound file be rewritten. Objects can be accessed individually, without having to load the entire file. Structured storage also provides facilities for concurrent access by several processes and for transaction-based processing (commit and reverse functionality). The compound file implementation is operating system and file system independent. A compound file created, for example, on a FAT file system under Windows 95 can be reused from within a Windows NT application on an NTFS file system or by a Macintosh application using the Macintosh file system. Storage and stream objects are named according to a set of conventions. The root storage object is named as the underlying file, with the same restrictions on naming as applicable for the file system. Names of nested elements that are up to 32 characters in length (including any terminating null characters) must be supported by implementations. Whether a case conversion is applied to names or not is an implementation-defined behavior.

23
OLE, ACTIVEX, AND COM

Data Transfer
Transferring data between applications is accomplished through the IDataObject interface. This interface provides a mechanism for transferring data and also for notifications of changes in the data. Data transfer occurs with the help of two structures: FORMATETC and STGMEDIUM. The FORMATETC structure is defined as follows:
typedef struct tagFORMATETC {

424

OLE, COM, and MFC Applications PART IV


CLIPFORMAT cfFormat; DVTARGETDEVICE *ptd; DWORD dwAspect; LONG lindex; DWORD tymed; }FORMATETC; *LPFORMATETC;

This structure generalizes the idea of Clipboard formats, providing, in addition to the cfFormat parameter, parameters that identify the target device for which the data was composed and information on how the data should be rendered. The STGMEDIUM structure generalizes the idea of global memory handles used in traditional Windows Clipboard operations. This structure is defined as follows:
typedef struct tagSTGMEDIUM { DWORD tymed; union { HBITMAP hBitmap; HMETAFILEPICT hMetafilePict; HENHMETAFILE hEnhMetaFile; HGLOBAL hGlobal; LPOLESTR lpszFileName; IStream *pstm; IStorage *pstg; }; IUnknown *pUnkForRelease; }STGMEDIUM;

Through these two structures, providers and recipients of data can negotiate the types of data they can send and accept along with the most efficient storage mechanism used in transmitting the data.

Compound Documents
Compound documents may, in addition to native data, contain two types of items: linked objects and embedded objects. Linked objects represent objects that continue to reside where they were originally created (for example, in a file). The compound document contains a reference to this item (a link) and information on how the item should be presented. The container can present the linked item without activating the linkindeed, it can present the item even if the application that created the item is not available on a particular system. Activating the link means invoking the server application for editing and manipulating link data. The use of links results in small container documents and is also beneficial if the linked item is routinely maintained by a user other than the owner of the container document. Embedded objects reside physically within the container document. The advantage of using embedded objects is that documents can be manipulated as single files; in contrast, when linked items are used, several files may need to be exchanged between users. Furthermore, links are easily broken if linked items are moved. (Windows currently does not implement a tracking mechanism for linked items.)

OLE, ActiveX, and the Component Object Model CHAPTER 23

425

Servers for objects in a container document are implemented either as in-process servers or as local servers. An in-process server is essentially a DLL running in the process space of the container application. The major advantage of an in-process server is performance; methods in such a server are called directly, without the associated COM overhead. However, local servers offer several benefits also. They can support links (in-process servers cannot); they provide compatibility with OLE1; and they provide the added benefits of running in a separate process space (increased robustness and ability to serve multiple concurrent clients). Compound documents also support in-place activation. The in-place activation mechanism enables embedded items to be edited within the container applications window. Basic support for compound document containers and servers comes in the form of the IOleClientSite and IOleObject interfaces. Servers also implement the IDataObject and IPersistStorage interface; furthermore, in-process servers implement IViewObject2 and IOleCache2. In-place activation is accomplished via the IOleInPlaceSite, IOleInPlaceObject, IOleInPlaceFrame, and IOleInPlaceActiveObject interfaces.

Applications of COM and OLE


As the discussion to this point should have made obvious, COM is a set of specifications going far beyond mere object linking and embedding capability. Indeed, compound documents are just one of the many applications of COM; other uses include COM automation, OLE drag and drop, and ActiveX controls, which are reviewed shortly. COM also finds its way into specialized areas; for example, much of MAPI, the Messaging Application Programming Interface, is based on COM. Most COM applications require specific registration entries. Entries in the Windows Registry are typically made under the keys HKEY_CLASSES_ROOT\CLSID and HKEY_CLASSES_ROOT\Interface.

23
OLE, ACTIVEX, AND COM

OLE Document Containers and Servers


OLE containers and servers together implement compound document technology. OLE containers maintain compound documents consisting of linked and embedded items; OLE servers provide such linked and embedded items and the functionality required for their activation.

Automation
COM-based automation enables an automation server application to expose automation objects through a series of properties and methods. Information about properties and methods is published through the IDispatch interface. By querying this interface, automation clients can obtain information about the properties and methods identified by name. Automation objects need not be visible objects. For example, an automation server may perform scientific calculations, do spell checking, or supply physical constants identified by name, without ever presenting a visual interface on its own.

426

OLE, COM, and MFC Applications PART IV

Automation clients, also known as automation controllers, are applications that manipulate automation objects. Automation controllers can be generic (for instance a programming environment, such as Visual Basic) or can be developed to control specific automation objects.

OLE Drag and Drop


OLE drag and drop provides a powerful mechanism for implementing drag and drop functionality. OLE drag and drop capabilities are implemented through the IDropSource and IDropTarget interfaces and the DoDragDrop function. After receiving the data item that is the object of the drag and drop operation and a pointer to an IDropSource interface, DoDragDrop enters a loop during which it tracks mouse events. If the mouse is over a window, DoDragDrop checks whether that window registered as a valid drop target. DoDragDrop calls various methods of IDropTarget and IDropSource to carry out its operation. Drag and drop functionality and Clipboard cut and paste are very similar. Often it is beneficial to implement these two areas of functionality together, reusing code as much as possible.

ActiveX Controls
ActiveX controls represent a technology that came into existence as a 32-bit replacement technology for Visual Basic controls, but has since gone far beyond that, becoming one of Microsofts key technologies for the Internet. ActiveX controls are COM objects that provide an extended interface, which implements the behavior of a Windows control. ActiveX control servers are typically implemented as in-process servers (an OCX file is simply a dynamic link library with a special filename extension), which is exactly what makes them ideal objects for Internet content as well because the control DLL can be downloaded by a browser and then executed in its process space. ActiveX control containers are applications that support ActiveX controls in their windows or dialogs.

Custom Interfaces
For specialized applications, you can also develop your own custom COM interfaces. Custom interfaces are developed using the tools available in the Microsoft Windows Software Development Kit, including the Microsoft Interface Definition Language (MIDL) compiler.

A Simple Example
Before I began writing this chapter, I did not think that including a meaningful example was possible. While programs with OLE/COM capabilities are simple to write through the MFC Library, use of MFC frequently obscures the underlying technology. Presenting a set of instructions that tell you which buttons to push in the Visual Studio would have helped little in

OLE, ActiveX, and the Component Object Model CHAPTER 23

427

furthering your understanding of the underlying concepts and principles. Pasting a large, complex piece of AppWizard-generated code into my manuscript would have helped even less. However, after spending some time thinking on this issue, I realized that it is possible to write a COM program that is small and compact, yet demonstrates some of the COM fundamentals. The application I present in the remainder of this chapter is little more than 300 lines in length and contained within a single file, yet provides all the capabilities of an automation server. Why an automation server? In a sense, automation exposes the underlying mechanisms in their purest form. Were I to implement, for example, an OLE container, it would require dealing with formats, visual presentation of the data, management of windows, and other issues. The same applies for OLE drag and drop. The complexity of these applications would have ruled out their utility in demonstrating fundamental concepts in any meaningful manner. In contrast, an automation server can be implemented with a minimum of fuss, and its use can be easily demonstrated with a few lines of script in an automation client, such as Visual Basic.

Functional Description
What should the simplest automation server do? The venerable traditions of C and C++ programming dictate the only reasonable answer: Obviously, it should present the string Hello, World! The automation server that I set out to write does exactly that; it presents a string containing this text in the middle of its window, while also exposing the string as an OLE automation property through a pair of get and put methods. Assuming that the server has been properly installed, it can be used from any automation client. To exercise the server from Visual Basic, create a form with a single button and attach the code shown in Listing 23.1 to the button. This code activates the server and changes the default text of the server to the text specified in the Visual Basic code.

23
OLE, ACTIVEX, AND COM

Listing 23.1. Using the Hello Automation server from Visual Basic.
Sub Command1_Click () Dim hello As object Set hello = CreateObject(HELLO.Application) hello.text = Hello from Visual Basic! End Sub

As this Visual Basic code demonstrates, the Hello application exposes exactly one property that is a property of the application object. This text property determines what text is shown in the middle of the applications window and can be both read from and written to. Figure 23.1 shows the Hello application after its text has been changed from within Visual Basic.

428

OLE, COM, and MFC Applications PART IV

FIGURE 23.1.
Manipulating the Hello server from Visual Basic.

NOTE
Visual Basic is an excellent tool for testing automation servers. As this example demonstrates, you can test a server by adding a button to a Visual Basic form and writing a few lines of code in mere seconds. However, if you do not have Visual Basic installed on your system, do not despair; with many versions of Visual C++, Microsoft has included the program disptest.exe, a simplified version of Visual Basic 3.0, for the express purpose of aiding Automation server development. This tool sadly is missing from Visual C++5, but I have high hopes that its loss is due to a mere oversight and it will be included in upcoming versions of Visual C++.

The Hello Server Application


Listing 23.2 shows the complete Hello server application. We begin our review of its operations at the WinMain function. But first, a note: For the sake of compactness, this application contains no error checking code. If it works, it works; if it doesnt, strange things are bound to happen.

Listing 23.2. An Automation server.


#include <windows.h> #include <initguid.h> #ifndef INITGUID #define INITGUID #endif DEFINE_GUID(CLSID_CHello, 0xfeb8c280, 0xfd2d, 0x11ce, 0x87, 0xc3, 0x0, 0x40, 0x33, 0x21, 0xbf, 0xac); static PARAMDATA rgpDataTEXT = { OLESTR(TEXT), VT_BSTR }; enum IMETH_CTEXT { IMETH_SET = 0, IMETH_GET,

OLE, ActiveX, and the Component Object Model CHAPTER 23


}; enum IDMEMBER_CTEXT { IDMEMBER_TEXT = DISPID_VALUE }; static METHODDATA rgmdataCHello[] = { { OLESTR(TEXT), &rgpDataTEXT, IDMEMBER_TEXT, IMETH_SET, CC_CDECL, 1, DISPATCH_PROPERTYPUT, VT_EMPTY }, { OLESTR(TEXT), NULL, IDMEMBER_TEXT, IMETH_GET, CC_CDECL, 0, DISPATCH_PROPERTYGET, VT_BSTR } }; INTERFACEDATA idataCHello = { rgmdataCHello, 2 }; class CHello; class CText { public: STDMETHOD_(void, Set)(BSTR text); STDMETHOD_(BSTR, Get)(void); CText(CHello *pHello, char *p = NULL); ~CText(); void Paint(); HWND m_hwnd; private: char *m_text; CHello *m_pHello; }; class CHello : public IUnknown { public: static CHello *Create(char *p); STDMETHOD(QueryInterface)(REFIID riid, void **ppv); STDMETHOD_(unsigned long, AddRef)(void); STDMETHOD_(unsigned long, Release)(void); CHello(char *p = NULL); CText m_text; private: IUnknown *m_punkStdDisp; unsigned long m_refs; }; class CHelloCF : public IClassFactory { public: static IClassFactory *Create(); STDMETHOD(QueryInterface)(REFIID riid, void **ppv); STDMETHOD_(unsigned long, AddRef)(void); STDMETHOD_(unsigned long, Release)(void);

429

23
OLE, ACTIVEX, AND COM

continues

430

OLE, COM, and MFC Applications PART IV

Listing 23.2. continued


STDMETHOD(CreateInstance)(IUnknown *punkOuter, REFIID riid, void **ppv); STDMETHOD(LockServer)(BOOL fLock); CHelloCF() { m_refs = 1; } private: unsigned long m_refs; }; CHello *pHello; CText::CText(CHello *pHello, char *p) { m_pHello = pHello; if (p != NULL) { m_text = new char[strlen(p) + 1]; strcpy(m_text, p); } else m_text = NULL; m_hwnd = NULL; } CText::~CText() { delete[] m_text; } STDMETHODIMP_(void) CText::Set(BSTR p) { char *bf; int size; size = WideCharToMultiByte(CP_ACP, NULL, p, -1, NULL, 0, NULL, NULL); bf = new char[size]; WideCharToMultiByte(CP_ACP, NULL, p, -1, bf, size, NULL, NULL); delete[] m_text; if (p != NULL) { m_text = new char[strlen(bf) + 1]; strcpy(m_text, bf); } else m_text = NULL; if (m_hwnd != NULL) InvalidateRect(m_hwnd, NULL, TRUE); } STDMETHODIMP_(BSTR) CText::Get() { static WCHAR *wbf; BSTR bbf; int size; size = MultiByteToWideChar(CP_ACP, 0, m_text, -1, NULL, 0); wbf = new WCHAR[size]; MultiByteToWideChar(CP_ACP, 0, m_text, -1, wbf, size); bbf = SysAllocString(wbf); delete[] wbf;

OLE, ActiveX, and the Component Object Model CHAPTER 23


return bbf; } void CText::Paint() { HDC hDC; PAINTSTRUCT paintStruct; RECT clientRect; if (m_text != NULL) { hDC = BeginPaint(m_hwnd, &paintStruct); if (hDC != NULL) { GetClientRect(m_hwnd, &clientRect); DPtoLP(hDC, (LPPOINT)&clientRect, 2); DrawText(hDC, m_text, -1, &clientRect, DT_CENTER | DT_VCENTER | DT_SINGLELINE); EndPaint(m_hwnd, &paintStruct); } } } CHello *CHello::Create(char *p) { ITypeInfo *pTI; IUnknown *pUnk; CHello *pHello = new CHello(p); pHello->AddRef(); CreateDispTypeInfo(&idataCHello, LOCALE_SYSTEM_DEFAULT, &pTI); CreateStdDispatch(pHello, &(pHello->m_text), pTI, &pUnk); pTI->Release(); pHello->m_punkStdDisp = pUnk; return pHello; } STDMETHODIMP CHello::QueryInterface(REFIID riid, void **ppv) { if (IsEqualIID(riid, IID_IUnknown)) *ppv = this; else if (IsEqualIID(riid, IID_IDispatch)) return m_punkStdDisp->QueryInterface(riid, ppv); else { *ppv = NULL; return ResultFromScode(E_NOINTERFACE); } AddRef(); return NOERROR; } STDMETHODIMP_(unsigned long) CHello::AddRef() { return ++m_refs; } STDMETHODIMP_(unsigned long) CHello::Release()

431

23
OLE, ACTIVEX, AND COM

continues

432

OLE, COM, and MFC Applications PART IV

Listing 23.2. continued


{ if (--m_refs == 0) { if(m_punkStdDisp != NULL) m_punkStdDisp->Release(); PostQuitMessage(0); delete this; return 0; } return m_refs; } #pragma warning(disable:4355) CHello::CHello(char *p) : m_text(this, p) { m_refs = 0; } IClassFactory *CHelloCF::Create() { return new CHelloCF; } STDMETHODIMP CHelloCF::QueryInterface(REFIID riid, void **ppv) { if(IsEqualIID(riid, IID_IUnknown) || IsEqualIID(riid, IID_IClassFactory)) { AddRef(); *ppv = this; return NOERROR; } *ppv = NULL; return ResultFromScode(E_NOINTERFACE); } STDMETHODIMP_(unsigned long) CHelloCF::AddRef() { return m_refs++; } STDMETHODIMP_(unsigned long) CHelloCF::Release() { if (--m_refs == 0) { delete this; return 0; } return m_refs; } STDMETHODIMP CHelloCF::CreateInstance(IUnknown *punkOuter, REFIID riid, void **ppv) { if(punkOuter != NULL) return ResultFromScode(CLASS_E_NOAGGREGATION);

OLE, ActiveX, and the Component Object Model CHAPTER 23


return pHello->QueryInterface(riid, ppv); } STDMETHODIMP CHelloCF::LockServer(BOOL fLock) { return NOERROR; } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_PAINT: pHello->m_text.Paint(); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; IClassFactory *pHelloCF; unsigned long dwHelloCF = 0; unsigned long dwRegHello = 0; OleInitialize(NULL); pHello = CHello::Create(Hello, World!); pHelloCF = CHelloCF::Create(); CoRegisterClassObject(CLSID_CHello, pHelloCF, CLSCTX_LOCAL_SERVER, REGCLS_MULTIPLEUSE, &dwHelloCF); RegisterActiveObject(pHello, CLSID_CHello, NULL, &dwRegHello); pHelloCF->Release(); if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);

433

23
OLE, ACTIVEX, AND COM

continues

434

OLE, COM, and MFC Applications PART IV

Listing 23.2. continued


pHello->m_text.m_hwnd = hwnd; ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); RevokeActiveObject(dwRegHello, NULL); CoRevokeClassObject(dwHelloCF); pHello->Release(); OleUninitialize(); return msg.wParam; }

WinMain begins with a call to OleInitialize. Calling this function is necessary to initialize the OLE/COM libraries.

Next, two objects are created: an object of type CHello and an object of type CHelloCF. The first represents the interfaces of the application object; the second provides an IClassFactory interface. More about this in a moment. The availability of the class object is published through the call to CoRegisterClassObject. The active object that the class represents is registered through the call to RegisterActiveObject. Subsequently, the class object can be released. The rest of WinMain simply implements a plain window in which the applications text will be displayed and a standard message loop. When the application is about to terminate, calls are made to revoke the COM registrations, the application object is destroyed, and the OLE/COM library is uninitialized. Before we start exploring the CHello and CHelloCF classes, we should take a brief peek at the
WndProc function. This function processes two messages. When a WM_PAINT message is received,

the content of the applications window is refreshed with data taken from the applications one and only CHello object. When a WM_DESTROY message is received, the application terminates. At the heart of the automation implementation are the classes CHello and CHelloCF. Both classes implement the IUnknown interface. The implementations of the AddRef and Release functions are trivial and require little explanation. In the implementations of QueryInterface, CHello responds to requests for IUnknown and IDispatch; CHelloCF responds to requests for IUnknown and IClassFactory.
CHelloCF provides a simple implementation of CreateInstance, in which it returns a pointer to the appropriate interface provided by CHello.

Both

and CHelloCF objects are created through a static member function named The creation of CHelloCF is trivial. However, in CHello::Create some odd things are taking place. In order to provide an IDispatch interface, CHello relies on a standard implementation; this implementation is created through the calls to CreateDispTypeInfo and CreateStdDispatch. The information used by the standard implementation is encoded in the form of structures at the top of the file.
CHello Create.

OLE, ActiveX, and the Component Object Model CHAPTER 23

435

The actual implementations of the get and set methods that implement the text property are contained in a third class, CText. The first two member functions of this class, Get and Set, retrieve the value of its m_text member variable and set the value of that variable, respectively. An additional member function, Paint, is used to display the text within the window identified by the m_hwnd member variable. The Set member function causes this window to be redrawn (thus ensuring that the modified text is displayed) by calling InvalidateRect. Notice how the CText::Get and CText::Set functions handle strings; in particular, the conversion between OLE strings and plain ASCII strings that can be displayed by the application. Also notice how, in accordance with memory allocation rules, CText::Set does not free the memory allocated for its BSTR parameter. Nor does CText::Get free the memory it allocates for its return value; that will be freed by the caller (the OLE/COM library). The CLSID for this server has been obtained using the guidgen.exe application. Finally, here comes the horror of horrors: This application can simply be compiled from the command line! (If you abhor so-called examples in which the makefile alone exceeds the length of this simple program here, you are not alone.) To compile this program, type the following:
cl hello.cpp user32.lib gdi32.lib ole32.lib oleaut32.lib uuid.lib

Of course, if you plan to experiment with OLE/COM using this application as a basis and want to use the debugging features of Visual Studio, you can easily create a Visual C++ project file for this program.

23
OLE, ACTIVEX, AND COM

Registering and Running the Server


Although as soon as it is compiled the Hello server can be run standalone, in order for it to work as an automation server, entries must be made in the Registry. (Well, you didnt really expect a 300-line example program to do that for you, did you?) The entries shown in Listing 23.3 must be made in the Registry.

Listing 23.3. Registry entries.


HKEY_CLASSES_ROOT\ HELLO.Application = HELLO Application HKEY_CLASSES_ROOT\ HELLO.Application\CLSID = {FEB8C280-FD2D-11ce-87C3-00403321BFAC} HKEY_CLASSES_ROOT\ CLSID\{FEB8C280-FD2D-11ce-87C3-00403321BFAC} = HELLO Application HKEY_CLASSES_ROOT\ CLSID\{FEB8C280-FD2D-11ce-87C3-00403321BFAC}\LocalServer32 = HELLO.EXE /Automation HKEY_CLASSES_ROOT\ CLSID\{FEB8C280-FD2D-11ce-87C3-00403321BFAC}\ProgId = HELLO.Application

Note that unless the location of the Hello executable is in your path, it may be necessary to change the LocalServer32 key to reflect the complete pathname of the executable.

436

OLE, COM, and MFC Applications PART IV

Adding these entries to the Registry manually can be error-prone. To avoid potentially corrupting the Registry, you can add the entries using the Registry Editors import feature. Copy the entries in Listing 23.3 into an ASCII file (such as hello.reg). Note that the entries cannot be broken up as they appear here on the printed page; each of the five entries must be presented in a single line. This file can then be added to the Registry using the import feature of either the Windows 95/98 or the Windows NT version of the Registry Editor.

Accessing the Server from C++


Writing a general-purpose Automation client is a daunting task: youd be creating a whole new programming environment, something comparable to Visual Basic. However, it is possible to write server-specific code in C++. Listing 23.4 provides a C++ implementation of a client that works with the Automation server presented here.

Listing 23.4. An Automation client in C++.


#include <windows.h> #include <stdio.h> void main(void) { CLSID clsid; LPUNKNOWN punk; LPDISPATCH pdisp; DISPID dispid, dispidNamed = DISPID_PROPERTYPUT; OLECHAR *pszProp = Ltext; OLECHAR *pszVal = LHello from Visual C++!; DISPPARAMS dispparams; UINT uArgErr; VARIANTARG vArg; OleInitialize(NULL); CLSIDFromProgID(OLESTR(HELLO.Application), &clsid); CoCreateInstance(clsid, NULL, CLSCTX_SERVER, IID_IUnknown, (LPVOID *)&punk); punk->QueryInterface(IID_IDispatch, (LPVOID *)&pdisp); punk->Release(); pdisp->GetIDsOfNames(IID_NULL, &pszProp, 1, LOCALE_SYSTEM_DEFAULT, &dispid); dispparams.cArgs = 1; dispparams.cNamedArgs = 1; dispparams.rgdispidNamedArgs = &dispidNamed; dispparams.rgvarg = &vArg; vArg.vt = VT_BSTR; vArg.bstrVal = SysAllocString(pszVal); pdisp->Invoke(dispid, IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_PROPERTYPUT, &dispparams, NULL, NULL, &uArgErr); SysFreeString(vArg.bstrVal); pdisp->Release(); OleUninitialize(); }

Functionally, this code is equivalent to the Visual Basic code snippet shown in Listing 23.1. It is a lot more verbose because the C++ language does not provide intrinsic support for COM objects and their methods and properties like Visual Basic does.

OLE, ActiveX, and the Component Object Model CHAPTER 23

437

Because of this, several tasks that are performed implicitly by Visual Basic must be spelled out in C++ code. First, the OLE libraries must be initialized. Next, the object type must be identified and an instance must be created. Following that, we must find the interface that enables us to update the HELLO objects text property. Even the update is somewhat less straightforward: instead of a simple assignment statement as in Visual Basic, we must convert our string to a format compatible with Automation and invoke a method that is responsible for setting the property value. Finally, we must perform some explicit cleanup, including freeing memory used by the string, releasing the Automation object, and uninitializing the OLE library. This program can be compiled from the command line as follows:
cl hcl.cpp ole32.lib oleaut32.lib

Summary
The complex technology behind OLE and ActiveX is at the core of many modern Windows applications. At the heart of OLE and ActiveX is the Component Object Model (COM). COM objects are accessed through a set of interfaces, each consisting of a set of methods. The standard defines the interface but does not define the implementation of the interface, which is completely at the discretion of the programmer providing that implementation (to the point of choosing a programming language for the implementation). COM objects are black boxes. Although they represent reusable components, reusability is not possible in the fashion of object-oriented languages. COM objects can contain other COM objects and delegate the implementation of specific interfaces to other COM objects; however, you cannot derive an object from an existing object in the fashion of deriving a class from a base class in C++. COM objects are identified through GUIDs, globally unique interface identifiers. GUIDs are 128-byte numbers that are statistically unique; programmers need not worry about anyone else using the same GUID that has been generated through tools or functions provided for this purpose. All COM objects implement a specific interface, IUnknown. Through this interface, information about other interfaces a COM object may support can be obtained. Through the IClassFactory interface, implemented by COM class objects, applications that were written with no prior knowledge of an object can create such an object. Class objects are registered in a systemwide registration database (in Windows, the Registry) through their CLSID, which is just another GUID. Objects communicate with each other in one of two ways. If the method called is within the process space of the caller, the function implementing that method is called directly, with no overhead. However, if the method is outside the process space of the calling process, the call is made through an intervening infrastructure. This interface makes use of proxy objects and the

23
OLE, ACTIVEX, AND COM

438

OLE, COM, and MFC Applications PART IV

techniques of marshaling and unmarshaling, which are used to render the parameters of a method for transmission and unpackaging those parameters at the receiving end. COM objects can also be identified by name through monikers. Various types of monikers can be used to identify objects stored in files or within other objects. Monikers can also be concatenated and used in conjunction with antimonikers to create new monikers. Using OLE for linking and embedding objects is based on the compound document technology. This technology relies, in addition to COM, on Structured Storage and Uniform Data Transfer. The former provides a means by which compound data can be stored in a single file in a hierarchical form reminiscent of directories and files in a file system. The latter represents a mechanism for communicating objects between applications. Other uses of COM include automation, ActiveX controls, and OLE drag and drop. Specialized applications can also implement custom COM interfaces using the Microsoft Windows Software Development Kit.

OLE Servers CHAPTER 24

439

OLE Servers

24

IN THIS CHAPTER
s Server Concepts 440 s Creating a Server Application with MFC 440 s Customizing a Skeleton Server 447

24
OLE SERVERS

440

OLE, COM, and MFC Applications PART IV

OLE servers or OLE component applications are applications that provide OLE components for use within OLE container applications. MFC and Visual C++ support the development of servers, containers, and server-containers in a variety of ways.

Server Concepts
This section offers a review of a few server-related concepts before we begin exploring how MFC supports the construction of OLE servers.

Full-Servers and Mini-Servers


There is a distinction between full-servers and mini-servers. A full-server is an OLE component application that can also be run standalone. A mini-server can only operate when launched from within a container application. Note that a mini-server should not be confused with the concept of an in-process server. An in-process server runs in the process space of the client application. A typical example of an OLE in-process server is an OLE control. Mini-servers are executable applications; however, they do not offer facilities that would make their execution in a stand-alone configuration meaningful. For example, a mini-server typically does not offer functions to save a file to disk, nor does it have printing capabilities on its own.

In-Place Editing
In-place editing represents the capability of presenting a server item within the container applications window. As simple as it sounds, it requires a complex series of interactions between the server and the container application. In addition to presenting the server item in a rectangular area within the container applications window, servers also take over the container applications toolbars and parts of their menu bars.

Server Activation
Servers are activated through verbs. Executing a verb means calling the method DoVerb of the IOleObject interface. A series of predefined verb values corresponds to editing the object inplace or within its own window, activating or hiding an object.

Creating a Server Application with MFC


OLE component server applications can be created using the Visual C++ AppWizard. Server capabilities can be easily added or modified through the ClassWizard.

OLE Servers CHAPTER 24

441

Using AppWizard to Create an Application Skeleton


To create an OLE server application through AppWizard, you must first specify an application that is either single-documentbased or multiple-documentbased. Server capabilities are not supported for dialog-based applications. If you wish to develop a server using a dialog-like interface (such a server would typically not support visual editing and in-place sessions), you can still use a view class based on CFormView. Regardless of whether your new program is an SDI or MDI application, you specify OLE server capabilities in Step 3 of AppWizard (Figure 24.1). Three forms of support are available. You can select mini-server, full-server, or container-server support for your application.

FIGURE 24.1.
Creating an OLE server through AppWizard.

On this page, you can also specify whether or not your application should use OLE compound files. Using OLE files requires more disk space, but it offers the advantages of improved performance and a standardized file structure. On this page of AppWizard, there are also check boxes for OLE automation and OLE controls. These options are not connected to OLE container and OLE component server support and are of no concern to us at this moment.

24
OLE SERVERS

The OLE Server Skeleton Application


This section examines a full-server skeleton application that was created by AppWizard. The OSRV application was created with default AppWizard settings as a multiple-document interface full-server application. Figure 24.2 shows the classes created by AppWizard for this application.

442

OLE, COM, and MFC Applications PART IV

FIGURE 24.2.
Classes created by AppWizard for an OLE server.

In what way do these classes differ from the classes that AppWizard creates for a non-OLE application? There are two readily visible differences: two new classes, CInPlaceFrame and COSRVSrvrItem. Another difference is less obvious; if you double-click on the class name COSRVDoc, you may notice that this class is not derived from CDocument as would be the case for a nonOLE application. Instead, this class is derived from COleServerDoc. The reason for these changes is easy to explain. First of all, the applications document class is now derived from COleServerDoc because it is this base class that provides a number of services that implement the applications document in a server environment. Among other things, this class implements interaction with an in-place frame, provides container notification functions, and has helper functions that assist in positioning items (possibly zoomed) within a container applications window during in-place editing. The role of the new class CInPlaceFrame is closely related. This class, derived from COleIPFrameWnd, represents the in-place frame window. During an in-place editing session, it is this window that takes over the role of the child frame window (or, in the case of SDI applications, the applications frame window). Figure 24.3 compares the relationship between frame windows and views during stand-alone and in-place sessions. The other new class, COSRVSrvrItem, is a class that represents the OLE interface during a session with a container application.

The Server Item


The COleServerItem-derived class representing the server item has a very special role. This class, among other things, implements the COM interfaces IOleObject and IDataObject, which facilitate its role in representing an OLE server item.
COleServerItem-derived server items are closely linked with COleDocument-derived documents. Their purpose is to represent all or part of a document in an OLE data transfer context. However, their role is not limited to OLE linking or embedding; other roles of COleServerItemderived objects include facilitating clipboard transfers and OLE drag and drop.

OLE Servers CHAPTER 24

443

FIGURE 24.3.
Comparing in-place and stand-alone sessions.

Container Application Container Document

Server Application

Server objects

Document objects View window

View window In-place frame window

Main frame window

Container frame and view windows

Among the overridable member functions of COleServerItem are OnDraw and Serialize. Both of these functions play a very important role in OLE. COleServerItem::Serialize is used to serialize an embedded item for storage in the container application. COleServerItem::OnDraw is called to render the items appearance; typically, this function is called to render the embedded item into a metafile device context. The resulting metafile object will be stored by the container application, thus avoiding the need to activate the server whenever the item needs to be redrawn. The default, AppWizard-supplied implementation of COleServerItem::Serialize (Listing 24.1) relies on the document classs Serialize member function to accomplish its task. While satisfactory in simple situations, this implementation needs to be revised if a COleServerItemderived object is used to represent only parts of a document. This is the case when the server enables OLE links and when COleServerItem is used to transfer portions of a document (the current selection) to the clipboard.

24
OLE SERVERS

Listing 24.1. AppWizard-supplied implementation of the Serialize member function of a server item class.
void COSRVSrvrItem::Serialize(CArchive& ar) { // COSRVSrvrItem::Serialize will be called by the framework if // the item is copied to the clipboard. This can happen // automatically through the OLE callback OnGetClipboardData. // A good default for the embedded item is simply to delegate // to the documents Serialize function. If you support // links, then you will want to serialize just a portion of // the document.

continues

444

OLE, COM, and MFC Applications PART IV

Listing 24.1. continued


if (!IsLinkedItem()) { COSRVDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); pDoc->Serialize(ar); } }

The default implementation of COleServerItem::OnDraw (Listing 24.2) does not perform any drawing. You must provide your own implementation of this function. In simple situations, the drawing function here may be a replica of the OnDraw member function of your applications view class.

Listing 24.2. AppWizard-supplied implementation of the OnDraw member function of a server item class.
BOOL COSRVSrvrItem::OnDraw(CDC* pDC, CSize& rSize) { // Remove this if you use rSize UNREFERENCED_PARAMETER(rSize); COSRVDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: set mapping mode and extent (The extent is usually // the same as the size returned from OnGetExtent) pDC->SetMapMode(MM_ANISOTROPIC); pDC->SetWindowOrg(0,0); pDC->SetWindowExt(3000, 3000); // TODO: add drawing code here. Optionally, fill in the // HIMETRIC extent. All drawing takes place in the metafile // device context (pDC). return TRUE; }

COleDocument and Document Items


The class COleDocument, which is the base class of COleServerDoc, offers a very useful capability. It can maintain a list of document items of type CDocItem. The CDocItem class is used, among other things, as the base class for COleServerItem; however, you can also use it to implement application-specific document items. Thus, it is possible to create a simple OLE server application without adding any code to the applications COleDocument-derived document class; the code provided by AppWizard already manages the list of CDocItem objects.

OLE Servers CHAPTER 24

445

The In-Place Frame Window


The in-place frame window essentially has the same functionality as the frame window in SDI applications. It is the parent window for the current view; it also manages menus and toolbars. However, it does so by cooperating with the container applications frame window, replacing the menus and toolbars of that frame window for the duration of the in-place session.

Modes of Operation and Resources


When you work with an OLE server, you must never forget about this dual role of the application. Did I say dual? Actually, a full OLE server has three basic modes of operation. It can operate in stand-alone mode, it can be used to edit an OLE item in its own window, and it can be used for in-place editing. This fact is also reflected in the AppWizard-generated resource file (Figure 24.4). As you can see, the application has three accelerators, four menus, and two different toolbars.

FIGURE 24.4.
Duplicate resources in an OLE server application.

24
OLE SERVERS
Of the four menus, two are found in every multiple document application; one is displayed when the application has no documents open; otherwise, the other one is used. An OLE server, however, has to provide a third menu reflecting the state when it is used to edit an embedded item. The basic difference between this and the regular menu bar is that the File menu contains the choices Update and Save Copy As, instead of Save and Save As. This is to reflect the fact that saving an embedded item implies serializing it into the container application; saving it under a different filename creates a standalone copy of the embedded item. The fourth menu is a very special one. This menu (Figure 24.5) is displayed when the server is used for in-place editing. What makes this menu so special is that it is incomplete; in place of two of its top-level menu items there are only separate bars.

446

OLE, COM, and MFC Applications PART IV

FIGURE 24.5.
The incomplete menu used during in-place editing.

During an in-place session, this menu and a similarly incomplete menu provided by the server are combed together, forming the menu seen by the user. This combing reflects the fact that during the in-place session, various functions are carried out by the server application, while other functions (for example, window management) remain the responsibility of the container application. Figure 24.6 shows how the complete menu is formed from the incomplete menus provided by the server and the container.

FIGURE 24.6.
How server and container menus are combined during inplace editing.

File

Window

From the container application

Edit View

Help

From the server application

Container Application File Edit View Window Help

What the user sees

If you know that there are separate menus, the need for separate acceleration tables is easy to understand. As for the toolbars, a brief look at them (Figure 24.7) quickly reveals that the toolbar used during in-place editing simply lacks those file-related functions that the server does not offer during such sessions.

OLE Servers CHAPTER 24

447

FIGURE 24.7.
Toolbars in an OLE server skeleton.

Running the Server Skeleton


As is, the server skeleton supplied by AppWizard can be compiled and run. Although it does not perform any useful function, the interaction between the server and the container works and can be demonstrated. For example, you can use the Windows 95 WordPad application to invoke the new server. Note that you must run the server standalone first in order for it to create the appropriate entries in the Registry. Afterwards, when running WordPad, you can insert a new object of type OSRV Document by selecting the Object command from its Insert menu. Figure 24.8 shows an in-place editing session using the OSRV server with WordPad.

FIGURE 24.8.
In-place editing session.

24
OLE SERVERS

Notice how the Edit and View menus, the toolbar, and the status bar have been taken over by the server application; WordPads toolbar, ruler, and menus, which are normally visible, disappear for the sessions duration.

Customizing a Skeleton Server


Next you want to add some very simple functionality to your skeleton server. The new OSRV server will do a very simple task: It will display a string in the middle of view windows. The string will be stored in the form of a member variable of the servers document class and will be serialized by that class.

448

OLE, COM, and MFC Applications PART IV

In order to make modification of the string possible, we must add a dialog. The dialog will be invoked when the user clicks within a view using the left mouse button. Although this is clearly a very simple example, it nevertheless demonstrates all the key aspects that one must pay attention to while developing a server. A real-life OLE server application is significantly more complex, but the basic steps and principles remain the same.

Modifying the Document


The first thing to do to implement the new behavior is to add a member variable to the document class, COSRVDoc. A member variable of type CString should be added in the Attributes section of the declaration of COSRVDoc:
// Attributes public: COSRVSrvrItem* GetEmbeddedItem() { return (COSRVSrvrItem*)COleServerDoc::GetEmbeddedItem(); } CString m_sData;

This member variable should also be initialized to some meaningful value. In the implementation file of COSRVDoc, modify COSRVDoc::OnNewDocument as follows:
BOOL COSRVDoc::OnNewDocument() { if (!COleServerDoc::OnNewDocument()) return FALSE; // TODO: add reinitialization code here // (SDI documents will reuse this document) m_sData = _T(Hello, World!); return TRUE; }

Adding Drawing Code


In the case of nonserver applications, we had to add drawing code to the appropriate view class. In the case of a server application, we also have to add drawing code to the OnDraw member function of the OLE server item class. To add drawing code to the view class, modify the implementation of COSRVView::OnDraw function as follows:
void COSRVView::OnDraw(CDC* pDC) { COSRVDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here CRect rect; CFont font, *pOldFont; GetClientRect(&rect); font.CreateStockObject(SYSTEM_FONT);

OLE Servers CHAPTER 24


pOldFont = pDC->SelectObject(&font); pDC->DPtoLP(&rect); pDC->TextOut((rect.left + rect.right) / 2, (rect.top + rect.bottom) / 2, pDoc->m_sData); pDC->SelectObject(pOldFont); }

449

A similar modification must be made to COSRVSrvrItem::OnDraw:


BOOL COSRVSrvrItem::OnDraw(CDC* pDC, CSize& rSize) { // Remove this if you use rSize UNREFERENCED_PARAMETER(rSize); COSRVDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: set mapping mode and extent (The extent is usually // the same as the size returned from OnGetExtent) pDC->SetMapMode(MM_ANISOTROPIC); pDC->SetWindowOrg(0,0); pDC->SetWindowExt(3000, 3000); // TODO: add drawing code here. Optionally, fill in the // HIMETRIC extent. All drawing takes place in the metafile // device context (pDC). CRect rect; CFont font, *pOldFont; rect.TopLeft() = pDC->GetWindowOrg(); rect.BottomRight() = rect.TopLeft() + pDC->GetWindowExt(); font.CreateStockObject(SYSTEM_FONT); pOldFont = pDC->SelectObject(&font); pDC->TextOut((rect.left + rect.right) / 2, (rect.top + rect.bottom) / 2, pDoc->m_sData); pDC->SelectObject(pOldFont); return TRUE; }

24
OLE SERVERS

In a more sophisticated application, the first part of this function would also be modified, reflecting the true size of the item (as opposed to the rather arbitrary size chosen here). Similarly, the COSRVSrvrItem::OnGetExtent function would also use a more sophisticated mechanism for determining the size of the embedded item. At this time, you can recompile and run the application. To test it from within an OLE container application, use the containers Insert Object function, and insert an OSRV document. Figure 24.9 shows the new OSRV application during an in-place session with the Windows 95 WordPad. If you play around with OSRV, you may notice that our attempt to center the text is not entirely successful. Instead of properly centering the text, we place its upper-left corner in the center of the drawing area. Shouldnt we remedy this, perhaps by calculating the texts extent or replacing the call to OutText with a call to the more sophisticated DrawText function?

450

OLE, COM, and MFC Applications PART IV

FIGURE 24.9.
OSRV and the Windows 95 WordPad.

Unfortunately, the answer is not so simple. Modifying the function COSRVView::OnDraw can be accomplished easily enough; however, you may find that whether you use DrawText with the DT_CENTER attribute or call GetTextExtent and manually try to position the text, your changes will not operate as expected in COSRVSrvrItem::OnDraw. The reason? While COSRVView::OnDraw functions with a real device context (representing a windows client area), COSRVSrvrItem::OnDraw draws into a metafile. Certain concepts make little or no sense in a metafile context; for instance, calculating the extent of text may not yield meaningful results as the true extent will only be known when the metafile is played back on a target device (with its own font configuration). We are not going to be concerned about this right now (consider it a documented bug); however, let this serve as a warning that the two OnDraw functions (one in the view class, the other in the OLE server item class) are invoked under very different circumstances and must be tested separately.

Adding a Dialog
In order to make the text displayed by OSRV changeable, we need to add a dialog where the user can enter or modify the text item. Adding the dialog is easy using the built-in dialog editor. The dialog, as shown in Figure 24.10, should contain a single edit field, IDC_TEXT, where the new text can be entered.

FIGURE 24.10.
The Window Text dialog in OSRV.

OLE Servers CHAPTER 24

451

To complete creation of this dialog, invoke the ClassWizard. Let the ClassWizard create a new class representing this dialog (CTextDlg) and add a member variable, m_sText, that represents the edit field IDC_TEXT. How will this dialog be invoked? To avoid having to mess with menus or toolbars, I decided to use the simplest possible method. The dialog will be invoked whenever a view window receives a WM_LBUTTONDOWN message; that is, whenever the user clicks within the view using the left mouse button. The simplest way to add a handler to the view class for these messages is to open the implementation file for the OSRV view class (OSRVView.cpp) and use the ClassWizard. In the ClassWizard, select the Message Maps tab and double-click the WM_LBUTTONDOWN item in the Messages list. A new handler for this message will be added automatically. The handler function is shown in Listing 24.3. Nothing mysterious here; the function simply creates and invokes the dialog and transfers the data between the dialog and the document. Note that in order for this function to compile successfully, you must also include the TextDlg.h header file at the top of the file OSRVView.cpp.

Listing 24.3. Implementation of COSRVView::OnLButtonDown.


void COSRVView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default // CView::OnLButtonDown(nFlags, point); CTextDlg dlg; COSRVDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); dlg.m_sText = pDoc->m_sData; if ((dlg.DoModal() == IDOK) && (dlg.m_sText != pDoc->m_sData)) { pDoc->m_sData = dlg.m_sText; pDoc->UpdateAllViews(NULL); pDoc->UpdateAllItems(NULL); pDoc->SetModifiedFlag(); }

24
OLE SERVERS

In addition to calling the document class member functions UpdateAllViews (to reflect the users entry in all views of the document) and SetModifiedFlag (to notify the document class that its contents have changed and may need to be saved), the function also calls UpdateAllItems. Through UpdateAllItems, it notifies containers that the contents of the embedded item have changed and may need to be redrawn.

452

OLE, COM, and MFC Applications PART IV

Serialization
There is one thing missing before we can call our application complete: Its data must be saved. Namely, the m_sData member variable of the document class needs to be serialized in COSRVDoc::Serialize. This is very easily accomplished:
void COSRVDoc::Serialize(CArchive& ar) { if (ar.IsStoring()) { // TODO: add storing code here ar << m_sData; } else { // TODO: add loading code here ar >> m_sData; } }

At this time, our simple server application is complete and can be recompiled.

Registering the New Application


You might have noticed that we have run the OSRV application without ever attempting to enter anything into the Windows Registry. How do clients, such as the Windows 95 WordPad, know about our application? The reason it was not necessary to update the Registry manually is because AppWizardgenerated applications register themselves. Every time such an application is run standalone, it creates or updates the appropriate entries in the Registry. This also applies to mini-servers; although a mini-server would not typically be run standalone, you can do so for the purpose of registering. However, AppWizard also creates a file that contains all relevant Registry entries. In the case of OSRV, this file is called OSRV.reg. It contains the following Registry entries (all under the key HKEY_CLASSES_ROOT):
OSRV.Document = OSRV Document OSRV.Document\protocol\StdFileEditing\server = OSRV.EXE OSRV.Document\protocol\StdFileEditing\verb\0 = &Edit OSRV.Document\Insertable = OSRV.Document\CLSID = {494CDF20-FE4B-11CE-87C3-00403321BFAC}

It is through these items that an OLE container can identify OSRV documents as insertable objects. The last of these items identifies the CLSID under which additional information about the server can be found (note that your CLSID may be different from mine). A further set of Registry entries is made using this CLSID, under the key HKEY_CLASSES_ROOT\CLSID:
{494CDF20-FE4B-11CE-87C3-00403321BFAC} = OSRV Document {494CDF20-FE4B-11CE-87C3-00403321BFAC}\DefaultIcon = OSRV.EXE,1

OLE Servers CHAPTER 24


{494CDF20-FE4B-11CE-87C3-00403321BFAC}\LocalServer32 = OSRV.EXE {494CDF20-FE4B-11CE-87C3-00403321BFAC}\ProgId = OSRV.Document {494CDF20-FE4B-11CE-87C3-00403321BFAC}\MiscStatus = 32 {494CDF20-FE4B-11CE-87C3-00403321BFAC}\AuxUserType\3 = OSRV {494CDF20-FE4B-11CE-87C3-00403321BFAC}\AuxUserType\2 = OSRV {494CDF20-FE4B-11CE-87C3-00403321BFAC}\Insertable = {494CDF20-FE4B-11CE-87C3-00403321BFAC}\verb\1 = &Open,0,2 {494CDF20-FE4B-11CE-87C3-00403321BFAC}\verb\0 = &Edit,0,2 {494CDF20-FE4B-11CE-87C3-00403321BFAC}\InprocHandler32 = ole32.dll

453

Summary
OLE servers, or OLE component servers, are applications that maintain OLE component items that can be used from within OLE container applications. Creating an MFC-based OLE server application is easy using the Visual C++ AppWizard. All you need to do is to specify server support during AppWizard Step 3. OLE server capability is supported for single- or multiple-documentbased applications but not for dialog-based programs. However, you can develop an OLE server with a dialog-like interface using the CFormView class. The MFC distinguishes between mini-servers, full-servers, and container-servers. A mini-server is an application that can only be used in conjunction with embedded objects; a full-server is a program that can also operate in a stand-alone mode and can save its data to files. A container server is an OLE server that also offers container functionality. To create an OLE server with AppWizard, specify the desired server behavior in AppWizard Step 3. When compared to applications that do not support OLE, an OLE server offers two new classes. The server item class, derived from COleServerItem, represents an embedded item in the container application and offers the IOleObject COM interface through which the container application can communicate with the server. The in-place frame window class, derived from COleIPFrameWnd, represents the frame window during in-place sessions and manages the interaction between the server and the frame window of the container. The in-place frame window provides the functionality that is necessary for the menus and toolbars of the container and the server application to coexist. During an in-place session, the server application takes over the containers toolbar and status bars. The menu bar that is visible during such a session is created from a combination of menus provided by the container and the server. The server item class provides two member functions of fundamental importance. Its OnDraw member function is used to draw an image of the OLE item into a metafile device context; this image is used later by the container to display the item without invoking the server. The Serialize member function is used to serialize the item for storage in the container document.

24
OLE SERVERS

454

OLE, COM, and MFC Applications PART IV

Customizing an AppWizard-generated server skeleton consists of the following steps: 1. After having created the server skeleton, implement basic functionality by adding member variables and code to the applications document and view classes. Add other elements (resources, dialogs) as appropriate. 2. Modify the server items OnDraw member function to draw the item into a metafile device context. This is the image that will be displayed by containers when there is no active session with the server. 3. Modify the server items Serialize member function to serialize the item for embedding. If your application supports only embedded items, you can use the default implementation that relies on your document classs serialization function. If you support links, or if you use the server item class to facilitate clipboard transfers, add code here that serializes selected items only. Your application does not have to be registered; it registers itself (or updates the appropriate registry entries) whenever it is run as a stand-alone application. However, AppWizard also emits a file that can be used for registering the application manually or from within an installation program.

OLE Containers CHAPTER 25

455

OLE Containers

25

IN THIS CHAPTER
s Creating a Container Application Through AppWizard 456 s Customizing the Application 462

25
OLE CONTAINERS

456

OLE, COM, and MFC Applications PART IV

OLE containers are applications that manage, in addition to native data, a set of OLE server items. The MFC Library provides a high level of support for creating OLE containers. Using an AppWizard-generated container application skeleton, it is possible to build, with a minimal amount of added code, a decent working container application. In this chapter, you first review the AppWizard-generated skeleton for a container application. Subsequently, we add the necessary code for this application to handle the selection and editing of multiple embedded objects.

Creating a Container Application Through AppWizard


The Visual C++ AppWizard supports the creation of skeleton applications for two types of containers: simple containers and container-servers. The latter term refers to applications that offer both OLE container and OLE component server functionality. However, since the two areas of functionality are quite distinct, this chapter focuses on simple containers.

Creating the Skeleton Application


To create a container application through AppWizard, you first must specify a singledocumentbased or multiple-documentbased application. Container capabilities are not supported for dialog-based applications. While it is possible to create a container application based on the view class CFormView, I do not see how the dialog template-based view class and the visual embedding and in-place editing associated with OLE components can coexist. OLE container support in the new application skeleton must be specified in AppWizard Step 3 (Figure 25.1). You can either specify a container application or a container-server. Other than setting the container application option, I used the default settings when I created the OCON application skeleton through AppWizard. Throughout the rest of this chapter, it is this OCON application that we review and modify.

OLE Containers CHAPTER 25

457

FIGURE 25.1.
Creating an OLE container through AppWizard.

The OLE Container Skeleton Application


By looking at the set of classes created by AppWizard (Figure 25.2), you can see that AppWizard created one extra class when compared to a skeleton application with no OLE container support. This extra class is named COCONCntrItem and represents OLE components that are contained within the container applications documents.

FIGURE 25.2.
AppWizard-generated classes for an OLE container.

Of course, the appearance of a new class is not the only difference between an OLE container and other applications. There are also noticeable differences between the implementations of other classes.

25
OLE CONTAINERS

458

OLE, COM, and MFC Applications PART IV

Running the Skeleton Container


Before you proceed with analyzing the skeleton container application, running it may be a good idea. It will give you a picture of how the container works and what functionality we need to add to enhance its usefulness. To explore the container capabilities of the OCON application, compile and run this application and then select the Insert New Object command from its Edit menu. This command displays the dialog shown in Figure 25.3, where you can select the type of object you wish to insert into the document.

FIGURE 25.3.
The Insert Object dialog.

Next you want to insert a Bitmap Image object. This object type is available on all Windows 95/98/NT systems that have the Paint application installed. Figure 25.4 shows a session while the bitmap object is being edited inside the OCON container. You may notice a few peculiarities right here, pointing at areas of code that must be improved. First of all, there does not seem to be a way to terminate the in-place session. Normally you would expect that clicking outside the in-place editing area would terminate the session; however, this does not seem to happen. The reason is simple: As you see shortly, the OCON skeleton application does not provide a handler function for mouse clicks, so it is no wonder that nothing happens.

OLE Containers CHAPTER 25

459

FIGURE 25.4.
In-place editing of a bitmap object.

Another peculiarity can be noticed if you attempt to move or resize the in-place area. This area can be resized by dragging any of the eight resize handles (small black squares) around its border; it can also be moved by dragging the shaded border area. But look what happens (Figure 25.5): After the in-place area is moved, another image of your drawing appears at the original location!

FIGURE 25.5.
Discrepancy between in-place frame and embedded item positions.

25
OLE CONTAINERS

460

OLE, COM, and MFC Applications PART IV

As it turns out, that image is at the position where the container application thinks the image should be drawn. Code that would properly update this location to reflect any changes made during an in-place editing session does not yet exist. Although there is no trivial way to end an in-place editing session, you can actually save the container file even while an in-place session is active. After saving the file using the File Save command, you can terminate the in-place session by simply closing the document window altogether. Upon reopening the file, you can see that the embedded object was saved correctly and is redisplayed at its original location. At this stage, it is possible to either edit this object again or insert a new object, both through commands in the Edit menu. If you insert a new object, it will be positioned on top of the first object in the container. For this reason, it is not possible to see at this stage the third oddity concerning AppWizardgenerated container skeletons; namely, that the skeleton application only displays one object at a time. As you will see, this is a cosmetic problem only; the application is actually capable of saving container files with more than one object; it is the OnDraw member function of its view class that requires modification.

The Skeleton Container Code


It is time to stop playing with your new container application and start looking at how its functions are implemented. How are new objects inserted into the document? How are the new objects represented and saved with a container file? How are they drawn? How are in-place editing sessions managed? These are the questions that are answered in this section. First, take a look at how items in a container are represented. The AppWizard generated a new class, COCONCntrItem, for this purpose. This class is derived from COleClientItem and comes with a default implementation of several member functions (Figure 25.6).

FIGURE 25.6.
Container item class member functions.

OLE Containers CHAPTER 25

461

This class implements the necessary OLE interfaces for in-place editing. It also provides a series of member functions (such as Serialize) that enable it to exist within the MFC application framework. Looking at the implementation of this class, you can notice a few shortcomings. In particular, look at the implementations of the OnChangeItemPosition and OnGetItemPosition member functions (Listing 25.1). As you can see, OnGetItemPosition always returns an arbitrary, fixed position; on the other hand, OnChangeItemPosition, although it calls the base class implementation, does not make note of the new position in any way. (No wonder OnGetItemPosition cannot return a meaningful value!)

Listing 25.1. The default implementations of COCONCntrItem member functions OnChangeItemPosition and OnGetItemPosition.
BOOL COCONCntrItem::OnChangeItemPosition(const CRect& rectPos) { ASSERT_VALID(this); // During in-place activation COCONCntrItem::OnChangeItemPosition // is called by the server to change the position of the in-place // window. Usually, this is a result of the data in the server // document changing such that the extent has changed or as a // result of in-place resizing. // // The default here is to call the base class, which will call // COleClientItem::SetItemRects to move the item // to the new position. if (!COleClientItem::OnChangeItemPosition(rectPos)) return FALSE; // TODO: update any cache you may have of the items // rectangle/extent return TRUE; } void COCONCntrItem::OnGetItemPosition(CRect& rPosition) { ASSERT_VALID(this); // During in-place activation, COCONCntrItem::OnGetItemPosition // will be called to determine the location of this item. The // default implementation created from AppWizard simply returns a // hard-coded rectangle. Usually, this rectangle would reflect // the current position of the item relative to the view used for // activation. // You can obtain the view by calling // COCONCntrItem::GetActiveView. // TODO: return correct rectangle (in pixels) in rPosition rPosition.SetRect(10, 10, 210, 210); }

25
OLE CONTAINERS

462

OLE, COM, and MFC Applications PART IV

As the AppWizard-generated comments also imply, you will have to revisit this class shortly to manage these position changes. How are COCONCntrItem objects represented in your document class? Strangely enough, there is no additional code in your skeleton application for this purpose. The only change relative to a noncontainer application is that your applications document class, COCONDoc, is now derived from COleDocument as opposed to CDocument. COleDocument has the wonderful capability of managing and storing a list of CDocItem-derived items. When a new item of class COCONCntrItem (derived from COleClientItem which, in turn, is derived from CDocItem) is created, it is automatically added to the container documents list of items. The container document, in turn, can serialize itself, including this list of CDocItem-derived objects, without any additional code. If you add items of your own design to an OLE container application, you can rely on this capability. You can add your own CDocItem-derived items to the container, and the container will handle them correctly. The only catch is that in your application code you will have to be careful in how you handle application-specific items, container items, and (if the application also acts as an OLE server) server items. One possible solution is to create a series of wrapper functions that determine a particular objects type using MFC runtime type information. The differences between the implementations of the document classes in a container and a noncontainer application were relatively minor (although the effects of the difference in the base class from which the document classes are derived are rather significant). In contrast, the difference between the implementations of view classes in the two cases is very significant. The implementation file of the container applications view class is nearly twice as long, with several additional member functions (Figure 25.7).

FIGURE 25.7.
Container view class member functions.

OLE Containers CHAPTER 25

463

The implementation of these member functions will, in fact, answer our questions concerning the peculiar behavior of the skeleton container. The first thing to notice in the declaration of COCONView is the presence of a new member variable, m_pSelection:
class COCONView : public CView { protected: // create from serialization only COCONView(); DECLARE_DYNCREATE(COCONView) // Attributes public: COCONDoc* GetDocument(); COCONCntrItem* m_pSelection;

This member variable represents a very simple item selection mechanism; the mechanism only allows a single document item to be selected at any given time. While in many sophisticated applications such a selection mechanism would be completely inadequate, in our effort to build a simple container application, it is going to be sufficient. The m_pSelection member variable plays a role in the implementation of the OnDraw member function, answering our question with respect to why only a single item is drawn when a document with more than one embedded item is loaded. The implementation of this function (Listing 25.2) simply does not draw any other items.

Listing 25.2. The default implementation of COCONView::OnDraw.


void COCONView::OnDraw(CDC* pDC) { COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here // TODO: also draw all OLE items in the document // Draw the selection at an arbitrary position. This code // should be removed once your real drawing code is // implemented. This position corresponds exactly to the // rectangle returned by COCONCntrItem, to give the effect of // in-place editing. // TODO: remove this code when final draw code is complete. if (m_pSelection == NULL) { POSITION pos = pDoc->GetStartPosition(); m_pSelection = (COCONCntrItem*)pDoc->GetNextClientItem(pos); } if (m_pSelection != NULL) m_pSelection->Draw(pDC, CRect(10, 10, 210, 210)); }

25
OLE CONTAINERS

464

OLE, COM, and MFC Applications PART IV

The other shortcoming of this default implementation is made obvious by the comment embedded in this AppWizard-generated code. The selection item is drawn at an arbitrary, fixed position; it does not reflect in any way any positional changes that may happen during an inplace editing session. Clearly, a modified implementation must be made in conjunction with changes to COCONCntrItem::OnChangeItemPosition and COCONCntrItem::OnGetItemPosition. The AppWizard-generated view class implementation contains five additional member functions that are completely new (Listing 25.3). These member functions implement the insertion of new objects and also manage the in-place session. You look at these functions one by one.

Listing 25.3. New view class member functions.


BOOL COCONView::IsSelected(const CObject* pDocItem) const { // The implementation below is adequate if your selection consists // of only COCONCntrItem objects. To handle different selection // mechanisms, the implementation here should be replaced. // TODO: implement this function that tests for a selected OLE // client item return pDocItem == m_pSelection; } void COCONView::OnInsertObject() { // Invoke the standard Insert Object dialog box to obtain // information for new COCONCntrItem object. COleInsertDialog dlg; if (dlg.DoModal() != IDOK) return; BeginWaitCursor(); COCONCntrItem* pItem = NULL; TRY { // Create new item connected to this document. COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); pItem = new COCONCntrItem(pDoc); ASSERT_VALID(pItem); // Initialize the item from the dialog data. if (!dlg.CreateItem(pItem)) AfxThrowMemoryException(); // any exception will do ASSERT_VALID(pItem); // If item created from class list (not from file) then // launch the server to edit the item. if (dlg.GetSelectionType() == COleInsertDialog::createNewItem) pItem->DoVerb(OLEIVERB_SHOW, this);

OLE Containers CHAPTER 25


ASSERT_VALID(pItem); // As an arbitrary user interface design, this sets the // selection to the last item inserted. // TODO: reimplement selection as appropriate for your // application m_pSelection = pItem; //set selection to last inserted item pDoc->UpdateAllViews(NULL); } CATCH(CException, e) { if (pItem != NULL) { ASSERT_VALID(pItem); pItem->Delete(); } AfxMessageBox(IDP_FAILED_TO_CREATE); } END_CATCH EndWaitCursor(); } // The following command handler provides the standard keyboard // user interface to cancel an in-place editing session. Here, // the container (not the server) causes the deactivation. void COCONView::OnCancelEditCntr() { // Close any in-place active item on this view. COleClientItem* pActiveItem = GetDocument()->GetInPlaceActiveItem(this); if (pActiveItem != NULL) { pActiveItem->Close(); } ASSERT(GetDocument()->GetInPlaceActiveItem(this) == NULL); } // Special handling of OnSetFocus and OnSize are required for a // container when an object is being edited in-place. void COCONView::OnSetFocus(CWnd* pOldWnd) { COleClientItem* pActiveItem = GetDocument()->GetInPlaceActiveItem(this); if (pActiveItem != NULL && pActiveItem->GetItemState() == COleClientItem::activeUIState) { // need to set focus to this item if it is in the same view CWnd* pWnd = pActiveItem->GetInPlaceWindow(); if (pWnd != NULL) { pWnd->SetFocus(); // dont call the base class return; } }

465

25
OLE CONTAINERS

continues

466

OLE, COM, and MFC Applications PART IV

Listing 25.3. continued


CView::OnSetFocus(pOldWnd); } void COCONView::OnSize(UINT nType, int cx, int cy) { CView::OnSize(nType, cx, cy); COleClientItem* pActiveItem = GetDocument()->GetInPlaceActiveItem(this); if (pActiveItem != NULL) pActiveItem->SetItemRects(); }

The IsSelected member function is called by the framework to determine whether a particular item is selected within the view. Its implementation is straightforward and obvious. The OnInsertObject member function implements the Insert New Object command in the Edit menu. This function relies on the capabilities of one of the OLE common dialog classes, COleInsertDialog. This common dialog can be used not only to display the list of insertable OLE items that are available on your system, but its member function CreateItem can actually be used to create a new item of the type selected by the user. If the item is freshly created, OnInsertObject launches the server application; items created from a file are simply inserted into the container document. The implementation of this function utilizes MFC exceptionhandling macros to catch any errors that may occur during the creation of the new item. The member function OnCancelEditCntr terminates an in-place editing session. This function is called by the framework when the user presses the Escape key. The OnSetFocus member function is used to ensure that when a view with an active in-place session receives the focus, the focus is actually set to the in-place items frame window. The implementation of the OnSize member function ensures that if the view window changes, the in-place session has a chance to reflect such a change. While there are additional, minor differences between an OLE container and a non-OLE application, this concludes our review of the significant changes. The one thing you have not yet looked at is the container applications resource file.

Container Menus
A look at the AppWizard-generated resource file created for our OCON application (Figure 25.8) reveals two obvious differences: a new accelerator table and a new menu.

OLE Containers CHAPTER 25

467

FIGURE 25.8.
Container resources.

The new menu, IDR_OCONTYPE_CNTR_IP (Figure 25.9), is very obviously incomplete. The reason? This menu, during an in-place editing session, is combined with a similarly incomplete menu provided by the OLE component server application.

FIGURE 25.9.
Container menu for inplace editing session.

This combing of container and server menus, shown in Figure 25.10, ensures that commands that are the responsibility of the container are executed by the container application, and serverrelated commands are executed by the server. Understanding the need for a separate acceleration table during in-place editing is easy in view of the mechanism used to construct the menu for such a session. This concludes our review of the differences between container and noncontainer applications. The next section shows how to modify this skeleton application to implement some useful container functionality.

25
OLE CONTAINERS

468

OLE, COM, and MFC Applications PART IV

FIGURE 25.10.
How server and container menus are combined during inplace editing.

File

Window

From the container application

Edit View

Help

From the server application

Container Application File Edit View Window Help

What the user sees

Customizing the Application


Before we start writing code blindly, we should look at the changes we wish to add to the OLE container skeleton. In addition to the skeleton applications capabilities, our OCON application should 1. 2. 3. 4. 5. Reflect any changes in size and position made during an in-place session to an item Save size and position information in document files Display all container items in a document Implement a simple selection mechanism using the mouse Indicate the current selection visually

There are additional capabilities you can expect from a container application, but which we will not implement at this time. These are s Selection of multiple items s Sizing and positioning of nonactive items using the mouse s Application-specific custom objects

OLE Containers CHAPTER 25

469

Object Positions
When you look at the skeleton implementation of COCONCntrItem::OnChangeItemPosition, the need for a member variable holding the objects position becomes obvious. For this purpose, we can add a member variable of type CRect to the COCONCntrItem class:
class COCONCntrItem : public COleClientItem { ... // Attributes public: ... CRect m_rect;

This member variable obviously needs to be initialized. In the constructor of COCONCntrItem (Listing 25.4), you can set this item to represent a fixed rectangle.

Listing 25.4. Initialization of m_rect in the constructor of COCONCntrItem.


COCONCntrItem::COCONCntrItem(COCONDoc* pContainer) : COleClientItem(pContainer) { // TODO: add one-time construction code here m_rect = CRect(10, 10, 210, 210); }

A more sophisticated application may make an attempt to initialize m_rect to reflect a serversupplied size. To do this, you could set m_rect to a null rectangle to indicate an uninitialized state and make an attempt to update it from the server when the first call is made to COCONCntrItem::OnGetItemPosition. For now, you are going to stick with this simple implementation of a fixed default size. Where is the rectangle m_rect updated? Obviously, you need to change the implementation of In order to reflect the updated position, the function COCONCntrItem::OnGetItemPosition must also be altered.
COCONCntrItem::OnChangeItemPosition.

The change in the member function OnChangeItemPosition requires only two lines of new code (Listing 25.5). First, the rectangle must be updated; second, because the items position has changed, views must be updated to reflect the change.

25
OLE CONTAINERS

Listing 25.5. Modified version of COCONCntrItem::OnChangeItemPosition.


BOOL COCONCntrItem::OnChangeItemPosition(const CRect& rectPos) { ASSERT_VALID(this); if (!COleClientItem::OnChangeItemPosition(rectPos)) return FALSE;

continues

470

OLE, COM, and MFC Applications PART IV

Listing 25.5. continued


// TODO: update any cache you may have of the items // rectangle/extent m_rect = rectPos; GetDocument()->UpdateAllViews(NULL); return TRUE; }

The change to COCONCntrItem::OnGetItemPosition is equally simple. All you have to do is replace the default action that sets the rPosition parameter to a constant rectangle to a line that sets it to the value of m_rect. This modified function is shown in Listing 25.6.

Listing 25.6. Modified version of COCONCntrItem::OnGetItemPosition.


void COCONCntrItem::OnGetItemPosition(CRect& rPosition) { ASSERT_VALID(this); // TODO: return correct rectangle (in pixels) in rPosition // } rPosition.SetRect(10, 10, 210, 210); rPosition = m_rect;

All that is left to do to COCONCntrItem is a change to its Serialize member function. In order for the item positions to be persistent, they must be serialized. This is implemented by adding two lines to the COCONCntrItem::Serialize, as shown in Listing 25.7.

Listing 25.7. Modified version of COCONCntrItem::Serialize.


void COCONCntrItem::Serialize(CArchive& ar) { ASSERT_VALID(this); COleClientItem::Serialize(ar); // now store/retrieve data specific to COCONCntrItem if (ar.IsStoring()) { // TODO: add storing code here ar << m_rect; } else { // TODO: add loading code here ar >> m_rect; } }

OLE Containers CHAPTER 25

471

Now that you have implemented position and size information for objects in the container, it is time to turn your attention to the view class and reflect the new positions when the objects are drawn.

Drawing All Objects


We have identified two shortcomings of default skeleton implementation of the view class COCONView. First, only the item representing the current selection was drawn; second, the item was drawn at a fixed position, not reflecting any changes in size and position that might have occurred during an in-place editing session. The solution to these problems is shown in Listing 25.8. Instead of drawing a single item, this version of the COCONView::OnDraw function iterates through the list of all items in the document. As individual items are drawn, they are placed at the position indicated by their m_rect member variable, which reflects their current size and position.

Listing 25.8. Modified version of COCONView::OnDraw.


void COCONView::OnDraw(CDC* pDC) { COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: remove this code when final draw code is complete. // // // // // // // if (m_pSelection == NULL) { POSITION pos = pDoc->GetStartPosition(); m_pSelection = (COCONCntrItem*)pDoc->GetNextClientItem(pos); } if (m_pSelection != NULL) m_pSelection->Draw(pDC, CRect(10, 10, 210, 210)); POSITION pos = pDoc->GetStartPosition(); while (pos) { COCONCntrItem *pItem = (COCONCntrItem*)pDoc->GetNextClientItem(pos); pItem->Draw(pDC, pItem->m_rect); } }

Note that this code must be changed if your application also has application-specific objects (that is, objects other than those representing embedded items). Instead of using COleDocument::GetNextClientItem, you may wish to use COleDocument::GetNextItem for iteration and use runtime type information to determine the drawing action appropriate for the type of object retrieved.

25
OLE CONTAINERS

472

OLE, COM, and MFC Applications PART IV

Object Selection
To implement a simple object selection mechanism, you must do two things. First, a handler for the mouse event WM_LBUTTONDOWN must be added; second, the current selection must be reflected when the object is drawn in COCONView::OnDraw. To add a handler for the mouse event, use ClassWizard. The handler function should be added to the COCONView class; after all, selection of items is specific to the current view (and indeed, separate views may have different selections). The handler function is shown in Listing 25.9. This function begins by closing any active inplace item. Next, it calls InvalidateRect to invalidate the rectangle of the current selection; the significance of this becomes evident shortly, when you look at the code that implements the drawing of a selection rectangle indicating the selection.

Listing 25.9. Handler function for WM_LBUTTONDOWN messages.


void COCONView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default // CView::OnLButtonDown(nFlags, point); COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); COCONCntrItem *pItem = (COCONCntrItem *)pDoc->GetInPlaceActiveItem(this); if (pItem != NULL) pItem->Close(); if (m_pSelection != NULL) { CRect rect = m_pSelection->m_rect; rect.InflateRect(1, 1); InvalidateRect(rect); m_pSelection = NULL; } POSITION pos = pDoc->GetStartPosition(); while (pos) { pItem = (COCONCntrItem*)pDoc->GetNextClientItem(pos); if (pItem->m_rect.PtInRect(point)) m_pSelection = pItem; } if (m_pSelection != NULL) { CRect rect = m_pSelection->m_rect; rect.InflateRect(1, 1); InvalidateRect(rect); } }

OLE Containers CHAPTER 25

473

In the second half of this function, an iteration is made to identify the item on which the user clicked with the mouse. If such an item is found, it is set to become the current selection. The iteration continues, however, to ensure that eventually the item you pick as the current selection is actually the topmost item. This is important in case the mouse is clicked at a position that is covered by multiple items. After a new selection is found, its rectangle is also invalidated. Although this code implements selecting individual items with the mouse, you must also modify COCONView::OnDraw to provide a visual feedback of the new selection. Listing 25.10 shows this final version of COCONView::OnDraw.

Listing 25.10. Final version of COCONView::OnDraw.


void COCONView::OnDraw(CDC* pDC) { COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: remove this code when final draw code is complete. // // // // // // // if (m_pSelection == NULL) { POSITION pos = pDoc->GetStartPosition(); m_pSelection = (COCONCntrItem*)pDoc->GetNextClientItem(pos); } if (m_pSelection != NULL) m_pSelection->Draw(pDC, CRect(10, 10, 210, 210)); POSITION pos = pDoc->GetStartPosition(); while (pos) { COCONCntrItem *pItem = (COCONCntrItem*)pDoc->GetNextClientItem(pos); pItem->Draw(pDC, pItem->m_rect); if (pItem == m_pSelection) { CRectTracker tracker; tracker.m_rect = pItem->m_rect; tracker.m_nStyle = CRectTracker::resizeInside | CRectTracker::solidLine; if (pItem->GetItemState() == COleClientItem::openState || pItem->GetItemState() == COleClientItem::activeUIState) tracker.m_nStyle |= CRectTracker::hatchInside; tracker.Draw(pDC); } } }

25
OLE CONTAINERS

In this version of the OnDraw member function, you use the CRectTracker class to create a tracking rectangle around the selection. While you do not make use of all its features, this class could also be used to facilitate moving and resizing the object. In this implementation, you simply utilize this class to provide visual feedback.

474

OLE, COM, and MFC Applications PART IV

A notable shortcoming of this implementation is that the tracker will be drawn by the OnDraw member function every time. This includes, unfortunately, the cases when the framework uses OnDraw for printing or print preview. In a more realistic implementation, you would surround the code drawing the tracker with conditionals that determine whether this is a normal drawing situation. If it isnt, the conditionals would prevent the drawing of the tracking rectangle. This concludes your changes to the OCON application. After recompiling and running the application, you can explore its ability to manage multiple embedded objects and save and load documents while retaining object positions (Figure 25.11).

FIGURE 25.11.
Final version of the OCON application.

Other Features
There are several other container application features that can be added easily to the OCON application. In order to edit an embedded object in the current version of the application, you must first select it with the mouse and then invoke the Edit menu, select the objects submenu (for example, Bitmap Image Object), and use the Edit command. To provide an easier way to activate an embedded object, implement a handler for the WM_LBUTTONDBLCLK function. In it, you can utilize the DoVerb member function of the COCONCntrItem class to activate an item for inplace editing. In the handler for WM_LBUTTONDOWN, you can utilize the capabilities of the CRectTracker class to implement moving and sizing.

OLE Containers CHAPTER 25

475

The selection does not need to be restricted to a single item. Instead of a single pointer, you can implement the selection as a list of COCONCntrItem objects using a collection class such as CObList. Multiple selection can be implemented by either monitoring the Shift and Control keys in the handler for WM_LBUTTONDOWN or by adding a rubberbanding capability. While these capabilities are doubtless important in real-life applications, I believe that the OCON application serves the purpose of demonstrating basic container functions.

Summary
OLE containers are applications that handle embedded or linked items. Container applications can be created through the AppWizard. All you need to do is specify container support in AppWizard Step 3. OLE containers must be single-documentbased or multiple-document based applications; container capability for dialog-based applications is not supported. An AppWizard-generated container application skeleton supports the insertion of a single component object at a fixed position. The item is inserted through the Insert Object command in the Edit menu. The application does not reflect changes in the items size or position during an in-place editing session. The container application represents component objects with a class derived from COleClientItem. Other differences between a container and a noncontainer application include new code in the container applications view class that implements object insertion and a simple selection mechanism, and a new, partially complete menu that represents container-provided portions of the menu that is visible during an in-place editing session. Customizing a skeleton application may involve the following steps: 1. To add support for the proper positioning of container objects, modify the OnChangeItemPosition and OnGetItemPosition member functions of the container item class. Add a member variable reflecting the current size and position of the item. Make sure that this member variable is serialized. 2. To draw items at their correct position, modify the OnDraw member function of the view class. 3. To draw items other than the current selection, modify the OnDraw member function of the view class. 4. To provide visual feedback reflecting the current selection, modify the OnDraw member function of the view class. Utilize the CRectTracker class for drawing a rectangle around your selection. 5. To implement a mouse-driven selection mechanism, add a handler function to your view class for WM_LBUTTONDOWN events.

25
OLE CONTAINERS

476

OLE, COM, and MFC Applications PART IV

6. To invoke an object for in-place editing by double-clicking it, add a handler function to your view class for WM_LBUTTONDBLCLK events. 7. To implement the selection of multiple items, replace the m_pSelection member variable in your view class. Modify the WM_LBUTTONDOWN handler to implement a multiple selection mechanism. 8. In your WM_LBUTTONDOWN handler, you can utilize the capabilities of the CRectTracker class to implement sizing and positioning. 9. To add application-specific items to the document, consider deriving the class that represents these items from CDocItem. Utilize the capabilities of COleDocument for handling and serializing CDocItem objects. Revise the OnDraw member function in your view class to draw objects that are not container items. Revise your mouse event handlers as appropriate.

OLE Drag and Drop CHAPTER 26

477

OLE Drag and Drop

26
OLE DRAG AND DROP

26

IN THIS CHAPTER
s Drag-and-Drop Basics 478 s Creating a Container Application 478 s Adding Drag-and-Drop Support 482

478

OLE, COM, and MFC Applications PART IV

OLE provides an elegant, simple, standardized way to implement drag-and-drop capability in applications. To explore drag-and-drop support in this chapter, we build an OLE container application. This simple application is able to hold a series of container objects. Via the OLE drag-and-drop mechanism, this application supports dragging items between its view and another application, between its own view windows, and within a single view window.

Drag-and-Drop Basics
Drag and drop represents a technique of sharing data between applications that act as drag sources and applications that act as drop targets. Drag sources are applications that enable their items to be dragged from their windows to the windows of other applications. Drop targets are applications that accept items dragged from drag sources and released within the applications window (Figure 26.1).

FIGURE 26.1.
Drag and drop.

Drag Source Application Drag Source Document

Drop Target Application Drop Target Document

Implementing a drag source using MFC is very simple. Implementing a drop target is somewhat more difficult, but it is still not an overwhelmingly complex task. The simple implementations presented in this chapter are based on the drag-and-drop support in the class COleClientItem. In applications that support drag and drop of native data or drag and drop of multiple items, you may decide to utilize the drag-and-drop support in COleServerItem instead. Drag-and-drop functionality and clipboard functionality have many things in common. If you wish to implement clipboard cut, copy, and paste functions using OLE, you can share most of the clipboard support code and drag-and-drop code.

Creating a Container Application


The container application required for our purposes is based on the AppWizard-generated container application default, with a few modifications that make it possible to select items using the mouse and that support persistent storage of item positions.

OLE Drag and Drop CHAPTER 26

479

Creating the Application


To create the application, use AppWizards defaults, except for specifying container application support in AppWizard Step 3. I named the application OCON for the simple reason that it is based on the example presented in the previous chapter. Let me now briefly review the changes that need to be made to the AppWizard-generated skeleton to support persistent storage of embedded object positions and to support item selection using the mouse. If you read the previous chapter, these changes will already be familiar to you.

26
OLE DRAG AND DROP

Adding Positioning Support


The default AppWizard-supplied implementation of a container application positions inserted objects using a fixed rectangle. To make it possible for inserted items to be positioned anywhere in the drawing surface, a member variable of type CRect must be added to the class COCONCntrItem. This variable must also be serialized. The Draw member function of the view class must utilize this variable when drawing objects. This variable should also be utilized in the C O C O N C n t r I t e m member functions and OnGetItemPosition. Another change to COCONCntrItem is the addition of a new helper function, Invalidate; this function, taking a CWnd pointer as its parameter, invalidates the items rectangle in the specified window.
OnChangeItemPosition

To add the new member variable to the COCONCntrItem class and to declare the new helper function Invalidate, use the following code:
class COCONCntrItem : public COleClientItem { ... // Attributes public: ... CRect m_rect; ... // Operations public: void Invalidate(CWnd *pWnd); // Implementation ...

The new member variable must be initialized in the COCONCntrItem constructor (Listing 26.1).

Listing 26.1. Initialization of m_rect in the constructor of COCONCntrItem.


COCONCntrItem::COCONCntrItem(COCONDoc* pContainer) : COleClientItem(pContainer) { // TODO: add one-time construction code here m_rect = CRect(0, 0, 0, 0); }

480

OLE, COM, and MFC Applications PART IV

The modified versions of COCONCntrItem::OnChangeItemPosition and COCONCntrItem:: OnGetItemPosition (Listing 26.2) are used to reflect any changes in the items size or position that were made during an in-place session. This listing also makes use of the GetCachedExtent member function to update the items size and position if it has not yet been initialized.

Listing 26.2. COCONCntrItem::OnChangeItemPosition and COCONCntrItem::OnGetItemPosition.


BOOL COCONCntrItem::OnChangeItemPosition(const CRect& rectPos) { ASSERT_VALID(this); if (!COleClientItem::OnChangeItemPosition(rectPos)) return FALSE; // TODO: update any cache you may have of the items // rectangle/extent m_rect = rectPos; GetDocument()->UpdateAllViews(NULL); return TRUE; } void COCONCntrItem::OnGetItemPosition(CRect& rPosition) { ASSERT_VALID(this); // rPosition.SetRect(10, 10, 210, 210); if (m_rect.IsRectNull()) { CSize size; CClientDC dc(NULL); GetCachedExtent(&size, GetDrawAspect()); dc.HIMETRICtoDP(&size); m_rect = CRect(CPoint(10, 10), size); } rPosition = m_rect;

Listing 26.3 shows how COCONCntrItem::Serialize must be modified to support serialization of the new member variable.

Listing 26.3. Modified version of COCONCntrItem::Serialize.


void COCONCntrItem::Serialize(CArchive& ar) { ASSERT_VALID(this); COleClientItem::Serialize(ar); if (ar.IsStoring()) { // TODO: add storing code here

OLE Drag and Drop CHAPTER 26


ar << m_rect; } else { // TODO: add loading code here ar >> m_rect; } }

481

26
OLE DRAG AND DROP

Finally, the addition of the new Invalidate helper function (Listing 26.4) concludes changes to the implementation of COCONCntrItem.

Listing 26.4. The helper function COCONCntrItem::Invalidate.


void COCONCntrItem::Invalidate(CWnd *pWnd) { CRect rect = m_rect; rect.InflateRect(1, 1); pWnd->InvalidateRect(rect); }

Listing 26.5 shows the changes to COCONView::OnDraw; the new version takes into account the items stored positions and draws them accordingly. It also draws all items (as opposed to just the current selection). Furthermore, this version of the function uses a CRectTracker object to highlight the currently selected item.

Listing 26.5. Modified version of COCONCntrItem::OnDraw.


void COCONView::OnDraw(CDC* pDC) { COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); POSITION pos = pDoc->GetStartPosition(); while (pos) { COCONCntrItem *pItem = (COCONCntrItem*)pDoc->GetNextClientItem(pos); pItem->Draw(pDC, pItem->m_rect); if (pItem == m_pSelection) { CRectTracker tracker; tracker.m_rect = pItem->m_rect; tracker.m_nStyle = CRectTracker::resizeInside | CRectTracker::solidLine; if (pItem->GetItemState() == COleClientItem::openState || pItem->GetItemState() == COleClientItem::activeUIState) tracker.m_nStyle |= CRectTracker::hatchInside; tracker.Draw(pDC); } } }

482

OLE, COM, and MFC Applications PART IV

Adding Selection Support


In the previous section, we already implemented highlighting the current selection. Adding support for selection of items using the mouse requires adding a handler for WM_LBUTTONDOWN events to the view class. This handler, shown in Listing 26.6, can be added using the ClassWizard.

Listing 26.6. Handler function for WM_LBUTTONDOWN messages.


void COCONView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default // CView::OnLButtonDown(nFlags, point); COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); COCONCntrItem *pSelection = NULL; COCONCntrItem *pItem = (COCONCntrItem *)pDoc->GetInPlaceActiveItem(this); if (pItem != NULL) pItem->Close(); POSITION pos = pDoc->GetStartPosition(); while (pos) { pItem = (COCONCntrItem*)pDoc->GetNextClientItem(pos); if (pItem->m_rect.PtInRect(point)) pSelection = pItem; } if (pSelection != m_pSelection) { if (m_pSelection != NULL) m_pSelection->Invalidate(this); m_pSelection = pSelection; if (m_pSelection != NULL) m_pSelection->Invalidate(this); } }

Adding Drag-and-Drop Support


Although the two areas of functionality are usually mentioned together, the requirements for a drag source application and for a drop target application are quite distinct. Of the two, implementation of a drag source is the easier task. However, both tasks are supported extensively in part by the OLE architecture, in part by the MFC. The support for implementing a drag source comes in the form of the DoDragDrop member function of several classes, namely COleClientItem, COleServerItem, and COleDataSource. Drop target functionality is supported through a series of member functions of the CView class; namely, its member functions OnDrop, OnDragEnter, OnDragOver, and OnDragLeave. However,

OLE Drag and Drop CHAPTER 26

483

unlike the DoDragDrop function, which can be called as is, these functions require customized implementations in your application.

26
OLE DRAG AND DROP

Implementing a Drag Source


With the help of the COleClientItem::DoDragDrop member function, adding drag source capability to an OLE container is almost embarrassingly simple. All we need to do is modify the handler function for WM_LBUTTONDOWN events, COCONView::OnLButtonDown. If and when a valid object is selected by the user using the mouse, we must call the objects DoDragDrop member function to perform the drag-and-drop operation. We must also perform a minor housekeeping chore: If the object was moved from our application to another, we must delete it from the list of objects maintained by our document class. However, with CDocItem-derived objects, this is also a very simple task; it is sufficient to simply delete the object using the delete operator. The CDocItem destructor function will ensure that the object is properly removed from the documents list of objects. The modified version of COCONView::OnLButtonDown is shown in Listing 26.7.

Listing 26.7. COCONView::OnLButtonDown with drag source support.


void COCONView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default // CView::OnLButtonDown(nFlags, point); COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); COCONCntrItem *pSelection = NULL; COCONCntrItem *pItem = (COCONCntrItem *)pDoc->GetInPlaceActiveItem(this); if (pItem != NULL) pItem->Close(); POSITION pos = pDoc->GetStartPosition(); while (pos) { pItem = (COCONCntrItem*)pDoc->GetNextClientItem(pos); if (pItem->m_rect.PtInRect(point)) pSelection = pItem; } if (pSelection != m_pSelection) { if (m_pSelection != NULL) m_pSelection->Invalidate(this); m_pSelection = pSelection; if (m_pSelection != NULL) m_pSelection->Invalidate(this); } if (m_pSelection != NULL) { m_dragRect = m_pSelection->m_rect;

continues

484

OLE, COM, and MFC Applications PART IV

Listing 26.7. continued


if (m_pSelection->DoDragDrop(m_pSelection->m_rect, (CPoint)(point - m_pSelection->m_rect.TopLeft())) == DROPEFFECT_MOVE) { m_pSelection->Invalidate(this); delete m_pSelection; m_pSelection = NULL; } } }

Before moving on to the implementation of drop target functionality, I want to add a few notes concerning this drag source implementation. Obviously, most applications have functionality above and beyond being an OLE container (if they implement container functionality at all). How would you implement drag source capabilities for those applications? One possibility is to utilize the DoDragDrop member function of COleServerItem. If your application is an OLE server, it already has a COleServerItem-derived class defined to represent your applications document in an embedded item. Modify this class to support representation of only the current selection (as opposed to the entire document). This modification is easy; you need only to change the Serialize and OnDraw member functions and create a constructor that takes a parameter representing the current selection. The utility of this class goes beyond drag source functionality; it can also be used to represent linked items and to facilitate the transfer of the current selection to the clipboard. Once your COleServerItem-derived class is complete, you can create an item of this type in your WM_LBUTTONDOWN handler (or wherever you wish to implement drag source functionality). Subsequently, you can utilize the member function COleServerItem::DoDragDrop to implement drag source capability just as simply as we did for OLE client items. If you do not wish to use a COleServerItem-derived class for this purpose (for example, if your application is not an OLE server), you can also utilize the COleDataSource class. This class can be used to represent a selection for drag and drop and clipboard transfers. COleDataSource also has a DoDragDrop member function, so implementing drag source functionality using this class is equally simple.

Implementing a Drop Target


Implementing an OLE drop target requires a lot more work than implementing a drag source. Several member functions of your view class require override versions. The view class must be registered as a drop target. Special considerations must be made to ensure that objects originating from within the application itself are handled properly and efficiently; for example, if the drag-and-drop operation effectively reduces to a move within the same window, it should be treated that way.

OLE Drag and Drop CHAPTER 26

485

The set of CView member functions that require overrides is listed in Table 26.1.

26
Table 26.1. Drag-and-drop related overridables in CView. Member function Description
OnDragEnter OnDragLeave OnDragOver OnDrop

OLE DRAG AND DROP

Called when an item is dragged into the window Called when a dragged item leaves the window Called while an item is dragged within the window Called when an item is released in the window

Before we start writing code madly, heres a summary of exactly what we would like to see in our drop target application. First, the obvious: If an object is released over our applications view window, we would like the object to appear at that location, preferably preserving its original size. We would also like to see a tracking rectangle while the mouse is inside the view window. The rectangle will reflect the size of the object that is about to be dropped in the window. Lastly, we would like to ensure that a drag-and-drop operation that reduces to merely moving an object within the same window is treated accordingly. The implementation of the OnDragEnter, OnDragLeave, and OnDragOver function overrides requires a few additional member variables. These variables will be used to remember the drag rectangles size and position and other drag characteristics during a drag operation. In addition, a member variable of type COleDropTarget is also required in order to register the view window as a drop target. The declaration of these variables should be added to the declaration of the view class as follows:
class COCONView : public CView { ... // Attributes public: ... COCONCntrItem* m_pSelection; COleDropTarget m_dropTarget; BOOL m_bInDrag; DROPEFFECT m_prevDropEffect; CRect m_dragRect; CPoint m_dragPoint; CSize m_dragSize; CSize m_dragOffset; ...

The declarations for the overrides of OnDragEnter, OnDragLeave, OnDragOver, and OnDrop should be added using the ClassWizard.

486

OLE, COM, and MFC Applications PART IV

Our first task in making the application work as a drop target is to register it as one. This is accomplished by adding a member variable of type COleDropTarget and calling its Register member function at the appropriate moment of time. The most appropriate place for this is in the view classs OnCreate member function. To implement this registration, create a handler for WM_CREATE messages using the ClassWizard and add the code shown in Listing 26.8.

Listing 26.8. The COCONView::OnCreate member function.


int COCONView::OnCreate(LPCREATESTRUCT lpCreateStruct) { if (CView::OnCreate(lpCreateStruct) == -1) return -1; // TODO: Add your specialized creation code here m_dropTarget.Register(this); return 0; }

The OnDragEnter, OnDragLeave, and OnDragOver member functions are used to manage visual feedback during a drag operation. OnDragEnter (Listing 26.9) attempts to retrieve the items size by querying the item for the Object Descriptor clipboard type. This data type, when supplied, contains information about the transfer item, including its size and the offset of the mouse pointer relative to the items upper-left corner.

Listing 26.9. The COCONView::OnDragEnter member function.


DROPEFFECT COCONView::OnDragEnter(COleDataObject* pDataObject, DWORD dwKeyState, CPoint point) { // TODO: Add your specialized code here and/or call the base class // return CView::OnDragEnter(pDataObject, dwKeyState, point); ASSERT(m_prevDropEffect == DROPEFFECT_NONE); m_dragSize = CSize(0, 0); m_dragOffset = CSize(0, 0); HGLOBAL hObjDesc = pDataObject->GetGlobalData(cfObjectDescriptor); if (hObjDesc != NULL) { LPOBJECTDESCRIPTOR pObjDesc = (LPOBJECTDESCRIPTOR)GlobalLock(hObjDesc); ASSERT(pObjDesc != NULL); m_dragSize.cx = (int)pObjDesc->sizel.cx; m_dragSize.cy = (int)pObjDesc->sizel.cy; m_dragOffset.cx = (int)pObjDesc->pointl.x; m_dragOffset.cy = (int)pObjDesc->pointl.y; GlobalUnlock(hObjDesc); GlobalFree(hObjDesc); } CClientDC dc(NULL);

OLE Drag and Drop CHAPTER 26


dc.HIMETRICtoDP(&m_dragSize); dc.HIMETRICtoDP(&m_dragOffset); m_dragPoint = point - CSize(1, 1); return OnDragOver(pDataObject, dwKeyState, point); }

487

26
OLE DRAG AND DROP

This function makes use of the global variable cfObjectDescriptor. Declare this variable at the top of your view classs implementation file as follows:
static cfObjectDescriptor = (CLIPFORMAT)::RegisterClipboardFormat(_T(Object Descriptor));

The next member function is OnDragOver (Listing 26.10). This function is called every time the mouse moves while within the view windows client area. This function plays a dual role. First, it determines the currently applicable drop effect. Then, based on the state of the Control, Shift, and Alt keys, it determines whether the item (if it was dropped in the window at this moment) should be copied, linked, or moved to this window.

Listing 26.10. The COCONView::OnDragOver member function.


DROPEFFECT COCONView::OnDragOver(COleDataObject* pDataObject, DWORD dwKeyState, CPoint point) { // TODO: Add your specialized code here and/or call the base class // return CView::OnDragOver(pDataObject, dwKeyState, point); DROPEFFECT de = DROPEFFECT_NONE; point -= m_dragOffset; if ((dwKeyState & (MK_CONTROL|MK_SHIFT)) == (MK_CONTROL|MK_SHIFT)) de = DROPEFFECT_LINK; else if ((dwKeyState & MK_CONTROL) == MK_CONTROL) de = DROPEFFECT_COPY; else if ((dwKeyState & MK_ALT) == MK_ALT) de = DROPEFFECT_MOVE; else de = DROPEFFECT_MOVE; if (point == m_dragPoint) return de; CClientDC dc(this); if (m_prevDropEffect != DROPEFFECT_NONE) dc.DrawFocusRect(CRect(m_dragPoint, m_dragSize)); m_prevDropEffect = de; if (m_prevDropEffect != DROPEFFECT_NONE) { m_dragPoint = point; dc.DrawFocusRect(CRect(point, m_dragSize)); } return de; }

488

OLE, COM, and MFC Applications PART IV

The other role of this function is to actually draw visual feedback. This is accomplished by drawing a rectangle using the CDC::DrawFocusRect function. The third function in this group is OnDragLeave (Listing 26.11). This function, the simplest of the three, is called to mark the end of a dragging operation.

Listing 26.11. The COCONView::OnDragLeave member function.


void COCONView::OnDragLeave() { // TODO: Add your specialized code here and/or call the base class // CView::OnDragLeave(); CClientDC dc(this); if (m_prevDropEffect != DROPEFFECT_NONE) { dc.DrawFocusRect(CRect(m_dragPoint, m_dragSize)); m_prevDropEffect = DROPEFFECT_NONE; }

Now its time to turn our attention to the OnDrop member function (Listing 26.12). This function is by far the most important one in our implementation of drop target functionality. As its name implies, this function is the one in which the actual insertion of a dropped item takes place.

Listing 26.12. The COCONView::OnDrop member function.


BOOL COCONView::OnDrop(COleDataObject* pDataObject, DROPEFFECT dropEffect, CPoint point) { // TODO: Add your specialized code here and/or call the base class // return CView::OnDrop(pDataObject, dropEffect, point); ASSERT_VALID(this); COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); CSize size; OnDragLeave(); CClientDC dc(NULL); point -= m_dragOffset; pDoc->SetModifiedFlag(TRUE); if ((dropEffect & DROPEFFECT_MOVE) && m_bInDrag) { ASSERT(m_pSelection != NULL); m_pSelection->Invalidate(this); m_pSelection->m_rect = m_dragRect + point - m_dragRect.TopLeft(); m_bInDrag = FALSE; return TRUE; }

OLE Drag and Drop CHAPTER 26


COCONCntrItem* pItem = NULL; TRY { pItem = new COCONCntrItem(pDoc); ASSERT_VALID(pItem); if (dropEffect & DROPEFFECT_LINK) { if (!pItem->CreateLinkFromData(pDataObject)) AfxThrowMemoryException(); } else { if (!pItem->CreateFromData(pDataObject)) AfxThrowMemoryException(); } ASSERT_VALID(pItem); pItem->GetExtent(&size, pItem->GetDrawAspect()); dc.HIMETRICtoDP(&size); pItem->m_rect = CRect(point, size); if (m_pSelection != NULL) m_pSelection->Invalidate(this); m_pSelection = pItem; if (m_pSelection != NULL) m_pSelection->Invalidate(this); } CATCH(CException, e) { if( pItem != NULL ) delete pItem; AfxMessageBox(IDP_FAILED_TO_CREATE); } END_CATCH return pItem != NULL; }

489

26
OLE DRAG AND DROP

This function first terminates the drag operation by calling OnDragLeave, and then it notifies the document that the contents are changing by calling CDocument::SetModifiedFlag. Next it makes an attempt to determine if the drag-and-drop operation actually represents moving an object within the same window. For this, we make use of the m_bInDrag member variable; this variable is set in COCONView::OnLButtonDown when a drag operation begins, as we see momentarily. If the operation is a move, the function simply updates the affected items rectangle and returns. In the case of a genuine drop operation, an attempt is made to create a new item of type COCONCntrItem using the drop data. If the attempt is successful, the items size is determined and the items rectangle is updated. As I mentioned, the key to determining whether a drag-and-drop operation reduces to a mere move is the m_bInDrag member variable. To set this variable, we have to implement yet another modification to COCONView::OnLButtonDown. This final version of this function is shown in Listing 26.13.

490

OLE, COM, and MFC Applications PART IV

Listing 26.13. Final version of COCONView::OnLButtonDown.


void COCONView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default // CView::OnLButtonDown(nFlags, point); COCONDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); COCONCntrItem *pSelection = NULL; COCONCntrItem *pItem = (COCONCntrItem *)pDoc->GetInPlaceActiveItem(this); if (pItem != NULL) pItem->Close(); POSITION pos = pDoc->GetStartPosition(); while (pos) { pItem = (COCONCntrItem*)pDoc->GetNextClientItem(pos); if (pItem->m_rect.PtInRect(point)) pSelection = pItem; } if (pSelection != m_pSelection) { if (m_pSelection != NULL) m_pSelection->Invalidate(this); m_pSelection = pSelection; if (m_pSelection != NULL) m_pSelection->Invalidate(this); } if (m_pSelection != NULL) { m_bInDrag = TRUE; m_dragRect = m_pSelection->m_rect; DROPEFFECT dropEffect = m_pSelection->DoDragDrop(m_pSelection->m_rect, (CPoint)(point - m_pSelection->m_rect.TopLeft())); m_pSelection->Invalidate(this); if (m_bInDrag == TRUE && dropEffect == DROPEFFECT_MOVE) { delete m_pSelection; m_pSelection = NULL; } m_bInDrag = FALSE; } }

member variable is set to T R U E just before calling If during the drag a callback is made to the same view object, by looking at m_bInDrag we can determine that the source of the drag operation is the same view window.
m_bInDrag COCONCntrItem::DoDragDrop.

As you can see, the

The OnDrop member function resets m_bInDrag to FALSE if a drag operation is successfully reduced to a move. In this case, OnLButtonDown will not delete the selected item. That deletion is necessary, however, if the item was moved to a different window.

OLE Drag and Drop CHAPTER 26

491

Finally, we need to initialize two of the member variables to ensure proper functioning of the view class. Listing 26.14 shows the initialization of m_bInDrag and m_prevDropEffect in the view classs constructor.

26
OLE DRAG AND DROP

Listing 26.14. Member variable initialization in the constructor of COCONView.


COCONView::COCONView() { m_pSelection = NULL; // TODO: add construction code here m_bInDrag = FALSE; m_prevDropEffect = DROPEFFECT_NONE; }

This concludes our construction of an OLE container supporting drag and drop. The completed application, shown in Figure 26.2, can serve as a drop target (or drag source) for word processor objects, spreadsheet cells, drawings, and many other types of OLE objects.

FIGURE 26.2.
The OCON drag-and-drop application.

This application still has a cosmetic problem: if a document is represented by multiple views, not all views are updated after a drag-and-drop operation. This can be remedied by using the CDocument: Update AllViews function. Clever use of this functions lHint parameter can help avoid excessive redrawing. I leave the implmentation of this solution as an exercise for you.

492

OLE, COM, and MFC Applications PART IV

Summary
Drag and drop represents the capability to select items in one application (the drag source) and, using the mouse, drag the items and drop them in the window of another application (the drop target). Drag and drop, from the users perspective, is a simpler mechanism for sharing data between applications than using the clipboard. OLE provides extensive drag-and-drop support. This support is encapsulated in the MFC Library in the form of a series of drag-and-drop related classes and functions. The implementation of a drag source is relatively easy. For this purpose, you can utilize the member functions COleClientItem, COleServerItem, and COleDataSource. Use of COleClientItem::DoDragDrop is recommended if the drag item is an embedded or linked OLE item represented by a COleClientItem-derived class. In this case, simply call the DoDragDrop function for the object that the user selected for dragging, and the framework does the rest. Remember to delete the object if the return value of DoDragDrop indicates that the selection has been moved (as opposed to copied or linked) to another application. Use of COleServerItem::DoDragDrop is recommended for applications that are OLE servers. This function is most useful if your COleServerItem-derived class is already capable of representing a selection (as opposed to the entire contents of a document). Just create a COleServerItem-derived object representing the drag selection and use its DoDragDrop member function to perform the drag operation. In applications that are not OLE servers, you can also consider using COleDataSource for implementing a drag source. Implementing a drop target is a more involved operation. In addition to providing an override version of the OnDrop member function of your view class, you must also override the OnDragEnter, OnDragLeave, and OnDragOver member functions. The purpose of these functions, called by the framework while the mouse is over the view window during a drag-and-drop operation, is twofold: first, they are used to provide visual feedback during the drag; and second, they are used to inform the framework regarding the allowable drop operations. In the simplest implementation of OnDrop, you can create a COleClientItem-derived object representing the drop item. In more involved implementations, you may consider inspecting the item and identifying native data originating from within your own application, or data available in formats that your application can recognize. In an application that acts both as a drag source and drop target, you should improve the applications efficiency by recognizing operations that reduce to a simple move. In this case, instead of removing the dragged item or items and creating new items, you can implement the operation as a simple position change.

Automation CHAPTER 27

493

Automation
IN THIS CHAPTER

27

s Building an Automation Server

494

s Standard Methods and Properties 506

27
AUTOMATION

494

OLE, COM, and MFC Applications PART IV

Automation (formerly known as OLE Automation) represents a communication mechanism between cooperating applications. Applications that are automation servers have the capability to expose their objects through a series of properties and methods. Applications that are automation clients or automation controllers can access these exposed properties and methods through the OLE IDispatch interface. One of the most popular uses of automation is to expose the capabilities of your application to the extent that your application becomes fully controllable from a general purpose automation controller, such as Microsofts Visual Basic. In this fashion, Visual Basic can act as a powerful macro language for your application. The MFC Library and the Visual C++ Developer Studio provide extensive support for developing automation servers. In this chapter, we first explore these capabilities by constructing a simple server application; later, we review other issues involved in automation development.

Building an Automation Server


The automation server that we build in this chapter is simple indeed; this tiny program performs one task only, and that is the multiplication of two numbers. The program is capable of running standalone using a simple dialog-based user interface. It also supports invocation from within an automation controller. Figure 27.1 shows the user interface of this application, ASRV. To use this application, first enter the two multiplicands in the two edit fields on the left side, and then click the = button to calculate the result.

FIGURE 27.1.
The user interface of the ASRV server.

The applications automation methods and properties closely correspond to these userinterface elements. Two properties, Multi1 and Multi2, correspond to the two multiplicands; these properties can be read and written. The Result property, representing the result of the multiplication, is a read-only property. Finally, the Set method corresponds to the function of the = button; this method carries out the actual multiplication.

Automation CHAPTER 27

495

This application is constructed in three steps. First, the applications skeleton is created. Next, the user interface is edited and the functions performing the calculations are added. Finally, automation capabilities are implemented.

Constructing the ASRV Application Skeleton


The ASRV application skeleton should be constructed using the AppWizard. The application should be a single-documentbased application with default settings. Support for automation is specified in Step 3 of AppWizard (Figure 27.2). Make sure the Automation check box is marked before proceeding from this step.

27
AUTOMATION

FIGURE 27.2.
Adding automation support.

We should also change the base class of the new applications view class before dismissing AppWizard. In AppWizard Step 6 (Figure 27.3), select the class CASRVView and select CFormView as its base class. As a CFormView-based class, our new view class uses a dialog template to provide its visual appearance. At this point, you can click on the Finish button to complete construction of the skeleton application.

Implementing the Calculation


I almost used the term Calculator Engine in the title, but I figured that such a term might cause some of you to burst out in hysterical laughter. After all, calling a simple multiplication operation an engine may be too much, even in this day and age of hype, pomp, and circumstance.

496

OLE, COM, and MFC Applications PART IV

FIGURE 27.3.
Changing the base class for the view.

The first step in implementing the calculator is to create its user interface. AppWizard created a dialog corresponding to the CFormView-derived view class; however, the dialog is Spartan, to say the least (Figure 27.4). Edit this dialog by removing the default TODO field and adding three edit fields, a button, and a static field as shown in Figure 27.1.

FIGURE 27.4.
Spartan default dialog supplied by AppWizard.

The two edit fields on the left side should be called IDC_MULTI1 and IDC_MULTI2, respectively. The edit field on the right-hand side should be called IDC_RESULT. You may consider making this field a read-only field. The button, with a caption set to a single equal sign, should be called IDC_CALCULATE. You may consider setting the Default property for this button to enable using the Enter key to perform the calculation. While the dialog is open for editing, invoke the ClassWizard. Using the ClassWizard, add a member function to the view class corresponding to a mouse click on the IDC_CALCULATE button (Figure 27.5).

Automation CHAPTER 27

497

FIGURE 27.5.
Adding a member function to handle clicks on the Calculate button.

27
AUTOMATION

At this point, you are probably expecting us to use the ClassWizard to add member variables corresponding to the edit fields. However, this is not how we are going to proceed at this time. The member variables that represent the multiplicands and the result are added to the document class instead; correspondingly, we have to modify the DoDataExchange member function of the view class manually. After dismissing the ClassWizard, open the header file for the document class for editing. In the Attributes section of the CASRVDoc class declaration, add declarations for three new member variables as follows:
class CASRVDoc : public CDocument { protected: // create from serialization only CASRVDoc(); DECLARE_DYNCREATE(CASRVDoc) // Attributes public: long m_lMulti1; long m_lMulti2; long m_lResult; ...

These variables must also be initialized. Initialization can take place in the constructor (Listing 27.1).

Listing 27.1. Variable initialization in the CASRVDoc constructor.


CASRVDoc::CASRVDoc() { // TODO: add one-time construction code here

continues

498

OLE, COM, and MFC Applications PART IV

Listing 27.1. continued


m_lMulti1 = 0; m_lMulti2 = 0; m_lResult = 0; EnableAutomation(); AfxOleLockApp(); }

To actually perform the calculation, we need a new member function in CASRVDoc. This member function, DoCalculate, will be declared as follows:
class CASRVDoc : public CDocument { ... // Operations public: void DoCalculate(); ...

The implementation of DoCalculate is shown in Listing 27.2.

Listing 27.2. Implementation of CASRVDoc::DoCalculate.


void CASRVDoc::DoCalculate() { m_lResult = m_lMulti1 * m_lMulti2; }

Two things remain to be done. First, we need to connect the member variables in CASRVDoc with the CFormView-based dialog. Second, we must add a call to CASRVDoc::DoCalculate from within the handler function for mouse clicks on the IDC_CALCULATE button. The modified version of CASRVView::DoDataExchange is shown in Listing 27.3. Basically, we do what ClassWizard would have done, if only the variables were local to this class.

Listing 27.3. Modified version of CASRVDoc::DoDataExchange.


void CASRVView::DoDataExchange(CDataExchange* pDX) { CFormView::DoDataExchange(pDX); //{{AFX_DATA_MAP(CASRVView) //}}AFX_DATA_MAP CASRVDoc *pDoc = GetDocument(); ASSERT_VALID(pDoc); DDX_Text(pDX, IDC_MULTI1, pDoc->m_lMulti1); DDX_Text(pDX, IDC_MULTI2, pDoc->m_lMulti2); DDX_Text(pDX, IDC_RESULT, pDoc->m_lResult); }

Automation CHAPTER 27

499

In the implementation of CASRVView::OnCalculate (Listing 27.4), we call the document classs member function DoCalculate to perform the actual calculation. We also utilize the UpdateData function to ensure that the member variables and the dialog fields are properly updated before and after the calculation.

Listing 27.4. Implementation of CASRVView::OnCalculate.


void CASRVView::OnCalculate() { // TODO: Add your control notification handler code here CASRVDoc *pDoc = GetDocument(); ASSERT_VALID(pDoc); UpdateData(TRUE); pDoc->DoCalculate(); UpdateData(FALSE); }

27
AUTOMATION

At this point, the application can be recompiled and should operate normally as a stand-alone application.

Adding Automation Support


The first step in adding automation support is through the ClassWizard. To add properties and methods to the CASRVDoc class, invoke the ClassWizard, select the Automation tab, and select the CASRVDoc class (Figure 27.6).

FIGURE 27.6.
Using the ClassWizard to add automation support.

500

OLE, COM, and MFC Applications PART IV

First, add two properties corresponding to the two multiplicands. Click on the Add Property button to invoke the Add Property dialog. In this dialog, specify Multi1 as the external name of the new property and long as the propertys type. Modify the default variable name supplied by ClassWizard to read m_lMulti1 (Figure 27.7). Click on the OK button to dismiss this dialog.

FIGURE 27.7.
Adding a new property.

Next, use the ClassWizard to add another property, Multi2, in a similar fashion. The third property, Result, differs from the previous two. Because this is a read-only property, you choose get/set methods as the propertys implementation. Or, to be more precise, you use a get method; implementing a read-only property is accomplished simply by erasing the ClassWizard-generated name for the set function (Figure 27.8).

FIGURE 27.8.
Adding a read-only property.

Automation CHAPTER 27

501

The one thing that remains is adding the Calculate method that performs the actual calculation. To add this method, click on the Add Method button. In the Add Method dialog, type Calculate as both the External and Internal name of the new method and select void as the methods return type (Figure 27.9).

FIGURE 27.9.
Adding a new method.

27
AUTOMATION

At this point, you are finished with ClassWizard; it can be dismissed by clicking on the OK button. Take a brief look at the header file of CASRVDoc. Towards the end of the class declaration, the ClassWizard added a series of items to the classs dispatch map:
class CASRVDoc : public CDocument { ... protected: ... // Generated OLE dispatch map functions //{{AFX_DISPATCH(CASRVDoc) long m_lMulti1; afx_msg void OnMulti1Changed(); long m_lMulti2; afx_msg void OnMulti2Changed(); afx_msg long GetResult(); afx_msg void Calculate(); //}}AFX_DISPATCH DECLARE_DISPATCH_MAP() DECLARE_INTERFACE_MAP() };

You should notice several things here. First, as you can see, the ClassWizard generated declarations for the member variables m_lMulti1 and m_lMulti2; therefore, the declarations you added by hand earlier are no longer needed and should be removed from the Attributes section.

502

OLE, COM, and MFC Applications PART IV

Second, the ClassWizard created the method function Calculate; as there is no need for this and a separate DoCalculate, these two functions can be consolidated into one, and you can get rid of DoCalculate. Third, notice that the variables m_lMulti1 and m_lMulti2 are in a section of the class declaration that is marked protected. However, you access these variables from CASRVView::DoDataExchange. The simplest way to ensure that the program can be compiled is to add CASRVView as a friend class to CASRVDoc. Because of these numerous changes, I have listed the final form of the CASRVDoc class declaration in its entirety. (See Listing 27.5.)

Listing 27.5. CASRVDoc class declaration with automation support.


class CASRVDoc : public CDocument { friend class CASRVView; protected: // create from serialization only CASRVDoc(); DECLARE_DYNCREATE(CASRVDoc) // Attributes public: long m_lResult; // Operations public: // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CASRVDoc) public: virtual BOOL OnNewDocument(); virtual void Serialize(CArchive& ar); //}}AFX_VIRTUAL // Implementation public: virtual ~CASRVDoc(); #ifdef _DEBUG virtual void AssertValid() const; virtual void Dump(CDumpContext& dc) const; #endif protected: // Generated message map functions protected: //{{AFX_MSG(CASRVDoc) // NOTE - the ClassWizard will add and remove member // functions here. DO NOT EDIT what you see in these // blocks of generated code ! //}}AFX_MSG

Automation CHAPTER 27
DECLARE_MESSAGE_MAP() // Generated OLE dispatch map functions //{{AFX_DISPATCH(CASRVDoc) long m_lMulti1; afx_msg void OnMulti1Changed(); long m_lMulti2; afx_msg void OnMulti2Changed(); afx_msg long GetResult(); afx_msg void Calculate(); //}}AFX_DISPATCH DECLARE_DISPATCH_MAP() DECLARE_INTERFACE_MAP() };

503

27
AUTOMATION

The ClassWizard added two notification handlers and two method functions to the implementation of CASRVDoc. After removing the DoCalculate member function from the implementation file, modify CASRVDoc::GetResult and CASRVDoc::Calculate as shown in Listing 27.6. The notification handler functions do not need to be altered.

Listing 27.6. CASRVDoc::GetResult and CASRVDoc::Calculate.


long CASRVDoc::GetResult() { // TODO: Add your property handler here return m_lResult; } void CASRVDoc::Calculate() { // TODO: Add your dispatch handler code here m_lResult = m_lMulti1 * m_lMulti2; }

One thing remains to be done before we can recompile the application. In the function CASRVView::OnCalculate there is a now obsolete reference to the DoCalculate member function of the document class. Change this reference to Calculate, and you can recompile the program.

The Type Library


Before you test your application, take a quick peek at one of the files generated and maintained by ClassWizard. The ASRV.odl file (Listing 27.7) is a script file created in Microsofts Object Definition Language. It is compiled using the MIDL compiler; the resulting type library (TLB) file is a compound document that can be accessed by automation clients using the ITypeComp, ITypeInfo, and ITypeLib COM interfaces.

504

OLE, COM, and MFC Applications PART IV

Listing 27.7. The ASRV.odl type library source file.


// ASRV.odl : type library source for ASRV.exe // This file will be processed by the MIDL compiler to produce the // type library (ASRV.tlb). [ uuid(CF21B4DE-8D39-11D0-B7B6-0207011C437C), version(1.0) ] library ASRV { importlib(stdole32.tlb); importlib(stdole2.tlb); // Primary dispatch interface for CASRVDoc

[ uuid(CF21B4DF-8D39-11D0-B7B6-0207011C437C) ] dispinterface IASRV { properties: // NOTE - ClassWizard will maintain property information here. // Use extreme caution when editing this section. //{{AFX_ODL_PROP(CASRVDoc) [id(1)] long Multi1; [id(2)] long Multi2; [id(3)] long Result; //}}AFX_ODL_PROP methods: // NOTE - ClassWizard will maintain method information here. // Use extreme caution when editing this section. //{{AFX_ODL_METHOD(CASRVDoc) [id(4)] void Calculate(); //}}AFX_ODL_METHOD }; // Class information for CASRVDoc

[ uuid(CF21B4DD-8D39-11D0-B7B6-0207011C437C) ] coclass CASRVDoc { [default] dispinterface IASRV; }; //{{AFX_APPEND_ODL}} };

The interfaces ICreateTypeInfo and ICreateTypeLib are used by the tools themselves (such as the MIDL compiler).

Testing the Application


Automation servers can best be tested from within a general-purpose automation controller. If you have Visual Basic installed on your system, you can use it as an ideal automation testing tool. (Both 16- and 32-bit versions should work fine together with 32-bit automation servers

Automation CHAPTER 27

505

created using Visual C++.) If you have no access to Visual Basic, you can use DISPTEXT.EXE; this scaled-down version of Visual Basic 3.0 is supplied with several versions of Visual C++ exactly for this purpose. To test the ASRV server from within DISPTEST, I created the simple form shown in Figure 27.10.

FIGURE 27.10.
Visual Basic form used for testing ASRV.

27
The three edit fields in this form are called Multi1, Multi2, and Result, respectively. The button with the equal sign in it is called Calculate. To perform the test, I attached the code shown in Listing 27.8 to the Form. The declaration (Dim statement) of ASRV is performed in the (general) section; the Form_Load subroutine is attached to the Load event of the Form object; the Calculate_Click subroutine is attached to the Click event of the Calculate button. By implementing the test application this way, you can avoid reloading the ASRV server every time a calculation is performed.

AUTOMATION

Listing 27.8. Visual Basic code used for testing ASRV.


Dim ASRV As object Sub Form_Load () Set ASRV = CreateObject(ASRV.Document) End Sub Sub Calculate_Click () ASRV.Multi1 = Val(Multi1.Text) ASRV.Multi2 = Val(Multi2.Text) ASRV.Calculate Result.Text = Str(ASRV.Result) End Sub

Note that during the test, the ASRV application never becomes visible. This is because when launched with the /Automation command-line parameter (and this is how Visual Basic launches it), the AppWizard-generated application never displays its main application window. If it were a multiple-document interface application that was already running, the behavior would be slightly different: The application window would remain visible, but no new frame window would be displayed to represent the new document. Either way, you should keep in mind that a document launched using automation may not have an associated view window. This Visual Basic application is identical in functionality and nearly identical in appearance to the original user interface of ASRV. The difference, of course, is that the ASRV application now performs as a black-box module. Not only does even this simple server application show the power of automation, it also demonstrates the potential of automation controllers such as Visual Basic to act as highly powerful system integration applications.

506

OLE, COM, and MFC Applications PART IV

Standard Methods and Properties


When building the ASRV server, I intentionally used a very simple set of methods and properties in order to keep the example program simple and focused. However, Microsoft actually provides a recommended set of properties and methods that production applications should implement: s Every application should implement at least one object, an Application object, with its standard set of methods and properties. s Applications should also implement, if applicable, collections of documents, document objects, and object collections within documents. This brings up an interesting issue that we have not had an opportunity to address during the construction of ASRV. ASRVs simple properties were merely long integers. The use of string properties is relatively straightforward, but what if the value of a property or the result of a method is another object? This is the case, for example, when an automation controller requests a particular item from an applications document collection. How is the document item passed to the controller? The answer is simple, but far from obvious. A method that returns another object should in fact return a pointer to an IDispatch COM interface. Such a pointer can be obtained for any CCmdTarget-derived object by calling the CCmdTarget member function GetIDispatch. For example, if your application object has a method that returns the active document, this method may be implemented similarly to the following:
LPDISPATCH GetActiveDocument() { CFrameWnd *pWnd = (CFrameWnd *)AfxGetMainWnd(); ASSERT_VALID(pWnd); CMyDocument *pDoc = pWnd->GetActiveDocument(); ASSERT_VALID(pDoc); return pDoc->GetIDispatch(TRUE); }

The hierarchy of standard objects that automation servers should support is shown in Figure 27.11.

FIGURE 27.11.
Standard automation objects.

Application Object

Documents Collection "Documents" Document

Objects Collection

Object

Automation CHAPTER 27

507

All objects must support two standard properties, shown in Table 27.1.

Table 27.1. Properties common to all standard objects. Property/method Mandatory Read/write
P: Application P: Parent Yes Yes Read-only Read-only

Description The application object The parent of the current object

All collection objects (for example, the documents collection) must support an additional set of properties and methods, shown in Table 27.2.

27
AUTOMATION

Table 27.2. Properties and methods for collection objects. Property/method Mandatory Read/write
P: Count P: _NewEnum M: Add M: Item M: Remove Yes Yes No Yes No Read-only Read-only

Description Number of items in collection Enumerator for iteration Adds an item to the collection Retrieves a collection item Removes a collection item

The Application Object


The application object represents the application itself. This object should support the set of properties and methods listed in Table 27.3.

Table 27.3. Properties and methods for application objects. Property/method Mandatory Read/write
P: ActiveDocument P: Caption P: DefaultFilePath P: Documents P: FullName P: Height P: Interactive P: Left No No No No Yes No No No Read-only Read/write Read/write Read-only Read-only Read/write Read/write Read/write

Description Active document object Application window title Default directory path The documents collection Executable full filename Height of application window User interaction allowed Application window left side
continues

508

OLE, COM, and MFC Applications PART IV

Table 27.3. continued Property/method


P: Name P: Path P: StatusBar P: Top P: Visible P: Width M: Help M: Quit M: Repeat M: Undo

Mandatory Yes No No No Yes No No Yes No No

Read/write Read-only Read-only Read/write Read/write Read/write Read/write

Description Application name Executable file path Status bar text Application window top Application window visible Width of application window Display online help Exit the application Repeat last action Undo last action

The Documents Collection


The documents collection represents all the applications currently active documents in a collection. This collection object should implement, in addition to properties and methods common to all objects and to all collections, an additional two methods, shown in Table 27.4.

Table 27.4. Methods of the documents collection. Property/method Mandatory Read/write


M: Close M: Open Yes Yes

Description Closes the document Opens a new document

The Document Object


The document object represents an individual document. In addition to standard properties and methods, document objects should support the properties and methods shown in Table 27.5.

Table 27.5. Properties and methods of document objects. Property/method Mandatory Read/write
P: Author P: Comments No No Read/write Read/write

Description Author of the document Document comments

Automation CHAPTER 27

509

Property/method P: FullName P: Keywords P: Name P: Path P: ReadOnly P: Saved P: Subject P: Title M: Activate M: Close M: NewWindow M: Print M: PrintOut M: PrintPreview M: RevertToSaved M: Save M: SaveAs

Mandatory Yes No No Yes No Yes No No Yes Yes No Yes No No No Yes Yes

Read/write Read-only Read/write Read-only Read-only Read-only Read/write Read/write Read/write

Description Document full filename Document keywords Document filename Document filepath Read-only document Changes were saved Subject of document Document title Activates first document window Closes the document Creates new window for document Prints the document Same as Print Activates Print Preview Reloads last saved version Saves the document Saves using filename

27
AUTOMATION

The Objects Collection


The objects collection is offered by document objects as applicable. This collection contains application-specific items. For example, in the case of a graphics application, the objects collection may represent a series of shapes that together comprise a drawing (a document). In the case of a word processor, the objects collection may represent paragraphs of text. Or, in the case of a spreadsheet application, this collection may represent the collection of spreadsheet cells. The implementation of the objects collection and that of individual objects (and indeed, whether these are implemented at all) is implementation-dependent. If you choose to implement these items, remember to add the methods and properties common to all objects, and those common to all collections.

510

OLE, COM, and MFC Applications PART IV

Summary
Automation is a means of communication between automation servers and automation controllers using the COM IDispatch interface. The MFC Library and Visual Studio provide extensive support for the development of automation servers. Constructing an automation server through AppWizard requires that you select the Automation check box in AppWizard Step 3. This adds automation support to your application and enables your applications document class for automation. Automation methods and properties are added through the ClassWizard. Methods are represented by member functions. Properties are either represented by member variables or by get/ set methods. If you wish to implement a read-only (or write-only) property, use get/set methods as the propertys implementation, and erase the unwanted method while still in ClassWizard. ClassWizard also creates a type library source file, which is a script written using the Object Description Language. When a type library is generated from this script using the MIDL compiler, it can be utilized by other applications for finding out about the servers automation interface. Automation servers can be tested and used from automation controllers. General-purpose automation controllers include programming environments such as Visual Basic or its scaled down test version, DISPTEST.EXE, which is supplied as a utility with Visual C++. The capability to use automation servers as black box components shows the power of automation as well as the power of Visual Basic and similar development systems as system integration tools. Microsoft recommends that automation applications expose a series of standard objects through properties and methods. These objects include the Application object, the applications documents collection, individual document objects, the collection of objects within a document, and, if applicable, objects (items) that comprise a document. A set of standard properties and methods is defined for these that other applications may attempt to utilize.

Building ActiveX Controls with MFC CHAPTER 28

511

Building ActiveX Controls with MFC

28

IN THIS CHAPTER
s Creating a Skeleton Control with AppWizard 513 s Customizing the Control 523 s Adding a Property Page Interface 531 s Testing, Distributing, and Using a Custom Control 534

28
BUILDING ACTIVEX CONTROLS WITH MFC

512

OLE, COM, and MFC Applications PART IV

Constructing an OLE custom control, as ActiveX controls were called until recently, used to be a mysterious task best left to gurus who write even their love letters in C++. Version 4 of Visual C++ changed that by adding advanced AppWizard support for OLE custom control development. It is now possible to create simple ActiveX controls with relatively little work. Visual C++ offers two mechanisms: You can build an MFC ActiveX control or you can build an ActiveX control based on the new ActiveX template library (ATL). This chapter describes the use of MFC in building ActiveX controls. In the next chapter, the use of ATL is reviewed. An ActiveX control is defined in terms of its appearance, properties, methods, and events. Control properties, methods, and events can be specified through the ClassWizard. To demonstrate the capabilities of an ActiveX control, I developed a simple control. This control, MCTL, consists of a single button that can have three different shapes (ellipse, rectangle, triangle). When the control is clicked on, it changes its state from deselected to selected or vice versa. When the control is selected, it appears red; otherwise, its color is green. The control generates events whenever its state changes. Figure 28.1 demonstrates the MCTL custom control in action.

FIGURE 28.1.
The MCTL control in action.

This control, like any other control, can be embedded in dialogs, used in HTML documents, or used otherwise by ActiveX control container applications. The Developer Studio fully supports the construction of dialogs that contain ActiveX controls and the use of ActiveX controls in applications through wrapper classes. The steps of creating an ActiveX control are as follows: 1. 2. 3. 4. 5. 6. 7. Create the ActiveX control skeleton through AppWizard. Add the controls properties, methods, and events through ClassWizard. Update the controls representative bitmap using the Developer Studio bitmap editor. Add drawing code to your control. Develop code for methods and events. Create the controls property page. Add code handling to the property page.

Building ActiveX Controls with MFC CHAPTER 28

513

In the remainder of this chapter, each of these steps is reviewed in detail. Throughout the review, the MCTL control serves as an example of putting the principles into practice.

Creating a Skeleton Control with AppWizard


The first step in constructing an ActiveX control is creating a control skeleton through AppWizardor, to be more precise, through the MFC ActiveX ControlWizard, which is a custom AppWizard developed specifically for the purposes of creating ActiveX controls.

Creating the Skeleton Control


To begin, select New from the Developer Studio File menu and then select the Projects tab in the New dialog. Select MFC ActiveX ControlWizard as the project type, type a name for the new project (for example, MCTL), and select a suitable directory (see Figure 28.2).

FIGURE 28.2.
Creating the MCTL control.

28
BUILDING ACTIVEX CONTROLS WITH MFC

After you click OK, the first page of the two-page MFC ActiveX ControlWizard appears (see Figure 28.3). On this page, you can specify the number of controls in your project (a single ActiveX control project can contain multiple controls), whether you want licensing support, and whether you want source comments and support for a help file. (Specifying help file support is generally a good idea; it is much easier to throw out code later than to try to add it to your project.) If you decide to add licensing support, the ControlWizard will create a default LIC file for your control. It will also add functions to your control class that support verification of licensing information. Without a valid license (LIC) file, it will not be possible to use the control in design mode. This is how users of your control should redistribute your control to end users (that is, without the license file); you, on the other hand, provide the control to your users with a valid license file that enables them to do development work with your control.

514

OLE, COM, and MFC Applications PART IV

FIGURE 28.3.
MFC ActiveX ControlWizard, page one.

In the second page of ControlWizard (see Figure 28.4), you can review and modify the class names that AppWizard generated for your controls. For every control in your project, AppWizard creates two classes: a control class and a class for the controls property page.

FIGURE 28.4.
ControlWizard, page two.

If your control requires more than one property page, dont worry; you will have an opportunity to create additional property pages. Use the ControlWizard to specify the parameters for the first property page. On this ControlWizard page, several additional properties can also be defined. I would like to bring your attention to two of these. The Invisible at runtime option enables you to create controls (such as the Microsoft Comm control) that do not have a visible interface at runtime.

Building ActiveX Controls with MFC CHAPTER 28

515

The Available in Insert Object dialog option enables you to create a control that can be inserted into container documents. This option should typically remain unchecked; after all, when you create a new button meant to be used in dialogs, you may not want someone to include it in a Word for Windows document! You can also specify a control class that the new custom control will subclass. For example, you can specify the BUTTON class to create a custom control that inherits some of the behavior of the standard Button control. The MCTL control was created with all ControlWizard settings left at their default values, with the exception of the setting that controls the generation of help files.

ActiveX Control Code Overview


Before proceeding, take a look at the code generated by ControlWizard and familiarize yourself with the new custom controls classes and resources. As always, it is a good idea to develop a basic understanding of these components before proceeding with custom modifications. If you look at your project in Class View (see Figure 28.5), you can see that the ControlWizard generated three classes for you. One represents the custom control library object; the other two represent the control object and the controls property page. (If you elected to create a project with more than one control, additional pairs of classes were also created by ControlWizard to represent additional control objects and their property pages.)

28
BUILDING ACTIVEX CONTROLS WITH MFC

FIGURE 28.5.
Classes generated by ControlWizard.

516

OLE, COM, and MFC Applications PART IV

Of the three classes in the MCTL project, the CMCTLApp class is perhaps the simplest. Derived from COleControlModule, this class represents the OCX object, that is, the library that contains your ActiveX controls. Only two member functions are declared (see Listing 28.1).

Listing 28.1. CMCTLApp class declaration.


class CMCTLApp : public COleControlModule { public: BOOL InitInstance(); int ExitInstance(); };

These functions have a trivial implementation (see Listing 28.2). In the same file, MCTL.cpp, there is also a pair of additional functions, DllRegisterServer and DllUnregisterServer. These two functions store and remove OLE registration information for your new controls in the Registry. The redistributable program regsvr32.exe calls these functions to register or unregister ActiveX controls.

Listing 28.2. CMCTLApp class implementation and control registration functions.


/////////////////////////////////////////////////////////////////// // CMCTLApp::InitInstance - DLL initialization BOOL CMCTLApp::InitInstance() { BOOL bInit = COleControlModule::InitInstance(); if (bInit) { // TODO: Add your own module initialization code here. } return bInit; } /////////////////////////////////////////////////////////////////// // CMCTLApp::ExitInstance - DLL termination int CMCTLApp::ExitInstance() { // TODO: Add your own module termination code here. return COleControlModule::ExitInstance(); } /////////////////////////////////////////////////////////////////// // DllRegisterServer - Adds entries to the system registry STDAPI DllRegisterServer(void) { AFX_MANAGE_STATE(_afxModuleAddrThis); if (!AfxOleRegisterTypeLib(AfxGetInstanceHandle(), _tlid)) return ResultFromScode(SELFREG_E_TYPELIB); if (!COleObjectFactoryEx::UpdateRegistryAll(TRUE)) return ResultFromScode(SELFREG_E_CLASS); return NOERROR; }

Building ActiveX Controls with MFC CHAPTER 28


/////////////////////////////////////////////////////////////////// // DllUnregisterServer - Removes entries from the system registry STDAPI DllUnregisterServer(void) { AFX_MANAGE_STATE(_afxModuleAddrThis); if (!AfxOleUnregisterTypeLib(_tlid)) return ResultFromScode(SELFREG_E_TYPELIB); if (!COleObjectFactoryEx::UpdateRegistryAll(FALSE)) return ResultFromScode(SELFREG_E_CLASS); return NOERROR; }

517

In terms of simplicity, the CMCTRLPropPage class follows. This class implements the property page of the control that design applications (such as the dialog template editor in Developer Studio) can display in a property sheet interface. The ControlWizard generates a blank property page for you (see Figure 28.6). It is up to you to add controls to this property page as they fit the requirements of your control.

FIGURE 28.6.
The default ControlWizardgenerated property page.

28
BUILDING ACTIVEX CONTROLS WITH MFC

Correspondingly, the default version of the CMCTLPropPage is equally simple. The class declaration (see Listing 28.3) only contains declarations for a constructor and a DoDataExchange member function. Notice the use of the DECLARE_DYNCREATE and DECLARE_OLECREATE_EX macros; the latter makes the runtime creation of the ActiveX control property page possible.

Listing 28.3. CMCTLPropPage class declaration.


class CMCTLPropPage : public COlePropertyPage { DECLARE_DYNCREATE(CMCTLPropPage) DECLARE_OLECREATE_EX(CMCTLPropPage) // Constructor public: CMCTLPropPage(); // Dialog Data //{{AFX_DATA(CMCTLPropPage) enum { IDD = IDD_PROPPAGE_MCTL }; // NOTE - ClassWizard will add data members here.

continues

518

OLE, COM, and MFC Applications PART IV

Listing 28.3. continued


DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_DATA // Implementation protected: virtual void DoDataExchange(CDataExchange* pDX); // Message maps protected: //{{AFX_MSG(CMCTLPropPage) // NOTE - ClassWizard will add and remove member functions // here. DO NOT EDIT what you see in these blocks of // generated code ! //}}AFX_MSG DECLARE_MESSAGE_MAP() }; //

Although this class is derived from COlePropertyPage, for all practical purposes, it works the same way as any CPropertyPage-derived class. In particular, you can use the ClassWizard to add member variables that correspond to controls in the property page. However, it is sometimes necessary to use additional data exchange functions and macrosfor example, when specifying a persistent custom control property. But more about that later in this chapter. The implementation file of CMCTLPropPage contains, in addition to trivial skeletal definitions of the constructor and the DoDataExchange member function, additional OLE-related elements (see Listing 28.4). Fortunately, it is rarely required to modify these elements manually; where necessary, the ClassWizard will insert items as applicable.

Listing 28.4. CMCTLPropPage class implementation.


IMPLEMENT_DYNCREATE(CMCTLPropPage, COlePropertyPage) /////////////////////////////////////////////////////////////////// // Message map BEGIN_MESSAGE_MAP(CMCTLPropPage, COlePropertyPage) //{{AFX_MSG_MAP(CMCTLPropPage) // NOTE - ClassWizard will add and remove message map entries // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_MSG_MAP END_MESSAGE_MAP() /////////////////////////////////////////////////////////////////// // Initialize class factory and guid IMPLEMENT_OLECREATE_EX(CMCTLPropPage, MCTL.MCTLPropPage.1, 0xb4a04ecc, 0x7ef0, 0x11d0, 0xae, 0xcd, 0x2, 0x7, 0x1, 0x1c, 0x43, 0x7c) /////////////////////////////////////////////////////////////////// // CMCTLPropPage::CMCTLPropPageFactory::UpdateRegistry // Adds or removes system registry entries for CMCTLPropPage BOOL CMCTLPropPage::CMCTLPropPageFactory::UpdateRegistry(BOOL bRegister) { if (bRegister) return AfxOleRegisterPropertyPageClass(AfxGetInstanceHandle(), m_clsid, IDS_MCTL_PPG);

Building ActiveX Controls with MFC CHAPTER 28


else return AfxOleUnregisterClass(m_clsid, NULL); } /////////////////////////////////////////////////////////////////// // CMCTLPropPage::CMCTLPropPage - Constructor CMCTLPropPage::CMCTLPropPage() : COlePropertyPage(IDD, IDS_MCTL_PPG_CAPTION) { //{{AFX_DATA_INIT(CMCTLPropPage) // NOTE: ClassWizard will add member initialization here // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_DATA_INIT } /////////////////////////////////////////////////////////////////// // CMCTLPropPage::DoDataExchange void CMCTLPropPage::DoDataExchange(CDataExchange* pDX) { //{{AFX_DATA_MAP(CMCTLPropPage) // NOTE: ClassWizard will add DDP, DDX, and DDV calls here // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_DATA_MAP DDP_PostProcessing(pDX); }

519

In fact, under normal circumstances, you will rarely need to edit this code manually at all; code supporting member variables that correspond to controls in the property page will be inserted automatically by ClassWizard. Instances involving the need for manual modification include the case when nonstandard control behavior is desired; but even in those cases, changes are often confined to the DoDataExchange member function. The last of the three ControlWizard-generated classes for the MCTL control is the class CMCTLCtrl. As its name implies, this class encapsulates the overall behavior of the custom control, and as such, can be considered perhaps the central element of the custom control project. The declaration of the CMCTLCtrl class is shown in Listing 28.5. I would like to call your attention to several elements in this declaration.

28
BUILDING ACTIVEX CONTROLS WITH MFC

Listing 28.5. CMCTLCtrl class declaration.


class CMCTLCtrl : public COleControl { DECLARE_DYNCREATE(CMCTLCtrl) // Constructor public: CMCTLCtrl(); // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CMCTLCtrl) public: virtual void OnDraw(

continues

520

OLE, COM, and MFC Applications PART IV

Listing 28.5. continued


CDC* pdc, const CRect& rcBounds, const CRect& rcInvalid); virtual void DoPropExchange(CPropExchange* pPX); virtual void OnResetState(); //}}AFX_VIRTUAL // Implementation protected: ~CMCTLCtrl(); DECLARE_OLECREATE_EX(CMCTLCtrl) // Class factory and guid DECLARE_OLETYPELIB(CMCTLCtrl) // GetTypeInfo DECLARE_PROPPAGEIDS(CMCTLCtrl) // Property page IDs DECLARE_OLECTLTYPE(CMCTLCtrl) // Type name and misc status // Message maps //{{AFX_MSG(CMCTLCtrl) // NOTE - ClassWizard will add and remove member functions here. // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_MSG DECLARE_MESSAGE_MAP() // Dispatch maps //{{AFX_DISPATCH(CMCTLCtrl) // NOTE - ClassWizard will add and remove member functions here. // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_DISPATCH DECLARE_DISPATCH_MAP() afx_msg void AboutBox(); // Event maps //{{AFX_EVENT(CMCTLCtrl) // NOTE - ClassWizard will add and remove member functions here. // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_EVENT DECLARE_EVENT_MAP() // Dispatch and event IDs public: enum { //{{AFX_DISP_ID(CMCTLCtrl) // NOTE: ClassWizard will add and remove enumeration // elements here. DO NOT EDIT what you see in these // blocks of generated code ! //}}AFX_DISP_ID }; };

The Implementation section of this declaration begins with a series of four macro calls that declare various OLE-related elements. There are corresponding macro calls implementing these elements in the implementation file. Of particular interest is the call to DECLARE_OLECREATE_EX; this macro call is different in the case of an ActiveX control that supports licensing. (It is replaced by a block of function declarations enclosed within a BEGIN_OLEFACTORY and an END_OLEFACTORY macro call.) In the remainder of the class declaration, the ActiveX controls message map, dispatch map, event map, and dispatch/event identifiers are declared. These maps are used to route messages and to define the controls OLE interface of properties, methods, and events. The ClassWizard adds items automatically to these sections of code; you rarely need to modify them directly.

Building ActiveX Controls with MFC CHAPTER 28

521

The implementation file of class CMCTLCtrl (see Listing 28.6) contains several elements that you may need to be familiar with.

Listing 28.6. CMCTLCtrl class implementation.


IMPLEMENT_DYNCREATE(CMCTLCtrl, COleControl) /////////////////////////////////////////////////////////////////// // Message map BEGIN_MESSAGE_MAP(CMCTLCtrl, COleControl) //{{AFX_MSG_MAP(CMCTLCtrl) // NOTE - ClassWizard will add and remove message map entries // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_MSG_MAP ON_OLEVERB(AFX_IDS_VERB_PROPERTIES, OnProperties) END_MESSAGE_MAP() /////////////////////////////////////////////////////////////////// // Dispatch map BEGIN_DISPATCH_MAP(CMCTLCtrl, COleControl) //{{AFX_DISPATCH_MAP(CMCTLCtrl) // NOTE - ClassWizard will add and remove dispatch map entries // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_DISPATCH_MAP DISP_FUNCTION_ID(CMCTLCtrl, AboutBox, DISPID_ABOUTBOX, AboutBox, VT_EMPTY, VTS_NONE) END_DISPATCH_MAP() /////////////////////////////////////////////////////////////////// // Event map BEGIN_EVENT_MAP(CMCTLCtrl, COleControl) //{{AFX_EVENT_MAP(CMCTLCtrl) // NOTE - ClassWizard will add and remove event map entries // DO NOT EDIT what you see in these blocks of generated code ! //}}AFX_EVENT_MAP END_EVENT_MAP() /////////////////////////////////////////////////////////////////// // Property pages // TODO: Add more property pages as needed. // Remember to increase the count! BEGIN_PROPPAGEIDS(CMCTLCtrl, 1) PROPPAGEID(CMCTLPropPage::guid) END_PROPPAGEIDS(CMCTLCtrl) /////////////////////////////////////////////////////////////////// // Initialize class factory and guid IMPLEMENT_OLECREATE_EX(CMCTLCtrl, MCTL.MCTLCtrl.1, 0xb4a04ecb, 0x7ef0, 0x11d0, 0xae, 0xcd, 0x2, 0x7, 0x1, 0x1c, 0x43, 0x7c) /////////////////////////////////////////////////////////////////// // Type library ID and version IMPLEMENT_OLETYPELIB(CMCTLCtrl, _tlid, _wVerMajor, _wVerMinor) /////////////////////////////////////////////////////////////////// // Interface IDs const IID BASED_CODE IID_DMCTL = { 0xb4a04ec9, 0x7ef0, 0x11d0, { 0xae, 0xcd, 0x2, 0x7, 0x1, 0x1c, 0x43, 0x7c } }; const IID BASED_CODE IID_DMCTLEvents = { 0xb4a04eca, 0x7ef0, 0x11d0, { 0xae, 0xcd, 0x2, 0x7, 0x1, 0x1c, 0x43, 0x7c } };

28
BUILDING ACTIVEX CONTROLS WITH MFC

continues

522

OLE, COM, and MFC Applications PART IV

Listing 28.6. continued


/////////////////////////////////////////////////////////////////// // Control type information static const DWORD BASED_CODE _dwMCTLOleMisc = OLEMISC_ACTIVATEWHENVISIBLE | OLEMISC_SETCLIENTSITEFIRST | OLEMISC_INSIDEOUT | OLEMISC_CANTLINKINSIDE | OLEMISC_RECOMPOSEONRESIZE; IMPLEMENT_OLECTLTYPE(CMCTLCtrl, IDS_MCTL, _dwMCTLOleMisc) /////////////////////////////////////////////////////////////////// // CMCTLCtrl::CMCTLCtrlFactory::UpdateRegistry // Adds or removes system registry entries for CMCTLCtrl BOOL CMCTLCtrl::CMCTLCtrlFactory::UpdateRegistry(BOOL bRegister) { if (bRegister) return AfxOleRegisterControlClass( AfxGetInstanceHandle(), m_clsid, m_lpszProgID, IDS_MCTL, IDB_MCTL, afxRegApartmentThreading, _dwMCTLOleMisc, _tlid, _wVerMajor, _wVerMinor); else return AfxOleUnregisterClass(m_clsid, m_lpszProgID); } /////////////////////////////////////////////////////////////////// // CMCTLCtrl::CMCTLCtrl - Constructor CMCTLCtrl::CMCTLCtrl() { InitializeIIDs(&IID_DMCTL, &IID_DMCTLEvents); // TODO: Initialize your controls instance data here. } /////////////////////////////////////////////////////////////////// // CMCTLCtrl::~CMCTLCtrl - Destructor CMCTLCtrl::~CMCTLCtrl() { // TODO: Cleanup your controls instance data here. } /////////////////////////////////////////////////////////////////// // CMCTLCtrl::OnDraw - Drawing function void CMCTLCtrl::OnDraw( CDC* pdc, const CRect& rcBounds, const CRect& rcInvalid) { // TODO: Replace the following code with your own drawing code. pdc->FillRect(rcBounds, CBrush::FromHandle((HBRUSH)GetStockObject(WHITE_BRUSH))); pdc->Ellipse(rcBounds); } /////////////////////////////////////////////////////////////////// // CMCTLCtrl::DoPropExchange - Persistence support void CMCTLCtrl::DoPropExchange(CPropExchange* pPX) {

Building ActiveX Controls with MFC CHAPTER 28


ExchangeVersion(pPX, MAKELONG(_wVerMinor, _wVerMajor)); COleControl::DoPropExchange(pPX); // TODO: Call PX_ functions for each persistent custom property. } /////////////////////////////////////////////////////////////////// // CMCTLCtrl::OnResetState - Reset control to default state void CMCTLCtrl::OnResetState() { COleControl::OnResetState(); // TODO: Reset any other control state here. } /////////////////////////////////////////////////////////////////// // CMCTLCtrl::AboutBox - Display an About box to the user void CMCTLCtrl::AboutBox() { CDialog dlgAbout(IDD_ABOUTBOX_MCTL); dlgAbout.DoModal(); }

523

The Message Map, Dispatch Map, and Event Map sections of this implementation are usually maintained by the ClassWizard. However, should you need to create additional property pages as part of your controls property page interface, you must manually modify the Property Pages section. For example, in a control that has three property pages, this section may contain the following code:
BEGIN_PROPPAGEIDS(CMCTLCtrl, 3) PROPPAGEID(CMCTLPropPage1::guid) PROPPAGEID(CMCTLPropPage2::guid) PROPPAGEID(CMCTLPropPage3::guid) END_PROPPAGEIDS(CMCTLCtrl)

28
BUILDING ACTIVEX CONTROLS WITH MFC

Of the functions defined in this file, three are of importance to the control programmer. The member function OnDraw is used to actually draw the control. The ControlWizard generates this member function with some default drawing instructions in it (either drawing a circle or, in the case of a subclassed control, calling the superclasss drawing function). The member function DoPropExchange implements property persistencythat is, the capability of saving properties at design time and reloading them when the control is displayed at runtime. More about this later in this chapter. The function OnResetState is used to reset property values to their defaults. Although the call to the base class implementation COleControl::OnResetState resets all persistent properties, it may be necessary to reset other properties manually in OnResetState.

Customizing the Control


Control properties, methods, and events can be added through the ClassWizard. A property is analogous to a data member of a C++ class, even though control properties are often implemented using get/set methods as opposed to member variables.

524

OLE, COM, and MFC Applications PART IV

A method is analogous to a member function (and is, in fact, implemented as a member function of the control class). Events are messages that the control can send to its parent window. When you add an event through the ClassWizard, the ClassWizard creates a member function in the control class that fires the event. You can call this member function from your own code as required. The MCTL control has two properties and one event. The first of the two properties is the Selected property; this Boolean property reflects the selected/deselected state of the control. Whenever the left mouse button is clicked on the control, the controls state will change from selected to deselected or vice versa. The Selected property will be implemented using get/set methods; this form of implementation enables us to create a read-only property by simply omitting the set method. The second property, Shape, is an integer property determining the controls shape. Its value can be 0 (elliptical shape), 1 (rectangular shape), or 2 (triangular shape). Although this is a read/ write property, we will make it writeable only in design mode. For this reason, this property will also be implemented using get/set methods, with a check for design mode as part of the set method member function. The single control event, the Select event, indicates to the parent window that the controls state changes. We will fire this event whenever the control receives a left mouse button click. (Needless to say, we must add a handler function for that event first.)

Changing the Controls Bitmap


Before we charge ahead adding code to the control, we must focus on the mundane task of updating the control bitmap. As mundane a task as it is, it should not be neglected, for it is the control bitmap that users of your control will most often see in tool palettes. Figure 28.7 shows the bitmap I drew for the MCTL control. (Okay, okayI know I am not a graphic artist!)

FIGURE 28.7.
MCTL control bitmap.

Adding Properties
Next you add the Selected and Shape properties to your control. To do so, invoke the ClassWizard and select the Automation tab. Select the CMCTLCtrl class (see Figure 28.8).

Building ActiveX Controls with MFC CHAPTER 28

525

FIGURE 28.8.
Adding control properties.

To add the Selected property, click on the Add Property button. In the Add Property dialog, type the name Selected in the External name field, select BOOL as the property Type, and select the get/set methods radio button under Implementation. At this point, the dialog automatically generates names for the propertys get and set functions. To render this property read-only, simply erase the name of the set function before dismissing the dialog with the OK button (see Figure 28.9).

28
BUILDING ACTIVEX CONTROLS WITH MFC

FIGURE 28.9.
Adding the Selected property.

Adding the Shape property is an almost identical process. There are two differences. First, the propertys type should be short; second, as this property is a read/write property, you should not erase the name of the set function this time.

526

OLE, COM, and MFC Applications PART IV

At this point, when you dismiss the ClassWizard by clicking the OK button, it generates a total of three new member functions in the control class. To reflect the controls selection state and shape, we must now manually add member variables to the control class declaration. Add the following two declarations to the declaration of the class CMCTLCtrl in the header file MCTLctl.h:
BOOL m_bSelected; short m_nShape;

It is also necessary to initialize these member variables to reasonable defaults in the constructor of the control class. Add the following lines to CMCTLCtrl::CMCTLCtrl:
m_bSelected = FALSE; m_nShape = 0;

With these member variables at hand, we can proceed with the implementation of the new get/set functions for which ClassWizard created skeletons. The completed functions are shown in Listing 28.7.

Listing 28.7. CMCTLCtrl property get/set methods.


BOOL CMCTLCtrl::GetSelected() { // TODO: Add your property handler here return m_bSelected; } short CMCTLCtrl::GetShape() { // TODO: Add your property handler here return m_nShape; } void CMCTLCtrl::SetShape(short nNewValue) { // TODO: Add your property handler here if (AmbientUserMode()) ThrowError(CTL_E_SETNOTSUPPORTEDATRUNTIME); else if (nNewValue < 0 || nNewValue > 2) ThrowError(CTL_E_INVALIDPROPERTYVALUE); else { m_nShape = nNewValue; SetModifiedFlag(); InvalidateControl(); } }

The functions GetSelected and GetShape are trivially simple; they just return the value of the m_bSelected and m_nShape member variables, respectively. The function SetSelected is somewhat more complicated; it uses the AmbientUserMode function to check whether the control container is in user mode or design mode, and disallows changing the controls shape in user mode. Note the use of the ThrowError function; this member function of COleControl is specifically intended to indicate an error while accessing a property via a get/set method.

Building ActiveX Controls with MFC CHAPTER 28

527

The framework provides an implementation for several stock properties. Stock properties include, among other things, properties relating to font and color. When adding a stock property, you can elect to use the stock implementation in COleControl instead of implementing the property yourself.

Making a Property Persistent


Having added a property is one thing; however, this does not automatically ensure that the propertys value is retained when it is set during a design session, for example. To make it happen, we must add a call to a PX_ function in the DoPropExchange member function of the control class. What does DoPropExchange do? Quite simply, this function is used to serialize property values into a property exchange object. The property achieves persistence by having its value stored in, and reloaded from, such an object. For example, the Developer Studio dialog editor writes the content of the property exchange object into the applications resource file, from which it is reloaded at runtime. For every persistent property, a function call of the appropriate PX_ function must be inserted in DoPropExchange. There is a large variety of PX_ functions corresponding to properties of various types. These are summarized in Table 28.1.

28
BUILDING ACTIVEX CONTROLS WITH MFC

Table 28.1. PX_ functions. Function name


PX_Blob PX_Bool PX_Color PX_Currency PX_Double PX_Float PX_Font PX_IUnknown PX_Long PX_Picture PX_Short PX_String PX_ULong PX_UShort PX_VBXFontConvert

Property type Binary large object (BLOB) data Boolean value (type BOOL) Color value (type OLE_COLOR) Currency value Value of type double Value of type float A font (pointer to a FONTDESC structure) An object with an IUnknown-derived interface Value of type long A picture (CPictureHolder reference) Value of type short A character pointer (type LPCSTR) Value of type unsigned long (ULONG) Value of type unsigned short Font-related properties of a VBX control

528

OLE, COM, and MFC Applications PART IV

In the case of MCTL, the one property that we wish to make persistent is the Shape property, which is of type s h o r t . Correspondingly, the following line must be added to the CMCTLCtrl::DoPropExchange function:
PX_Short(pPX, _T(Shape), m_nShape, 0);

Adding Methods
Adding methods to an ActiveX control is similar to adding properties. Methods are also added through the ClassWizard, using the Automation tab. Although we have not explicitly added any methods to the MCTL control, we have, nevertheless, added three methods; these are the get/set methods that we used for our two properties. A method can also have a list of parameters. These parameters are also defined through the ClassWizard. The COleControl class supports two stock events: (updates the controls appearance).
DoClick

(fires a

Click

event) and Refresh

Adding Events
Events are also added to an ActiveX control through the ClassWizard. To add an event, use the ClassWizards ActiveX Events tab (see Figure 28.10).

FIGURE 28.10.
Adding ActiveX events.

Building ActiveX Controls with MFC CHAPTER 28

529

To add the Select event, click on the Add Event button. In the Add Event dialog, type Select as the events External name. As you do that, ClassWizard automatically sets the internal name (the name of the function that causes the event to be fired) to FireSelect. Be sure to also add the IsSelected parameter to this event (this parameter will tell the recipient of the event whether the control has been selected or deselected) and specify this parameters type as BOOL (see Figure 28.11).

FIGURE 28.11.
Adding the Select event.

28
BUILDING ACTIVEX CONTROLS WITH MFC

As I mentioned before, this event will be fired when the controls state changes; that is, when the user clicks on the control with the left mouse button. To implement this behavior, use the ClassWizard to add a handler for the WM_LBUTTONDOWN message to the CMCTLCtrl class. In the handler function, shown in Listing 28.8, we negate the value of the m_bSelected Boolean variable and fire a Select event with this variables new value. We also ensure that the control is redrawn with the color properly indicating its selection state by calling the InvalidateControl function.

Listing 28.8. Handling a mouse event in MCTL.


void CMCTLCtrl::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default COleControl::OnLButtonDown(nFlags, point); m_bSelected = !m_bSelected; InvalidateControl(); FireSelect(m_bSelected); }

530

OLE, COM, and MFC Applications PART IV

Drawing the Control


There is nothing mysterious about actually drawing a control. When the control class member function OnDraw is called, it receives a pointer to a device context object and a rectangle identifying the controls bounds. With this information at hand, drawing the control is easy. Naturally, the controls appearance may depend on the values of various member variables representing the controls state. This is certainly the case with the MCTL control, where the m_nShape member variable determines the controls shape, and the m_bSelected member variable determines its color. Listing 28.9 shows my implementation of the MCTL controls drawing function.

Listing 28.9. CMCTLCtrl::OnDraw member function.


void CMCTLCtrl::OnDraw( CDC* pdc, const CRect& rcBounds, const CRect& rcInvalid) { // TODO: Replace the following code with your own drawing code. // pdc->FillRect(rcBounds, // CBrush::FromHandle((HBRUSH)GetStockObject(WHITE_BRUSH))); // pdc->Ellipse(rcBounds); CPen pen; CBrush foreBrush, backBrush; CPoint points[3]; pdc->SaveDC(); pen.CreatePen(PS_SOLID, 1, RGB(0, 0, 0)); backBrush.CreateSolidBrush(TranslateColor(AmbientBackColor())); foreBrush.CreateSolidBrush(GetSelected() ? RGB(255, 0, 0) : RGB(0, 255, 0)); pdc->FillRect(rcBounds, &backBrush); pdc->SelectObject(&pen); pdc->SelectObject(&foreBrush); switch (m_nShape) { case 0: pdc->Ellipse(rcBounds); break; case 1: pdc->Rectangle(rcBounds); break; case 2: points[0].x = rcBounds.left; points[0].y = rcBounds.bottom - 1; points[1].x = (rcBounds.left + rcBounds.right - 1) / 2; points[1].y = rcBounds.top; points[2].x = rcBounds.right - 1; points[2].y = rcBounds.bottom - 1; pdc->Polygon(points, 3); break; } pdc->RestoreDC(-1); }

Building ActiveX Controls with MFC CHAPTER 28

531

Note the use of the function AmbientBackColor. This is just one of several functions that can be used to retrieve the control containers ambient properties. In this case, this function ensures that the controls background color matches the background color of the control container. Note also that I use the TranslateColor function to translate an OLE color into an RGB value. This translation is necessary because the OLE color may occasionally represent a palette index as opposed to an RGB value.

Adding a Property Page Interface


Although implementation of your new controls behavior is complete, your task is not yet done. You must also implement one or more property pages through which your controls property settings can be controlled. These property pages will be used typically by applications in design mode (such as the Developer Studio dialog editor). The steps are no different from the steps used to implement an ordinary dialog. You must design the visual appearance of the property page and add the necessary code that connects the controls in the property page to the properties of the control. (As funny as the last sentence sounds, this is indeed the correct terminology. Controls in the property page refer to controls in a dialog; properties of the control refer to properties of the ActiveX control object. Unfortunately, the terminologies of Windows, ActiveX, C++, and object-oriented programming do conflict at times.)

28
BUILDING ACTIVEX CONTROLS WITH MFC

Editing the Property Page


To change the visual appearance of an ActiveX controls property page, just open the property pages dialog template. Editing it is no different from editing any other dialog resource. In the case of MCTLs single property page, there are only two controls to be added: a static label and a combo box. The combo box should be of type Drop List (this is important; otherwise, you will not be able to assign a member variable of type int to this control through ClassWizard), with the identifier IDC_CBSHAPE. The Developer Studio dialog editor can also be used to set its list box values (see Figure 28.12). Note that in order to enter multiple values in the area reserved for list box initializers, you must use the Ctrl+Enter key combinations to generate a line break, as pressing Enter alone would close the property sheet altogether.

Connecting the Property Page with Control Properties


A member variable for the new control can be added using regular ClassWizard procedures. What is not regular is how this member variable (that will be a member of the class CMCTLPropPage) connects with the appropriate property of the MCTL control. To create this connection, you must also specify the name of the ActiveX property in ClassWizards Add Member Variable dialog (see Figure 28.13).

532

OLE, COM, and MFC Applications PART IV

FIGURE 28.12.
Editing the MCTL property page.

FIGURE 28.13.
Connecting a control with a property through ClassWizard.

As a result of this, the ClassWizard inserts, in addition to any DDX_ and DDV_ function calls, a set of DDP_ function calls in the DoDataExchange member function of the property page class. This is the line that is inserted for MCTL:
DDP_CBIndex(pDX, IDC_CBSHAPE, m_nShape, _T(Shape) );

There are several DDP_ functions corresponding to the various control types. These functions are summarized in Table 28.2.

Table 28.2. DDP_ functions. Function name


DDP_CBIndex DDP_CBString

Description Combo box int transfer Combo box string transfer

Building ActiveX Controls with MFC CHAPTER 28

533

Function name
DDP_CBStringExact DDP_Check DDP_LBIndex DDP_LBString DDP_LBStringExact DDP_Radio DDP_Text

Description Combo box string transfer Check box List box int transfer List box string transfer List box string transfer Radio button int transfer Edit control transfer (string, numeric, and so on)

Additionally, the function DDP_PostProcessing should be called after the transfer of control properties through DDP_ functions has been completed. Take a look at the result. Figure 28.14 illustrates the MCTL control property page as it appears as part of a property sheet displayed by the Developer Studio dialog editor when an MCTL control is being added to a dialog template.

FIGURE 28.14.
MCTL control property sheet in Developer Studio.

28
BUILDING ACTIVEX CONTROLS WITH MFC

At this point, implementation of the MCTL control is complete. If you have not yet done so, you can recompile the project to create the OCX file. Compiling the project also automatically registers the control.

Additional Property Pages


If your ActiveX control requires additional property pages, those can be created by using the ControlWizard-generated property page as a model. To actually create the class, first create a dialog template; then use the ClassWizard and create a new class derived from COlePropertyPage. In case you are using the stock implementation of the color, font, or picture properties, you can use stock properties pages to provide an interface to them. To use a stock property page, just specify its class identifier when adding property pages in the implementation file of the control class. You do not need to create a class yourself for a stock property page.

534

OLE, COM, and MFC Applications PART IV

Testing, Distributing, and Using a Custom Control


Testing an ActiveX control is an important part of the development process. Not all real-life control projects are as simple as my tiny MCTL control. As reusable components, ActiveX controls are often created by one programmer but used by others. For this reason, it is important to know how a control can be distributed to its users. These issues and a brief reminder of how ActiveX controls are used in other applications are described in the remainder of this chapter.

Testing an ActiveX Control


In order to test an ActiveX control, you do not need to write your own control container application. Instead, you can use the ActiveX Control Test Container application that is supplied with Visual C++. You can use the Settings command under the Project menu in Developer Studio to specify tstcon32.exe as the debug executable; this way, you can exercise your control, set breakpoints, and control its execution through the Visual C++ debugger. You can also test your ActiveX control by attempting to use it within the Developer Studio dialog editor.

ActiveX Control Distribution


Distributing a custom control raises two questions. First, which files need to be distributed to control users and, in turn, by control users to end users of their applications? Second, what actions need to be taken during control installation to ensure that the control is correctly registered? The end user needs only the file with the .ocx extension. This file is the actual DLL containing the controls code. Control users (developers), on the other hand, may also need the TLB (type library) file, the EXP file, and the LIB file, all of which are generated when your control project is built. To register the control, run the server registration program supplied by Microsoft:
\msdev\bin\regsvr32.exe /s <controlname>.ocx

The regsvr32.exe file is freely redistributable. If you are writing a setup program that includes installation of ActiveX controls, you may want to use this file for control installation. However, you could also load the OCX file directly as a DLL and call its DllRegisterServer function. If you decide to use licensing for your control, remember to include the license (LIC) file with the files supplied to control users. This file should not be redistributed to end users.

Building ActiveX Controls with MFC CHAPTER 28

535

Using ActiveX Controls in Applications


Here is a brief summary of the steps that are required in order to use an ActiveX control in an application: 1. Add ActiveX control container support by calling AfxEnableControlContainer during application initialization. 2. Add the custom control to dialog templates as appropriate. 3. Set the controls initial properties using the controls property page interface. 4. Add member variables to represent the control or its properties as appropriate. 5. Add message handlers to handle messages from the control as appropriate.

ActiveX Controls on the Web


ActiveX controls can be used on Web Pages. In this case, the Web browser (typically Internet Explorer 3 or 4) acts as the control container. Generally, any self-registering ActvieX control can be used as Web content, although the security settings of Internet Explorer might prevent the instantiation of a control that has not been signed or marked.

Summary
Construction of an ActiveX control has been greatly simplified in Visual C++ 4 thanks to the MFC ActiveX ControlWizard. Visual C++ 5, further improved on this by adding support to the ActiveX Template Library. Constructing an ActiveX control using MFC involves the following steps: 1. 2. 3. 4. 5. 6. 7. Creating the ActiveX control skeleton through ControlWizard. Adding the controls properties, methods, and events through ClassWizard. Updating the controls bitmap. Adding drawing code to your control. Developing code for methods and events. Creating the controls property page. Adding code handling to the property page.

28
BUILDING ACTIVEX CONTROLS WITH MFC

ActiveX controls can be tested using the ActiveX Control Test Container that is supplied with Visual C++. Distributing a control to end users requires distributing the OCX file. Distributing the control to developers also requires distributing the LIB, TLB, and EXP files. Additionally, if the control is a licensed control, developers need the license (LIC) file.

536

OLE, COM, and MFC Applications PART IV

Before use, the control must be registered. Registration can be accomplished by using the freely distributable regsvr32.exe utility. Alternatively, you can call the controls DllRegisterServer function from within your own setup program. Self-registering controls can also be used on Web pages. To ensure that all users can run your control, sign it with a security certificate and mark it as safe.

Using the ActiveX Template Library CHAPTER 29

537

Using the ActiveX Template Library

29

IN THIS CHAPTER
s Why ATL 538 s Building an ActiveX Control with ATL 539

29
USING THE ACTIVEX TEMPLATE LIBRARY

538

OLE, COM, and MFC Applications PART IV

Microsofts ActiveX Template Library, or ATL, represents a new approach for providing class library support to the C++ programmer. This library wraps the Component Object Model (COM) in the form of a series of C++ class templates that correspond to COM object types. Special support is provided for key COM interfaces such as IUnknown, IClassFactory, or IDispatch. As such, ATL is ideally suited as the basis for ActiveX control or automation server projects. ATL is large and complex. Proper treatment of it would warrant several chapters, if not a book by itself. Therefore, I have limited this chapter to a cursory overview of ATLs functions and capabilities, followed by a hands-on introduction through a small ActiveX control project that is equivalent in terms of functionality to the MFC-based ActiveX control project presented in the preceding chapter. By comparing the two projects, you can decide for yourself which approach suits your needs best.

Why ATL?
Does ATL represent an alternative to MFC? Under what circumstances would you decide to use ATL in your projects? Can ATL and MFC coexist or are they mutually exclusive? What areas of functionality are unique to ATL? These are the questions that came into my mind when I first encountered this new library. ATL is indeed an MFC alternative when you build ActiveX objects. If anything, it provides a more thorough and flexible encapsulation of COM than MFC does. However, ATL is not an MFC replacement. MFC is intended for use in large, complex Windows applications, of which COM is but one feature. ATL, on the other hand, is mostly limited in scope to COM itself plus some basic dialog and window support. The most compelling reason for using ATL is the small footprint of the resulting code. Even the simple example presented later in this chapter demonstrates this very well. Do not be deceived by the apparent small size of MFC-based executables or libraries; these are typically a result of linking with the DLL version of MFC, which means that the bulk of MFC code is not in your executable or library file but in redistributable MFC files. It also means that you must distribute possibly several megabytes worth of library files with your control or automation object. A mere inconvenience with disk-based distributions, it can be a major headache if your library is meant to be used, for example, as active content in World Wide Web pages and distributed over the Internet.

Using the ActiveX Template Library CHAPTER 29

539

True, MFC projects can be recompiled using the static version of the MFC library. However, the resulting executable modules become much larger. The MFC-based ActiveX control project presented in this book yields an executable of approximately 150 kilobytes in size; in contrast, the ATL-based implementation of the same control is a mere one third in size. I think this difference in size speaks for itself. A strong reason that may prompt you to decide against the use of ATL is the lack of integration with Visual Studio. While MFC is supported thoroughly through the features of the Visual Studio ClassWizard, the same is not true for ATL. Support is essentially limited to the ATL COM AppWizard that helps you start off with a functional skeleton, and the New ATL Object command under the Insert menu, which helps you insert ATL objects into your project. ATL and MFC can coexist, but if you wish to use them together, you are on your own. In particular, conflicts are bound to happen if you are to use ATL and the MFC ActiveX support classes in the same project. Also, do not attempt to manage Windows objects created through ATL (for example, a property page or a control window) through MFC classes.

Building an ActiveX Control with ATL


The rest of this chapter presents, tutorial-style, the steps in constructing a functional ActiveX control using ATL. Here is a brief summary of these steps: 1. 2. 3. 4. 5. 6. 7. A skeleton ATL COM project is created using the ATL COM AppWizard. A control object is added to the project. The controls properties and methods are added. Code is added to draw the control. Support is added for events. Property pages are added. The controls bitmap is created.

29
USING THE ACTIVEX TEMPLATE LIBRARY

The control created in this chapter is illustrated in Figure 29.1. It consists of a single button that changes its color when it is clicked on by the user. The control can have three shapes (rectangle, ellipse, and triangle). The shape is determined by a property that can be changed only at design time.

540

OLE, COM, and MFC Applications PART IV

FIGURE 29.1.
The ACTL control.

Creating a Skeleton ATL COM Project


A skeleton project is created by invoking the ATL COM AppWizard. In the New dialog of Visual Studio, click on the Projects tab and specify ATL COM AppWizard as the project type. Specify a projects name and its location. I decided to call the control we are going to build ACTL. However, assigning the same name to both the control and the project that contains the control would result in conflicts, so the project itself will be called AC. (No, I am not fond of cryptic names. I pick them to ensure that project files will have names that do not exceed eight characters in length to avoid possible problems when they are copied to CD-ROM.) After you specify the projects type and its name, click the OK button. The ATL COM AppWizard dialog, illustrated in Figure 29.2, appears.

FIGURE 29.2.
Creating a project with the ATL COM AppWizard.

For our project, all the default settings presented in this dialog are fine. Our control will be an in-process server, that is, a DLL. We do not require the merging of MIDL-generated proxy and stub code into the DLL, and we do not need MFC support. We can therefore go ahead and click Finish, which invokes the New Project Information dialog (Figure 29.3) that summarizes project settings. Clicking OK here opens the new project in Visual Studio.

Using the ActiveX Template Library CHAPTER 29

541

FIGURE 29.3.
New ATL COM project summary.

Looking at the project in ClassView (Figure 29.4) makes it immediately obvious that an ATL project is very different from MFC projects. For one thing, there are no classes defined at all in the skeleton! All we got is a few tiny functions that implement standard DLL entry points. All these functions appear in the source file AC.cpp. Another file, AC.idl, provides type library information for our new project; in addition, the ATL COM AppWizard created a few more files, such as a nearly empty resource script, or a standard module definition file that specifies exported functions.

FIGURE 29.4.
The skeletal ATL COM project in ClassView.

29
USING THE ACTIVEX TEMPLATE LIBRARY

Before we proceed, we must look at the project configurations that were created by the ATL COM AppWizard. Selecting either the Configurations command from the Visual Studio Build menu or the Settings command from the Project menu (see Figure 29.5) can reveal that there are altogether two debug and four release configurations. Both debug and release configurations exist with and without Unicode support. For release configurations, you can also choose between minimizing the size of the resulting module and minimizing its dependencies. A minimum size ATL COM module requires the presence of the atl.dll library in order to run successfully.

542

OLE, COM, and MFC Applications PART IV

FIGURE 29.5.
ATL COM project settings.

The Project Settings dialog shown in Figure 29.5 is opened on the Link page with the Customize category showing. Here you can see that the projects output file is defined as AC.dll. Although technically this is correct, the convention is to use the .ocx extension for ActiveX controls. Should you decide to change the extension of the output file, please take note that you must do so individually for all six project configurations. You must also modify the module definition file (AC.def) to reflect this change.

Adding a Control
To insert a control into the project, select the New ATL Object from the Insert menu, and click on Controls in the ATL Object Wizard (Figure 29.6). Leave the default selection (Full Control) highlighted.

FIGURE 29.6.
Inserting a control with the ATL Object Wizard.

When you click Next, the ATL Object Wizard Properties dialog appears (Figure 29.7), where you can specify the various characteristics of the new control. Type ACTL in the Short Name field; this will cause all other fields in the Names page to assume compatible values.

Using the ActiveX Template Library CHAPTER 29

543

FIGURE 29.7.
ATL Object Wizard Properties: Names.

Before we dismiss this dialog, there is another setting that we must change, on the Attributes page. In order to support error handling, we must enable the Support ISupportErrorInfo setting (Figure 29.8). This setting will enable our control to do proper error handling.

FIGURE 29.8.
ATL Object Wizard Properties: Attributes.

None of the settings in the Miscellaneous and Stock Properties pages must be altered. You can complete the creation of the control by clicking OK. A look at the project in ClassView reveals two additions: a new class (CACTL) and a new COM interface (IACTL).

29
USING THE ACTIVEX TEMPLATE LIBRARY

Adding Properties and Methods


Properties and methods can be added to a COM interface using Visual Studio, as shown in Figure 29.9. For our ACTL control, two properties must be added. The Selected property will represent whether the object has been selected by clicking on it with the mouse, while the Shape property will specify the buttons current shape. Clicking on the Add Property menu command invokes the Add Property to Interface dialog. The first property, Shape, is to be of type short; the second, Selected, is to be of type BOOL. We also want to make the Selected property read-only because its value is set automatically when the user clicks on the control. To make a property read-only, simply clear the Put Function check box (Figure 29.10).

544

OLE, COM, and MFC Applications PART IV

FIGURE 29.9.
Adding COM properties in Visual Studio.

FIGURE 29.10.
Specifying a read-only property.

At this point, Visual Studio will have added three new member functions to the CACTL class. These functions are get_Shape, put_Shape, and get_Selected, and their implementations can be found in the file ACTL.cpp. Our next task is to add code to these functions that will properly implement the desired functionality. We must modify two files, ACTL.h and ACTL.cpp. ACTL.h contains the declaration of the class
CACTL. Toward the bottom of the file, we must add declarations for two member variables. Add

a few lines at the end of this classs declaration, as follows:


// IACTL public: STDMETHOD(get_Selected)(/*[out, retval]*/ BOOL *pVal); STDMETHOD(get_Shape)(/*[out, retval]*/ short *pVal); STDMETHOD(put_Shape)(/*[in]*/ short newVal); HRESULT OnDraw(ATL_DRAWINFO& di);

Using the ActiveX Template Library CHAPTER 29


protected: BOOL m_bSelected; short m_nShape; public: };

545

We also must initialize these variables in the classs constructor, defined earlier in the file:
public: CACTL() { m_bSelected = FALSE; m_nShape = 0; } DECLARE_REGISTRY_RESOURCEID(IDR_ACTL)

Next, we must provide an implementation for the get_Shape, put_Shape, and get_Selected functions in ACTL.cpp. These three functions are shown in Listing 29.1. The implementations of the two Get functions are trivially simple. The put_Shape function is somewhat more complex, because in addition to setting the m_nShape variable to a new value, we also perform error checking; furthermore, we notify the control that its appearance has changed and it needs to be redrawn.

Listing 29.1. Get and Put functions in ACTL.


STDMETHODIMP CACTL::get_Shape(short * pVal) { // TODO: Add your implementation code here *pVal = m_nShape; return S_OK; } STDMETHODIMP CACTL::put_Shape(short newVal) { // TODO: Add your implementation code here BOOL bUserMode; GetAmbientUserMode(bUserMode); if (bUserMode) return Error(_T(Not supported at run-time)); else if (newVal < 0 || newVal > 2) return Error(_T(Invalid property value)); else { m_nShape = newVal; FireViewChange(); } return S_OK; } STDMETHODIMP CACTL::get_Selected(BOOL * pVal) {

29
USING THE ACTIVEX TEMPLATE LIBRARY

continues

546

OLE, COM, and MFC Applications PART IV

Listing 29.1. continued


// TODO: Add your implementation code here *pVal = m_bSelected; return S_OK; }

Error checking is used to verify two things: that the new shape value is between 0 and 2, and that the control is being operated in design mode. The latter is accomplished by calling the CComControl::GetAmbientUserMode function. Errors are returned through the CComCoClass::Error member function. This functionality is supported because we selected the Support ISupportErrorInfo setting when the control was created.

Adding Drawing Code


Now that we have the properties set up that define our controls appearance, it is time to add code that draws the control. This is accomplished by modifying the default implementation of the CACTL::OnDraw member function. (See Listing 29.2.)

Listing 29.2. The CACTL::OnDraw member function.


HRESULT CACTL::OnDraw(ATL_DRAWINFO& di) { RECT& rc = *(RECT*)di.prcBounds; HPEN hPen; HBRUSH hForeBrush, hBackBrush; POINT points[3]; OLE_COLOR ocBack; COLORREF crBack; HPALETTE hPalAmb; SaveDC(di.hdcDraw); hPen = CreatePen(PS_SOLID, 1, RGB(0, 0, 0)); GetAmbientBackColor(ocBack); GetAmbientPalette(hPalAmb); OleTranslateColor(ocBack, hPalAmb, &crBack); hBackBrush = CreateSolidBrush(crBack); hForeBrush = CreateSolidBrush(m_bSelected ? RGB(255, 0, 0) : RGB(0, 255, 0)); FillRect(di.hdcDraw, &rc, hBackBrush); SelectObject(di.hdcDraw, hPen); SelectObject(di.hdcDraw, hForeBrush); switch (m_nShape) { case 0: Ellipse(di.hdcDraw, rc.left, rc.top, rc.right, rc.bottom); break; case 1: Rectangle(di.hdcDraw, rc.left, rc.top, rc.right, rc.bottom); break;

Using the ActiveX Template Library CHAPTER 29


case 2: points[0].x = rc.left; points[0].y = rc.bottom - 1; points[1].x = (rc.left + rc.right - 1) / 2; points[1].y = rc.top; points[2].x = rc.right - 1; points[2].y = rc.bottom - 1; Polygon(di.hdcDraw, points, 3); break; } RestoreDC(di.hdcDraw, -1); return S_OK; }

547

The new implementation contains no mysteries. Depending on the property settings, the appropriate shape with the appropriate color is drawn within the rectangle that this function receives through its ATL_DRAWINFO parameter.

Adding a Property Page


The next major item that must be added to our control is a property page where the user can specify the controls shape. We would like to have a simple dialog in which a combo box provides the three shape selections. Adding the property page again requires the use of the New ATL Object command from the Insert menu. Once again, select Controls in the ATL Object Wizard dialog, but this time select the Property Page icon (see Figure 29.11). Click Next.

FIGURE 29.11.
Adding a property page.

29
USING THE ACTIVEX TEMPLATE LIBRARY

In the ATL Object Wizard dialog, type ACTLProp as the short name of our property (see Figure 29.12). Other values will be updated automatically. Next, click on the Strings tab (see Figure 29.13). Type the appropriate strings in the Title and Doc String fields. Erase the contents of the Helpfile field; our control does not have a help file.

548

OLE, COM, and MFC Applications PART IV

FIGURE 29.12.
Specifying the name of the property page.

FIGURE 29.13.
Property page strings.

When you click OK, the ATL Object Wizard creates the new property page. This includes a new resource, the dialog IDD_ACTLPROP. Three new files will also have been created that declare the class CACTLProp (in ACTLProp.h), provide its definition (ACTLProp.cpp), and provide a Registry script for the property page object (ACTLProp.rgs). The remaining steps are relatively straightforward. First, edit the dialog template resource and add a combo box and a label as shown in Figure 29.14. Set the identifier of the combo box to IDC_CBSHAPE; also make sure that its style is set to Drop List and that it is not sorted. Save the dialog.

FIGURE 29.14.
The property page dialog template.

Using the ActiveX Template Library CHAPTER 29

549

An interesting question that might come into mind at this point is this: Why dont we use the built-in feature of the Visual Studio dialog editor to specify the list of items in the combo box? Unfortunately, this feature is MFC-specific; in non-MFC applications like our ATL control, we must populate the combo box manually. This is, indeed, one of the tasks that remains: to add code that initializes the dialog. We must also ensure that selections made in this dialog are properly reflected by the control by modifying the default implementation of CACTLProp::Apply. First, we need two message handler functions. One will respond to WM_INITDIALOG messages; the other, a command handler, will respond to CBN_SELCHANGE notifications from the combo box. These functions must be declared in the CACTLProp class declaration and added to the message map, by modifying ACTLProp.h as follows:
BEGIN_MSG_MAP(CACTLProp) MESSAGE_HANDLER(WM_INITDIALOG, OnInitDialog) COMMAND_HANDLER(IDC_CBSHAPE, CBN_SELCHANGE, OnShapeSelchange); CHAIN_MSG_MAP(IPropertyPageImpl<CACTLProp>) END_MSG_MAP() LRESULT OnInitDialog(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled); LRESULT OnShapeSelchange(WORD wNotify, WORD wID, HWND hWnd, BOOL& bHandled);

The implementation of these two functions, shown in Listing 29.3, must be inserted into the file ACTLProp.cpp. CACTLProp::OnInitDialog inserts three strings into the combo box; it also sets the combo box to reflect the current shape in the control. Notice the test for m_nObject == 1; this is used in order to properly handle cases when the property page is invoked for more than one control. When that happens, the combo box will not be set to a specific value because several controls can have different shape settings.

Listing 29.3. Message handler functions in CACTLProp.


LRESULT CACTLProp::OnInitDialog(UINT, WPARAM, LPARAM, BOOL &bHandled) { HWND hCB = GetDlgItem(IDC_CBSHAPE); ::SendMessage(hCB, CB_RESETCONTENT, 0, 0); ::SendMessage(hCB, CB_ADDSTRING, 0, (LPARAM)_T(Ellipse)); ::SendMessage(hCB, CB_ADDSTRING, 0, (LPARAM)_T(Rectangle)); ::SendMessage(hCB, CB_ADDSTRING, 0, (LPARAM)_T(Triangle)); if (m_nObjects == 1) { CComQIPtr<IACTL, &IID_IACTL> pACTL(m_ppUnk[0]); short nShape; pACTL->get_Shape(&nShape); ::SendMessage(hCB, CB_SETCURSEL, nShape, 0); } bHandled = TRUE; return S_OK; } LRESULT CACTLProp::OnShapeSelchange(WORD, WORD, HWND, BOOL& bHandled) { SetDirty(TRUE); bHandled = TRUE; return S_OK; }

29
USING THE ACTIVEX TEMPLATE LIBRARY

550

OLE, COM, and MFC Applications PART IV

The CACTLProp::OnShapeSelchange member function merely calls IPropertyPageImpl::SetDirty to indicate that the property page has changed so that, for example, the Apply button can be enabled in a property sheet containing this property page. The function in ACTLProp.h that corresponds to the Apply button, CACTLProp::Apply, is shown in Listing 29.4. This function updates the control in accordance with selections made in the property page.

Listing 29.4. The CACTLProp::Apply member function.


STDMETHOD(Apply)(void) { ATLTRACE(_T(CACTLProp::Apply\n)); for (UINT i = 0; i < m_nObjects; i++) { CComQIPtr<IACTL, &IID_IACTL> pACTL(m_ppUnk[i]); HWND hCB = GetDlgItem(IDC_CBSHAPE); int nShape = ::SendMessage(hCB, CB_GETCURSEL, 0, 0); if (nShape < 0) nShape = 0; if (nShape > 2) nShape = 2; pACTL->put_Shape(nShape); } m_bDirty = FALSE; return S_OK; }

We are almost finished. Two little housekeeping chores remain. First, the file AC.h must be included at the top of ACTLProp.cpp right after stdafx.h; otherwise, the project will not compile successfully. Next, we must modify ACTL.h to include a reference to the new property page:
BEGIN_PROPERTY_MAP(CACTL) // Example entries // PROP_ENTRY(Property Description, dispid, clsid) PROP_ENTRY(Shape, 1, CLSID_ACTLProp) END_PROPERTY_MAP()

Event Handling
In MFC projects, event handler functions can easily be specified through the ClassWizard. In ATL projects, although some of the subtasks are automated, much of the work must be done manually. This is not exactly a simple exercise. Hopefully, in the not-too-distant future, Visual Studio will provide features that better automate the repetitive elements of the procedure described here. Why is event handling complicated? Because we want to do more than just process a Windows message. We want the control to respond by firing an event itself, an event that can be processed by the control container. Accomplishing this requires several steps.

Using the ActiveX Template Library CHAPTER 29

551

The first step in supporting events is specifying an event interface. Several lines must be added to the AC.idl file that contains the interface definitions for our control. These new lines appear in the second half of this file as follows:
library ACLib { importlib(stdole32.tlb); importlib(stdole2.tlb); [ uuid(C0E5303C-D750-11D1-8DA7-00E02910AE47), helpstring(Event interface for ACTL) ] dispinterface _AEvents { properties: methods: [id(1)] void Select([in]BOOL bSelected); }; [ uuid(C0E5303A-D750-11D1-8DA7-00E02910AE47), helpstring(ACTL Class) ] coclass ACTL { [default] interface IACTL; [default, source] dispinterface _AEvents; }; [ uuid(C0E5303B-D750-11D1-8DA7-00E02910AE47), helpstring(ACTLProp Class) ] coclass ACTLProp { interface IUnknown; }; };

29
USING THE ACTIVEX TEMPLATE LIBRARY

Note that the GUIDs in your project may differ from the ones shown here. For the new event interface, I created a GUID from the GUID of the property page by simply incrementing the first group of hexadecimal digits by one. These lines, in addition to specifying the event interface, also make it the default source interface. A source interface indicates that the control is the source of the notifications. To support events, we must also implement a connection point interface. This is accomplished using the ATL Proxy Generator component in the Gallery. In order to use this component, however, you must first create the controls type library (TLB) file. You can create this file without rebuilding the entire project by opening the file AC.idl in Visual Studio and selecting the Compile command from the Build menu. After you have that file, invoke the Component and Controls Gallery (Project menu, Add to Project submenu, Components and Controls command), and select the Developer Studio Components folder. Double-click the ATL Proxy Generator component (see Figure 29.15).

552

OLE, COM, and MFC Applications PART IV

FIGURE 29.15.
Invoking the ATL Proxy Generator.

In the ATL Proxy Generator dialog (Figure 29.16) click the ellipses button and select the AC.tlb file. This will cause _AEvents and IACTL to appear in the Not Selected list. To generate a connection point for the event interface, select _AEvents and click the > button. You can now click the Insert button and accept the suggested filename (CPAC.h) that appears in the standard Save dialog. Click on the Close buttons to dismiss the ATL Proxy Generator dialog and the Components and Controls dialog.

FIGURE 29.16.
Adding the connection point.

Using the ActiveX Template Library CHAPTER 29

553

We now must add several things to ACTL.h. First, include the file CPAC.h at the top:
#ifndef __ACTL_H_ #define __ACTL_H_

#include resource.h #include CPAC.h

// main symbols

Next, we must add two classes to the IConnectionPointContainerImpl:

CACTL

class inheritance list,

CProxy_AEvents

and

class ATL_NO_VTABLE CACTL : public CComObjectRootEx<CComSingleThreadModel>, public CComCoClass<CACTL, &CLSID_ACTL>, public CComControl<CACTL>, public IDispatchImpl<IACTL, &IID_IACTL, &LIBID_ACLib>, public IProvideClassInfo2Impl<&CLSID_ACTL, NULL, &LIBID_ACLib>, public IPersistStreamInitImpl<CACTL>, public IPersistStorageImpl<CACTL>, public IQuickActivateImpl<CACTL>, public IOleControlImpl<CACTL>, public IOleObjectImpl<CACTL>, public IOleInPlaceActiveObjectImpl<CACTL>, public IViewObjectExImpl<CACTL>, public IOleInPlaceObjectWindowlessImpl<CACTL>, public IDataObjectImpl<CACTL>, public ISupportErrorInfo, public ISpecifyPropertyPagesImpl<CACTL>, public CProxy_AEvents<CACTL>, public IConnectionPointContainerImpl<CACTL> {

To make eter:

_AEvents

IProvideClassInfo2Impl

the default outgoing interface, we must also modify the entry in the inheritance list, making this interface the second param-

public IProvideClassInfo2Impl<&CLSID_ACTL, &DIID_ _AEvents, &LIBID_ACLib>,

29
USING THE ACTIVEX TEMPLATE LIBRARY

The connection point interface must also be exposed. To do so, add a COM_INTERFACE_ENTRY_IMPL entry to the COM map (the block of code enclosed by the BEGIN_COM_MAP and END_COM_MAP macros):
COM_INTERFACE_ENTRY_IMPL(IConnectionPointContainer)

The last thing that must be done for connection points is to specify to the ATL implementation which connection points are available. This is accomplished by adding a connection point map to your class declaration. To do so, add the following three lines of code after the COM map:
BEGIN_CONNECTION_POINT_MAP(CACTL) CONNECTION_POINT_ENTRY(DIID_ _AEvents) END_CONNECTION_POINT_MAP()

We are finished adding the connection point; finally, we can add the actual event. Doing so requires first of all adding a new macro to the list of message handlers in ACTL.h and declaring the message handler function:

554

OLE, COM, and MFC Applications PART IV


BEGIN_MSG_MAP(CACTL) MESSAGE_HANDLER(WM_PAINT, OnPaint) MESSAGE_HANDLER(WM_SETFOCUS, OnSetFocus) MESSAGE_HANDLER(WM_KILLFOCUS, OnKillFocus) MESSAGE_HANDLER(WM_LBUTTONDOWN, OnLButtonDown) END_MSG_MAP() ... // IACTL public: STDMETHOD(get_Selected)(/*[out, retval]*/ BOOL *pVal); STDMETHOD(get_Shape)(/*[out, retval]*/ short *pVal); STDMETHOD(put_Shape)(/*[in]*/ short newVal); HRESULT OnDraw(ATL_DRAWINFO& di); LRESULT OnLButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled); protected: BOOL m_bSelected; short m_nShape; };

The implementation of the OnLButtonDown function, shown in Listing 29.5, goes into the ACTL.cpp file. Add this function right after the definition of CACTL::OnDraw. Again, there are few mysteries; when the user clicks the left mouse button, the m_bSelected variable is negated. CComControl::FireViewChange is then called to update the controls visual appearance; CProxy_AEvents::Fire_Select is then called to send the event to the controls container.

Listing 29.5. The CACTL::OnLButtonDown member function.


LRESULT CACTL::OnLButtonDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled) { m_bSelected = !m_bSelected; FireViewChange(); Fire_Select(m_bSelected); bHandled = TRUE; return S_OK; }

Whew. This concludes adding an event to our control. I really really miss that ClassWizard!

Adding a Bitmap
The last step in constructing our control is purely cosmetic: we want to add a bitmap that container applications will use when representing the control on a toolbar button, for example. As it turns out, the ATL COM AppWizard has already created a Registry entry for this bitmap in the Registry script for our control. If you open the file ACTL.rgs, you will find the following line in it:
ForceRemove ToolboxBitmap32 = s %MODULE%, 1

Using the ActiveX Template Library CHAPTER 29

555

This line specified the bitmap with identifier 1 as the bitmap representing the control. So all that remains to be done is to create that bitmap! Create a bitmap using Visual Studio, open its properties page using the Properties command in the View menu, and specify its settings as shown in Figure 29.17. In particular, make sure whatever the bitmaps identifier is, its numeric value is set to 1.

FIGURE 29.17.
Specifying a bitmap representing the control.

Thats it. The new control can be recompiled and is ready for use (see Figure 29.18).

FIGURE 29.18.
The completed ACTL control.

29
USING THE ACTIVEX TEMPLATE LIBRARY

556

OLE, COM, and MFC Applications PART IV

Testing the Control


We can test our newly created control by recompiling it and invoking it from the ActiveX Control Test Container helper application. To invoke our control, start the ActiveX Control Test Container, select the Insert OLE Control command from its Edit menu, and pick ACTL Class from the list of controls (see Figure 29.19). The control should appear in the client area of the ActiveX Control Test Container as a green rectangle.

FIGURE 29.19.
Inserting the ACTL control.

Clicking on the control causes it to change color from green to red and vice versa. You can also watch events sent to the container by selecting the Event Log command from the View menu; every time you click the control, a new event will be registered by the container application (see Figure 29.20).

FIGURE 29.20.
Watching events in the ActiveX Control Test Container.

Using the ActiveX Template Library CHAPTER 29

557

Invoking the Properties command (Edit | Embedded Object Functions) opens the Properties dialog, in which you can see the property page we created for our control. Selecting a different shape and clicking the Apply button will change the controls shape, but we must do something else first: we must first change the UserMode ambient property to FALSE; otherwise, our control will report an error and not allow its Shape property to be updated. To change the UserMode property, invoke the Set Ambient Properties command from the Edit menu, select UserMode, and click on False (see Figure 29.21).

FIGURE 29.21.
Setting the UserMode ambient property.

Before we finish with the test container, note the new button in its toolbar. This button can be used to insert additional controls of this type, and contains the bitmap we recently drew to represent our control.

Summary
The ActiveX Template Library represents a new way of creating ActiveX objects. Its advantage over MFC is its significantly smaller footprintof special significance for ActiveX objects that are meant to be transferred over the Internet, for example. Its major disadvantage is a much more involved process in creating objects, with significantly less support from Visual Studio. Visual Studio support for ATL consists of the ATL COM AppWizard, the ATL Object Wizard, and the ATL Proxy Generator component in the Gallery. The main steps in creating an ActiveX control using ATL can be summarized as follows: 1. A skeleton ATL COM project is created using the ATL COM AppWizard 2. A control object is added to the project 3. The controls properties and methods are added

29
USING THE ACTIVEX TEMPLATE LIBRARY

558

OLE, COM, and MFC Applications PART IV

4. 5. 6. 7.

Code is added to draw the control Support is added for events Property pages are added The controls bitmap is created

Many of these steps are rather involved and require a fair degree of understanding of the Component Object Model and the ActiveX Template Library.

ActiveX Documents CHAPTER 30

559

ActiveX Documents

30

IN THIS CHAPTER
s Overview 560 s The ActiveX Document Interface 561 s Creating ActiveX Document Applications 562

30
ACTIVEX DOCUMENTS

560

OLE, COM, and MFC Applications PART IV

Have you ever used the Microsoft Office Binder? You know, the one that lets you create a workbook of sorts consisting of individual Word documents, Excel spreadsheets, and whatever else? If you used this application, youre already familiar with the concept of ActiveX documents.

Overview
At first sight, this looks like another example of the standard magic of OLE (see Figure 30.1). Obviously, the Office Binder is a container application that can display embedded Word documents, Excel spreadsheets, and other objects. However, if you look a little closer youll notice some significant differences between the Office Binder and a standard OLE container.

FIGURE 30.1.
The Microsoft Office Binder (Office 97).

For starters, objects no longer appear embedded in a parent document. A Word document in an Office Binder doesnt appear as an icon or a rectangle; instead, it takes over the entire client area of the Office Binder window. Menus and toolbars also behave differently. Indeed, the differences between OLE in-place editing and ActiveX documents can be summarized as follows: s An ActiveX document, if displayed, is always active, whereas an embedded or linked OLE object must be activated for in-place editing. s An ActiveX document server implements the view for the documents for which it acts as a server. As a consequence, for every document object a server application must be present on the system before the object can be viewed (however, this server can be a stripped-down viewer as opposed to the full-featured application used for editing).

ActiveX Documents CHAPTER 30

561

s When an ActiveX document is active, it takes over the entire document window of the container application, not just a specific area surrounded by a hatched border. s When an ActiveX document is active, all menu and toolbar commands of the container application can be routed to it. When I used the Microsoft Office Binder as an example for ActiveX document technology, it was no accident: the technology first appeared in the form of that application. Since then, Microsoft realized that the technology was sufficiently generic to be released to outside programmers. ActiveX documents can also be viewed in Internet Explorer. This provides seamless integration between Web pages and other documents.

The ActiveX Document Interface


The technology of ActiveX documents extends the traditional set of COM interfaces used to implement OLE. In particular, three new interfaces are introduced: IOleDocument , IOleDocumentSite, and IOleDocumentView. Before you proceed to the practical matter of implementing ActiveX support, first examine the role of these interfaces in ActiveX document container clients and servers. Note that in addition to these interfaces, ActiveX document containers and servers must still implement all COM interfaces that are required to support regular OLE in-place activation and editing, such as IOleInPlaceSite and IOleInPlaceFrame for containers, or IPersistStorage, IPersistSite, IOleObject, IDataObject, IOleInPlaceObject, and IOleInPlaceActiveObject for servers. It might also be necessary for ActiveX containers and servers to dispatch commands to each other. This is supported through the IOleCommandTarget interface that can be optionally implemented by both containers and servers. A good example for its use is the Print command (in particular, how an ActiveX document container dispatches the print request to its various component documents); additional support for implementing seamless printing is provided through the IPrint interface.

ActiveX Document Servers


The most important interface an ActiveX document must implement is IOleDocument. Through this interface a container application can ask a document object to create views of itself. This is accomplished through the CreateView method. Another method, EnumViews, can be used to enumerate the existing views of a document. A view is represented by the IOleDocumentView interface. Containers use the methods of this interface to manipulate a documents views. A document can have several different types of views or only one view type.

30
ACTIVEX DOCUMENTS

562

OLE, COM, and MFC Applications PART IV

ActiveX Document Containers


A container application that is to support ActiveX document objects must implement the IOleDocumentSite interface. This interface has a single method, ActivateMe. By calling this method, ActiveX documents can ask a document container to directly activate the document, as opposed to going through the normal sequence of in-place application. Optionally, the document can also provide a pointer to an IOleDocumentView interface, which identifies the view that is to be activated.

Creating ActiveX Document Applications


Using low-level functions to implement the ActiveX document architecture can be tedious and complicated. Fortunately, at least for ActiveX document servers, the Visual C++ AppWizard and the MFC library come to the rescue. It is also possible to modify an existing OLE server application to also act as an ActiveX document server. Later in this chapter these techniques are reviewed through examples, but first, let us build an ActiveX document container.

Creating an ActiveX Document Container


Support for ActiveX document containers has not yet been implemented within the Microsoft Foundation Classes, at least not as of the Visual C++ 5 release. That is a blessing in disguise. Although the lack of support might be an inconvenience, I believe that the construction of an ActiveX document container from scratch is something few programmers are required to do. On the other hand, this situation gives me the perfect opportunity to expose a little bit the internals of how ActiveX document support is implemented in practice. This is possible because although building a full-blown ActiveX document container application is beyond the scope of the present chapter, we can create an example that implements the most essential feature of such a program: the ability to host document objects. To begin, we must create an OLE container application using the MFC AppWizard, with all the default settings. I named my sample project ADC (in case you didnt guess, it stands for ActiveX Document Container). Other than selecting Container support in step 3 of the AppWizard, I did not change any settings. As is, this application provides support for regular in-place activated OLE items. To change this and add support for ActiveX document objects, we must change the CADCCntrItem class. Lets start with the class declaration (see Listing 30.1). We must make three changes here. First, we must declare an override for the Release member function (because, as you will see momentarily, we must release additional interfaces when this member function is invoked). Next, we must change the declaration of the GetActiveView member function. The new version will have a return type of LPOLEDOCUMENTVIEW and it will return a pointer to an IOleDocumentView interface, stored in a new member variable, m_pActiveView. Finally, in addition to declaring this member variable, we must provide a declaration for the IOleDocumentSite interface using standard MFC helper macros.

ActiveX Documents CHAPTER 30

563

Listing 30.1. The declaration of CADCCntrItem in cntritem.h.


class CADCDoc; class CADCView; class CADCCntrItem : public COleClientItem { DECLARE_SERIAL(CADCCntrItem) // Constructors public: CADCCntrItem(CADCDoc* pContainer = NULL); // Note: pContainer is allowed to be NULL to enable IMPLEMENT_SERIALIZE. // IMPLEMENT_SERIALIZE requires the class have a constructor with // zero arguments. Normally, OLE items are constructed with a // non-NULL document pointer. //Overridables public: virtual void Release(OLECLOSE dwCloseOption = OLECLOSE_NOSAVE); // Attributes public: CADCDoc* GetDocument() { return (CADCDoc*)COleClientItem::GetDocument(); } LPOLEDOCUMENTVIEW GetActiveView() { return m_pActiveView; } // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CADCCntrItem) public: virtual void OnChange(OLE_NOTIFICATION wNotification, DWORD dwParam); virtual void OnActivate(); protected: virtual void OnGetItemPosition(CRect& rPosition); virtual void OnDeactivateUI(BOOL bUndoable); virtual BOOL OnChangeItemPosition(const CRect& rectPos); //}}AFX_VIRTUAL // Implementation public: ~CADCCntrItem(); #ifdef _DEBUG virtual void AssertValid() const; virtual void Dump(CDumpContext& dc) const; #endif virtual void Serialize(CArchive& ar); protected: LPOLEDOCUMENTVIEW m_pActiveView; void Show(); BEGIN_INTERFACE_PART(OleDocumentSite, IOleDocumentSite) INIT_INTERFACE_PART(CADCCntrItem, OleDocumentSite) STDMETHOD(ActivateMe)(LPOLEDOCUMENTVIEW pViewToActivate); END_INTERFACE_PART(OleDocumentSite) DECLARE_INTERFACE_MAP() };

30
ACTIVEX DOCUMENTS

564

OLE, COM, and MFC Applications PART IV

The implementation of the modified container item class contains four changes, all shown in Listing 30.2 (the remainder of the class definition, which remains unchanged from the version created by the MFC AppWizard, is not listed here).

Listing 30.2. The first part of the implementation of CADCCntrItem in cntritem.cpp.


///////////////////////////////////////////////////////////////////////////// // CADCCntrItem implementation IMPLEMENT_SERIAL(CADCCntrItem, COleClientItem, 0) BEGIN_INTERFACE_MAP(CADCCntrItem, COleClientItem) INTERFACE_PART(CADCCntrItem, IID_IOleDocumentSite, OleDocumentSite) END_INTERFACE_MAP() CADCCntrItem::CADCCntrItem(CADCDoc* pContainer) : COleClientItem(pContainer) { // TODO: add one-time construction code here m_pActiveView = NULL; } CADCCntrItem::~CADCCntrItem() { // TODO: add cleanup code here } ///////////////////////////////////////////////////////////////////////////// // IOleDocumentSite interface STDMETHODIMP_(ULONG) CADCCntrItem::XOleDocumentSite::AddRef() { METHOD_PROLOGUE_EX(CADCCntrItem, OleDocumentSite) return pThis->ExternalAddRef(); } STDMETHODIMP_(ULONG) CADCCntrItem::XOleDocumentSite::Release() { METHOD_PROLOGUE_EX(CADCCntrItem, OleDocumentSite) return pThis->ExternalRelease(); } STDMETHODIMP CADCCntrItem::XOleDocumentSite::QueryInterface( REFIID iid, LPVOID* ppvObj) { METHOD_PROLOGUE_EX(CADCCntrItem, OleDocumentSite) return pThis->ExternalQueryInterface(&iid, ppvObj); } STDMETHODIMP CADCCntrItem::XOleDocumentSite::ActivateMe( LPOLEDOCUMENTVIEW pViewToActivate) { METHOD_PROLOGUE_EX(CADCCntrItem, OleDocumentSite) LPOLEDOCUMENT lpDocument; LPOLECLIENTSITE lpClientSite = pThis->GetClientSite(); LPOLEINPLACESITE lpInPlaceSite =

ActiveX Documents CHAPTER 30


(LPOLEINPLACESITE) pThis->GetInterface(&IID_IOleInPlaceSite); RECT rect; if (pViewToActivate == NULL) { if (pThis->m_pActiveView == NULL || pThis->m_pView == NULL) { pThis->m_lpObject->QueryInterface(IID_IOleDocument, (void **)&lpDocument); lpDocument->CreateView(lpInPlaceSite, NULL, 0, &pViewToActivate); lpDocument->Release(); } } else if (pThis->m_pActiveView != pViewToActivate) { pViewToActivate->SetInPlaceSite(lpInPlaceSite); pViewToActivate->AddRef(); } if (pViewToActivate != NULL && pThis->m_pActiveView != pViewToActivate) { if (pThis->m_pActiveView != NULL) { pThis->m_pActiveView->Show(FALSE); pThis->m_pActiveView->UIActivate(FALSE); pThis->m_pActiveView->Release(); } pThis->m_pActiveView = pViewToActivate; } pThis->m_pActiveView->UIActivate(TRUE); pThis->m_pView->GetClientRect(&rect); pThis->m_pActiveView->SetRect(&rect); pThis->m_pActiveView->Show(TRUE); return NOERROR; } void CADCCntrItem::Release(OLECLOSE dwCloseOption) { if (m_pActiveView) m_pActiveView->Release(); COleClientItem::Release(dwCloseOption); } ... void CADCCntrItem::OnActivate() { }

565

The first change is easy: using MFC macros, we must provide a definition for the new interface, IOleDocumentSite . The second change is even easier: the new member variable, m_pActiveView, must be initialized in the constructor. But now we must come to the real meat of it: the implementation of the IOleDocumentSite interface. It consists of four methods, the first three of which are standard implementations for the IUnknown interface methods QueryInterface, AddRef, and Release. However, the fourth

30
ACTIVEX DOCUMENTS

566

OLE, COM, and MFC Applications PART IV

method is anything but standard: it is the ActivateMe method that is really responsible for the document container behavior of our new application. The behavior of this method is different depending on whether its single parameter, pViewToActivate, is NULL. If it is, and we already have a pointer to an IOleDocumentView interface, we activate it; otherwise, we request the document object (using CreateView) to give us a pointer to such an interface first. If we were given an interface pointer through pViewToActivate, we call its SetInPlaceSite and AddRef methods. Next, if we have an old view that we must deactivate, we do so by calling its Show, UIActivate, and Release methods. Finally, we activate the new view, again by calling its UIActivate and Show methods. We set its size to the size of the client area in the current window. The last function in this block of changes is an override of the Release member function: now you can see why this override is necessary. In order to ensure that interfaces are properly released, we call the Release method of the active view, if any. The hard part is over. Only one change remains: we must remove the body of the OnActivate member function. In an ActiveX document container, more than one item can be in-place active at a time, therefore there is no need to deactivate an in-place item if the view is deactivated. Thats it! If you compile this application you will have a functional ActiveX document container in your hands, even if it has more than a few rough edges. First, the application does not support more than one item in a container. This is the standard behavior of an AppWizard-generated MFC container application, which we have not modified; however, in order for this application to have any practical value, that change would have to be made (along with accompanying design decisions about the user interface to handle multiple document objects). Second, we provide no support for printing and print preview of container documents. Third, we do not activate a view when a document is loaded. The ADCDoc::Serialize member function would need to be changed to ensure that after a container document is loaded, its first object is activated. Fourth, we do not filter types that are not ActiveX documents when the user selects the Insert New Object command from the Edit menu. Therefore, our application behaves like an odd hybrid, allowing both ActiveX document objects and ordinary in-place editable OLE objects to be inserted. This is really bad; a more functional implementation would override the standard behavior of the COleInsertDialog class to filter the kind of items that are displayed here (perhaps by scanning the Registry).

ActiveX Documents CHAPTER 30

567

This should also tell you that in order to test the ActiveX document container behavior of the new application, you should only select ActiveX document object types when using the Insert New Object command. Such document types include, for example, Microsoft Word documents and Microsoft Excel spreadsheets, which are certainly available on most Windows developers computers.

Creating an ActiveX Document Server


If you are using Visual C++, an ActiveX document server can be created with the help of the MFC AppWizard. To enable ActiveX document server support, you must select full container support in step three of the MFC AppWizard. You can then select ActiveX Document Server support (see Figure 30.2). Note that as soon as you select this option, the Finish button at the bottom of the AppWizard dialog becomes disabled. It will remain disabled until you specify a filename extension to be associated with your server.

FIGURE 30.2.
Enabling ActiveX document server support.

That filename extension can be specified in step four of the MFC AppWizard. You must click the Advanced button and enter the desired extension in the File Extension field (see Figure 30.3). Notice that the contents of all the other fields in this dialog are updated dynamically as you type the new extension.

30
ACTIVEX DOCUMENTS

568

OLE, COM, and MFC Applications PART IV

FIGURE 30.3.
Specifying a filename extension for the ActiveX document server.

In order to understand the AppWizard-generated server code, it is best to compare it against another project generated with identical parameters but with ActiveX document server support not enabled. If you do this comparison, you discover the following differences: s The InitInstance function in the main application class initializes the Registry differently. s The document class contains an implementation of the GetDocObjectServer member function. s The in-place frame class is now derived from COleDocIPFrameWnd instead of COleIPFrameWnd. s The server item class is now derived from CDocObjectServerItem instead of COleServerItem. Let us examine these differences in more detail. In order for an ActiveX document server to function correctly, several additional entries must be present in the Registry. These include keys such as HKEY_CLASSES_ROOT\My.Document\DocObject. Fortunately, there is no need to manually add or maintain any of these entries; it can be done automatically by calling the CWinApp::UpdateRegistry with the OAT_DOC_OBJECT_SERVER parameter. The document class of the ActiveX server application contains an implementation of the function. By returning a pointer to a valid CDocObjectServer object, the function indicates support for ActiveX documents.
GetDocObjectServer

ActiveX Documents CHAPTER 30

569

The parent class of the in-place frame is changed from COleIPFrameWnd to COleDocIPFrameWnd. This means several changes in the files ipframe.h and ipframe.cpp. Basically, every occurrence of COleIPFrameWnd is replaced with COleDocIPFrameWnd. The new parent class provides the necessary support for the ActiveX document item to operate within an ActiveX container application. The cooperation between OLE server applications and containers is provided through code implemented by the COleServerItem class. In the case of ActiveX document servers, another class, CDocObjectServerItem, must be used. This class, which is derived from COleServerItem, implements OLE server verbs that are specific to ActiveX document servers. Finally, there is a small change in stdafx.h. A new file, afxdocob.h, is listed among the header files used by the project.

Converting an Existing OLE Server to Support ActiveX Documents


Regardless of the level of class library support, converting a client application to support the ActiveX document architecture would normally involve a design change. Applications that are designed around embedded or linked objects are not likely to be able to work with ActiveX servers that take over the entire frame window. On the other hand, it is relatively easy to convert an OLE server application to also act as an ActiveX document server. Indeed, if the application is an MFC application, the conversion may be trivially simple. The previous section listed the differences between an OLE server application with and without ActiveX document server support. These differences can also serve as our checklist for converting an application. Here is a brief summary of what must be done. First, an ActiveX document server application must initialize the Registry differently. Whereas an OLE server calls CWinApp::UpdateRegistry with the OAT_INPLACE_SERVER parameter, an ActiveX document server must call this function with OAT_DOC_OBJECT_SERVER. The applications document class must also contain an implementation of the GetDocObjectServer function. In its trivial implementation, this function may look like the following:
CDocObjectServer *CMyDoc::GetDocObjectServer(LPOLEDOCUMENTSITE pDocSite) { return new CDocObjectServer(this, pDocSite); }

30
ACTIVEX DOCUMENTS

It is also necessary to change the class from which your applications in-place frame window is derived. By changing the parent class from COleIPFrameWnd to COleDocIPFrameWnd, you add the necessary support that enables your ActiveX document to work within container applications. Finally, your applications server item class, normally derived from COleServerItem, must be changed so that it is now derived from CDocObjectServerItem.

570

OLE, COM, and MFC Applications PART IV

Summary
ActiveX document technology extends the capabilities inherent in OLE embedding and enables items in a container full control over the containers frame window, menus, and toolbars. ActiveX document support is implemented through a series of new COM interfaces: IOleDocument and IOleDocumentView for servers, IOleDocumentSite for containers. These interfaces exist in addition to the standard set of interfaces for OLE in-place activation and editing. It is possible to modify an AppWizard-generated container application skeleton to provide ActiveX document container support. The key modification is a change to the applications COleClientItem-derived classsupport for the IOleDocumentSite interface must be added here. Other changes that might need to be implemented include support for multiple document objects in a container document, container document printing, activation after a container document is loaded, and a filtered version of the Insert New Object command that displays only ActiveX document object types. ActiveX server applications can be created with much greater ease through the MFC AppWizard. When compared with an OLE in-place server, an ActiveX document server contains different code for Registry initialization, and changes to its document, in-place frame, and server item classes. It is also possible to convert an existing in-place OLE server to support ActiveX document technology by implementing these changes manually.

Distributed COM CHAPTER 31

571

Distributed COM
IN THIS CHAPTER

31
DISTRIBUTED COM

31

s COM and DCOM: An Evolution s Explicit Coding Practices 575 s Beyond DCOM: COM+ 577

572

572

OLE, COM, and MFC Applications PART IV

Distributed COM, or DCOM, is one of the hottest recent buzzwords in Windows software development. It would be difficult indeed to convince a respectable publisher to release a book on Windows programming that does not discuss this subject. On the other hand, the author faces the difficult situation of having very little to write about. Why? Certainly not because DCOM isnt important! No, this time around we have Microsofts programmers to thank (and I do mean thank) for this situation: DCOM, it appears, delivers everything as promised, not the least important of which is its seamless compatibility with ordinary COM applications. Consequently, existing COM programs require very few if any changes in order to utilize this new technology.

COM and DCOM: An Evolution


According to literature published by Microsoft, the Component Object Model, or COM, was always intended to evolve beyond single-computer configurations. The underlying mechanism used for intertask communication is RPC, or Remote Procedure Calls, which itself is networkready. Of course this does not solve all the problems of invoking objects on a remote computer. The first such problem concerns finding the computer on which a DCOM service is located. Existing COM configuration entries in the Registry contain no information about such servers, so consequently, new configuration information is required. The second issue is that of security and client authentication. Obviously, it is highly undesirable to allow any client to make unrestricted use of DCOM services on a particular server. DCOM must provide a means to ensure that only authorized clients can connect and utilize DCOM services. A facility where DCOM access permissions can be provided must also be configured. Needless to say, applications that rely on the assumption that different COM components run on the same physical hardware might not be so readily adaptable.

Configuring the Client Workstation


In order for a client workstation to find a DCOM server for a specific application, the servers identity must be recorded somewhere. That somewhere, unsurprisingly, is the Registry. Several new Registry entries must be created that tell the client where on the network a server application is located. Consider an Automation server, for example the application developed in Chapter 23, OLE, ActiveX, and Component Object Model. In order to make this server visible to clients, several Registry entries had to be created, shown in Listing 31.1.

Distributed COM CHAPTER 31

573

Listing 31.1. Registry entries for the HELLO Automation server.


[HKEY_CLASSES_ROOT\HELLO.Application] @=HELLO Application [HKEY_CLASSES_ROOT\HELLO.Application\CLSID] @={FEB8C280-FD2D-11ce-87C3-00403321BFAC} [HKEY_CLASSES_ROOT\CLSID\{FEB8C280-FD2D-11ce-87C3-00403321BFAC}] @=HELLO Application [HKEY_CLASSES_ROOT\CLSID\{FEB8C280-FD2D-11ce-87C3-00403321BFAC}\LocalServer32] @=HELLO.EXE /Automation [HKEY_CLASSES_ROOT\CLSID\{FEB8C280-FD2D-11ce-87C3-00403321BFAC}\ProgId] @=HELLO.Application

31
DISTRIBUTED COM

When a client runs on a remote workstation, a different set of Registry entries is required. We no longer need the LocalServer32 entry. In fact, you must not have this entry; otherwise, a local copy of the server application, not a copy on a remote machine, will be invoked. What you need instead is two entries that provide the location of the remove server:
[HKEY_CLASSES_ROOT\CLSID\{FEB8C280-FD2D-11ce-87C3-00403321BFAC}] AppId={FEB8C280-FD2D-11ce-87C3-00403321BFAC} [HKEY_CLASSES_ROOT\AppId\{FEB8C280-FD2D-11ce-87C3-00403321BFAC}] RemoteServerName=remote.system.com

At first it might appear that this extra level of indirection is unnecessary. Why couldnt you have a RemoteServer entry under the applications CLSID key? The reason is simple: This extra indirection makes it possible for multiple applications to reference the same AppId entry. Thus, if the servers identity changes, its sufficient to modify just one Registry entry on the system, as opposed to one for every server application referenced. The name of the remote system, by the way, is a DNS name; that is, the symbolic Internet name of the remote machine. Note that all these new Registry entries can also be configured using DCOMCNFG.EXE, so manual editing of the Registry is rarely necessary.

Using DCOMCNFG.EXE
Setting up a client to connect to the right server is only one side of the story. The server must also be configured to accept connection requests from authorized clients. The tool that is used to configure DCOM server access is called DCOMCNFG.EXE. This application lets you set default DCOM security settings, and also enables you to configure individual server applications that are to be accessed by remote clients (see Figure 31.1).

574

OLE, COM, and MFC Applications PART IV

FIGURE 31.1.
The DCOMCNFG.EXE application.

In order to make your server application on Windows NT accessible across the network, you must change its DCOM configuration parameters. To do so, launch DCOMCNFG.EXE (from the Start menu or from an MS-DOS window), click the application name (HELLO Application) and click the Properties button. This displays a second dialog in which settings specific to the selected application can be reviewed or changed. Under the Location tab, shown in Figure 31.2, make sure that only the Run application on this computer setting is selected. (On a remote workstation, you would instead specify the name of the server on which the DCOM server application is to run.) Under the Security tab (see Figure 31.3), ensure that the Guest user has both access and launch permissions for this application. (Note that depending on your system configuration, you might need to enter different permissions here.) Finally, under the Identity tab select the interactive user option. This is necessary because this particular server displays a visible user interface, so it needs access to the desktop. Under Windows 95/98, several options of DCOMCNFG.EXE are not present: these include launch permissions and identity options. Note that under Windows 95/98, you can only use DCOMCNFG.EXE (and you can only run DCOM servers) if user-level access control is implemented. This can be done through the network configuration applet in the Control Panel. It is still possible to configure access to remote DCOM servers via the Registry when share-level access control is in use.

Distributed COM CHAPTER 31

575

FIGURE 31.2.
DCOMCNFG.EXE:

the

31
DISTRIBUTED COM

Location tab.

FIGURE 31.3.
DCOMCNFG.EXE:

the

Security tab.

Explicit Coding Practices


The ability to configure DCOM servers through the Registry makes it simple to install existing COM applications in a DCOM environment. Sometimes, however, it is desirable to access servers under programmatic control. As it turns out, this is also relatively easy to accomplish. Only a few modifications must be made to a COM program in order to access a server across the network. The main difference is that the call to CoCreateInstance, used to create an instance of the server object, must be replaced with CoCreateInstanceEx. This new function accepts a parameter that is a pointer to a COSERVERINFO structure, which in turn contains information about the remote server.

576

OLE, COM, and MFC Applications PART IV

Listing 31.2 shows the result. The program first presented in Chapter 23, used to access an Automation server, has been modified for remote access. The DNS (Internet) name of the remote server must be specified on the command line.

Listing 31.2. Automation client using DCOM.


#include <windows.h> #include <stdio.h> void main(int argc, char *argv[]) { CLSID clsid; LPUNKNOWN punk; LPDISPATCH pdisp; DISPID dispid, dispidNamed = DISPID_PROPERTYPUT; OLECHAR *pszProp = Ltext; OLECHAR *pszVal; int nSize; DISPPARAMS dispparams; UINT uArgErr; VARIANTARG vArg; COSERVERINFO srvinfo = {0, NULL, NULL, 0}; MULTI_QI mqi[] = {{&IID_IDispatch, NULL, 0}}; if (argc != 3) { printf(Usage: %s remote-system string-to-send\n, argv[0]); exit(1); } nSize = MultiByteToWideChar(CP_ACP, 0, argv[1], -1, NULL, 0); srvinfo.pwszName = new WCHAR[nSize]; MultiByteToWideChar(CP_ACP, 0, argv[1], -1, srvinfo.pwszName, nSize); nSize = MultiByteToWideChar(CP_ACP, 0, argv[2], -1, NULL, 0); pszVal = new WCHAR[nSize]; MultiByteToWideChar(CP_ACP, 0, argv[2], -1, pszVal, nSize); OleInitialize(NULL); CLSIDFromProgID(OLESTR(HELLO.Application), &clsid); CoCreateInstanceEx(clsid, NULL, CLSCTX_SERVER, &srvinfo, sizeof(mqi) / sizeof(mqi[0]), mqi); pdisp = (LPDISPATCH)(mqi[0].pItf); pdisp->GetIDsOfNames(IID_NULL, &pszProp, 1, LOCALE_SYSTEM_DEFAULT, &dispid); dispparams.cArgs = 1; dispparams.cNamedArgs = 1; dispparams.rgdispidNamedArgs = &dispidNamed; dispparams.rgvarg = &vArg; vArg.vt = VT_BSTR; vArg.bstrVal = SysAllocString(pszVal); pdisp->Invoke(dispid, IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_PROPERTYPUT, &dispparams, NULL, NULL, &uArgErr); SysFreeString(vArg.bstrVal); pdisp->Release();

Distributed COM CHAPTER 31


OleUninitialize(); delete pszVal; delete srvinfo.pwszName; }

577

31
DISTRIBUTED COM

This program can be compiled with the following command line:


cl -D_WIN32_DCOM HCL.CPP OLE32.LIB OLEAUT32.LIB

Alternatively, you may also define the symbol _WIN32_DCOM in the source code of your application, in which case the -D option on the command line is no longer necessary.

Beyond DCOM: COM+


COM and DCOM are among the most powerful features of the Win32 environment, yet they suffer from a significant drawback: they are very difficult to use. Although class and template libraries like MFC or ATL can hide much of the gory details, they are not always applicable. Hiding COMs internals is not always the same as simplifying their use. Microsofts promised answer to this problem is the next generation COM: COM+. The most significant aspect of this new technology is compiler integration. COM+ compatible compilers will provide syntax that allows the programmer to declare COM classes without having to implement common code, extract type library information from runtime modules, and integrate that information with common language operators. For example, in C++ it will be possible to declare a COM class using the coclass operator and let the compiler worry about implementing common functions like DllRegisterServer or the IUnknown interface. Although previews of COM+ have been released by Microsoft as early as September 1997, the technology is not yet incorporated into Visual C++. However, it might become available in the form of an interim update in the near future. When it is released, creation of COM applications will be dramatically simplified. Consider the code example presented in Listing 31.2. With COM+, this code would reduce to something similar to Listing 31.3.

Listing 31.3. What COM+ is expected to look like.


#include <windows.h> #include <stdio.h> #import hello.exe void main(int argc, char *argv[]) { CHello *pHello; pHello = new CHello; pHello->text = LHello from COM+!; }

578

OLE, COM, and MFC Applications PART IV

Quite an improvement, is it not? Server-side code will also become significantly easier to write. The downside is a compiler syntax that contains even more nonstandard extensions than before. However, despite the portability promise of DCOM, I would expect that the majority of C++ programmers writing COM+ code will be targeting the Win-32 operating platform exclusively, so perhaps this portability problem will be a nonissue after all.

Summary
DCOM extends the capabilities of Microsofts object architecture, COM, across the network. With DCOM, it becomes possible to create COM applications that consist of multiple components running on separate server hosts. The use of DCOM is simple. In fact, most COM applications can easily be adapted to work in a DCOM environment. All that must be done is to add the correct configuration entries to the Windows Registry. Much of this can be accomplished using the DCOM configuration tool, DCOMCNFG.EXE. It is also possible to specify DCOM servers programmatically. The key to this is the function CoCreateInstanceEx, which replaces CoCreateInstance and allows the programmer to specify the servers identity explicitly. Microsofts plans for the future of COM and DCOM have been announced. The new architecture, COM+, will provide integration and support at the compiler level, making the creation and use of COM and DCOM objects significantly easier.

IN THIS PART
s Database Programming Through ODBC 581 s Data Access Objects 607

s OLE DB and ADO 625 s Writing a Windows NT Service s MTS and the Three-Tiered Model 639 653

PART

Client/Server Solutions

Database Programming Through ODBC CHAPTER 32

581

32

Database Programming Through ODBC


32
DATABASE PROGRAMMING THROUGH ODBC

IN THIS CHAPTER
s ODBC in Action 582 s The SQL Standard and ODBC 591 s ODBC in MFC Applications 593

582

Client/Server Solutions PART V

ODBC, or Open Database Connectivity, represents a vendor-independent mechanism for accessing data in a variety of data sources. It is also the oldest member of a family of data access technologies that have since become known as MDAC: Microsoft Data Access Components. (Other MDAC technologies, such as OLE DB and ActiveX Data Objects, are covered in the subsequent chapters.) ODBC drivers are available for many different types of data sources. You can use ODBC to retrieve data from text files, dBase tables, Excel spreadsheets, SQL Server databases, and many other sources. Many ODBC drivers are redistributable. You can package your application for installation with the appropriate ODBC drivers and software for driver installation and management. At the heart of ODBC is its capability to execute Structured Query Language (SQL) statements against data sources. In addition to reviewing ODBC in this chapter, you also take a (very) brief look at SQL itself. If you plan to perform extensive development work using ODBC, I recommend a reference manual on SQL, such as C. J. Dates A Guide to the SQL Standard. The MFC Library provides extensive support for ODBC applications. A series of classes exists for encapsulating ODBC databases, tables, and records. The AppWizard supports the creation of ODBC applications, and further support for ODBC is provided by ClassWizard.

ODBC in Action
This section presents a review of some of the fundamental concepts of ODBC that you must cover before you can begin an attempt to create an ODBC application.

The ODBC Setup Applet


Invoked through the Control Panel or as a standalone application, the ODBC setup applet is used to register data sources. What exactly is a data source? That depends on the driver. In the case of a driver such as the SQL Server driver, the data source can be a database on a server. In the case of a driver such as the Microsoft Access or Microsoft Excel drivers, the database is a file (an MDB or XLS file). In the case of the Microsoft Text driver, the database is a disk directory that contains text files, which serve as tables in the database from the drivers perspective. The latest versions of ODBC distinguish between three types of data sources. A user data source is a data source visible only to its creator. A system data source is visible to all users on a given computer. A file data source is a data source whose specifications are stored in a file and can be shared between users on different computers. To add a data source, invoke the ODBC setup applet from the Control Panel, click the tab corresponding with the desired data source type, and select the Add button. In the resulting dialog (see Figure 32.1), pick a driver and click OK.

Database Programming Through ODBC CHAPTER 32

583

FIGURE 32.1.
The ODBC Create New Data Source dialog.

32
DATABASE PROGRAMMING THROUGH ODBC

Next, a driver-specific dialog is displayed (see Figure 32.2), where you can select the database and adjust the desired characteristics of the driver.

FIGURE 32.2.
ODBC Text Setup.

The ODBC setup applets main dialog (see Figure 32.3) lists all installed data sources. You can add or delete data sources, or you can modify the setup of existing data sources using this dialog.

ODBC API Concepts


Applications that use ODBC rely on ODBC drivers for data access. Drivers can be single-tier or multiple-tier. Single-tier drivers process ODBC calls and SQL statements. Multiple-tier

584

Client/Server Solutions PART V

drivers process ODBC calls and pass SQL statements to the data source (potentially a server residing elsewhere on the network).

FIGURE 32.3.
The ODBC setup applet.

The ODBC standard defines three conformance levels. The Core API includes those fundamental ODBC calls that are required to access a data source and execute SQL commands. The Level 1 API contains a set of additional calls used to retrieve information about data sources and the driver itself. The Level 2 API contains additional calls, such as calls that operate using parameter and result arrays. Because some drivers might not support Level 2 calls (although most support Level 1), it is important to know whether a particular command is available; ODBC references clearly mark each command with the API level to which it conforms. With respect to the SQL grammar, ODBC defines a core grammar and two variants: a minimum SQL grammar and an extended grammar. Note that ODBC is not equivalent to Embedded SQL. Embedded SQL uses SQL statements in source programs written in another language. Such a hybrid program is processed by a precompiler before it is passed to the compiler of the host programming language. In contrast, ODBC interprets SQL statements at runtime. The host program does not need to be recompiled to execute different SQL statements, nor is it necessary to compile separate versions of a host program for different data sources. An ODBC application must perform a series of steps to connect to a data source before it can execute SQL statements. These steps are illustrated in Figure 32.4. The first of the calls in Figure 32.4, SQLAllocEnv, allocates an ODBC environment. In effect, this call initializes the ODBC library and returns an environment handle of type SQLENVH.

Database Programming Through ODBC CHAPTER 32

585

FIGURE 32.4.
A typical set of ODBC calls.

SQLAllocEnv();

Allocates ODBC environment

SQLAllocConnect();

Allocates memory for connection

SQLConnect();

Loads driver, connects to source

SQLAllocStmt();

Allocates memory for SQL statement

// Execute ODBC statemtents SQLPrepare(); SQLExecute(); SQLBindCol(); SQLFetch(); ... SQLFreeStmt(); Frees statement memory

32
DATABASE PROGRAMMING THROUGH ODBC

SQLDisconnect();

Deallocates driver, disconnects from source

SQLFreeConnect();

Frees connection memory

SQLAllocEnv();

Frees environment, terminates session

The second call, SQLAllocConnect, allocates memory for a connection. The handle that is returned by this function, of type SQLHDBC, is used in subsequent ODBC function calls to refer to a specific connection. One application can maintain several open connections. The third call, SQLConnect, establishes a connection by loading the driver and connecting to the data source. This call has alternatives; for example, the SQLDriverConnect call can be utilized to connect to data sources that are not set up via the ODBC setup applet. Memory for a SQL statement is allocated through a call to SQLAllocStmt. By allocating memory for statements in a separate step, ODBC offers a mechanism for constructing, using, and reusing statements before the memory allocated for them is released.

NOTE
More recent versions of ODBC replace the calls to SQLAllocEnv, SQLAllocConnect, and SQLAllocStmt with a new function, SQLAllocHandle. This chapter continues to use the oldstyle allocation functions because they are compatible with all ODBC versions.

After these four calls, a typical ODBC application performs a series of calls to execute SQL statements against a database. It can use SQLPrepare to prepare (compile) an SQL statement for execution and SQLExecute to actually execute it. It can use a variety of calls to bind variables to statements and to retrieve the results of a statement.

586

Client/Server Solutions PART V

When its work is finished, the application should free the ODBC resources it has allocated. The statement handle is freed by calling SQLFreeStmt. The connection is terminated by calling SQLDisconnect; the memory allocated for the collection is released by a call to SQLFreeConnect. Finally, the ODBC environment is released by calling SQLFreeEnv.

A Simple ODBC Example


To put this into practice, I developed a very simple ODBC application that reads rows stored in an Excel spreadsheet. When an Excel spreadsheet is accessed using the Microsoft Excel ODBC driver, worksheets play the role of database tables, and rows in a worksheet play the role of records in a table. The spreadsheet is shown in Figure 32.5. It is a simple table of peoples last names, first names, and ages.

FIGURE 32.5.
A simple Excel spreadsheet to be accessed through ODBC.

Instead of installing this Excel spreadsheet as a data source through the ODBC setup applet, I opted to utilize the capabilities of the SQLDriverConnect function. This function enables you to connect to a data source even if it has not been previously installed through the ODBC setup applet. When, years ago, I created my first ODBC test programs, I believed that it was not possible to write console applications that utilize ODBC. This turned out to be only partially correct; my problems were not due to an inherent limitation of ODBC, they were caused by a faulty ODBC version that I received with a beta copy of Visual C++. Yet another object lesson about the use of beta software!

Database Programming Through ODBC CHAPTER 32

587

The application shown in Listing 32.1 is actually an ODBC console application. You can compile this program from the command line by typing cl ages.c odbc32.lib. Using this program requires the file ages.xls to be available in the current directory. (Needless to say, it also requires that the Excel ODBC driver be installed.)

Listing 32.1. Simple ODBC console application.


#include #include #include #include <windows.h> <sql.h> <sqlext.h> <string.h>

32
DATABASE PROGRAMMING THROUGH ODBC

#define CONNSTR DBQ=AGES.XLS;DRIVER={Microsoft Excel Driver (*.xls)} #define CONNLEN (sizeof(CONNSTR)-1) #define SQLTRY(x,y) \ { \ rc = y; \ if (rc != SQL_SUCCESS) \ { \ char szState[6]; \ char szMsg[255]; \ SDWORD sdwNative; \ SWORD swMsgLen; \ SQLError(hEnv, hDBC, hStmt, szState, &sdwNative, \ szMsg, sizeof(szMsg), &swMsgLen); \ printf(Error %d performing %s\nSQLState = %s\n \ SQL message = %s\n, rc, x, szState, szMsg); \ goto Terminate; \ } \ } void main(void) { SQLHENV hEnv = 0; SQLHDBC hDBC = 0; SQLHSTMT hStmt = 0; SQLCHAR szConnStr[255]; SQLCHAR szStmt[255]; SQLCHAR szFirstName[255]; SQLCHAR szLastName[255]; long nAge; SWORD cbConnStr; RETCODE rc; SDWORD sdwLNLen; SDWORD sdwFNLen; SDWORD sdwALen; int i; char szResult[1000]; SQLTRY(SQLAllocEnv, SQLAllocEnv(&hEnv)) SQLTRY(SQLAllocConnect, SQLAllocConnect(hEnv, &hDBC)) SQLTRY(SQLDriverConnect, SQLDriverConnect(hDBC, NULL, CONNSTR, CONNLEN, szConnStr, sizeof(szConnStr), &cbConnStr, SQL_DRIVER_NOPROMPT)) SQLTRY(SQLAllocStmt, SQLAllocStmt(hDBC, &hStmt))

continues

588

Client/Server Solutions PART V

Listing 32.1. continued


sprintf(szStmt, SELECT * FROM [Sheet1$]); SQLTRY(SQLPrepare, SQLPrepare(hStmt, szStmt, strlen(szStmt))) SQLTRY(SQLBindCol, SQLBindCol(hStmt, 1, SQL_C_CHAR, (PTR)szLastName, sizeof(szLastName), &sdwLNLen)) SQLTRY(SQLBindCol, SQLBindCol(hStmt, 2, SQL_C_CHAR, (PTR)szFirstName, sizeof(szFirstName), &sdwFNLen)) SQLTRY(SQLBindCol, SQLBindCol(hStmt, 3, SQL_C_SLONG, (PTR)&nAge, sizeof(nAge), &sdwALen)) SQLTRY(SQLExecute, SQLExecute(hStmt)) for (i = 1; (rc = SQLFetch(hStmt)) == SQL_SUCCESS; i++) { printf(Record #%d\tLast Name: %s\tFirst Name: %s\t Age: %d\n, i, szLastName, szFirstName, nAge); } if (rc != SQL_NO_DATA_FOUND) { SQLTRY(SQLFetch, rc) } printf(Successfully completed.\n); SQLCloseCursor(hStmt); Terminate: if (hStmt) SQLFreeStmt(hStmt, SQL_CLOSE); if (hDBC) SQLDisconnect(hDBC); if (hDBC) SQLFreeConnect(hDBC); if (hEnv) SQLFreeEnv(hEnv); }

Because so many things can go wrong during an ODBC call, I did not think I could get away with an example that has no error checking. Fortunately, ODBC calls use a uniform error reporting mechanism; thus, it was easy to create a simple macro, SQLTRY, and use that for simple error reporting. In a more sophisticated application you may utilize, for example, exception processing for this purpose (instead of that nasty goto). The rest is fairly simple. After the obligatory calls to SQLAllocEnv and SQLAllocConnect, the program calls SQLDriverConnect. This call enables it to open a table that has not been set up using the ODBC setup applet, and do it (optionally) without presenting a user interface. This is exactly what we are doing here; note the constants CONNSTR and CONNLEN that are used for this purpose. In CONNSTR, the drivers name must be specified exactly; otherwise, the call will fail. After the database (spreadsheet) has been successfully connected to, a single SQL statement is executed:
SELECT * FROM [Sheet1$]

The name Sheet1$ (enclosed in square brackets because it contains a character, $, not recognized by SQL) is the driver-supplied name for the first spreadsheet in an Excel workbook. The

Database Programming Through ODBC CHAPTER 32


SELECT SQL statement is used to retrieve a record or set of records; in its present form, it is used to simply retrieve all fields in all records.

589

The next three calls bind C variables to table columns. This is the purpose of the SQLBindCol function. When records are subsequently retrieved, field values are deposited into these variables. The records themselves are retrieved by SQLFetch and displayed, in a rather pedestrian fashion, using printf. SQLFetch is called repeatedly until its return value is something other than SQL_SUCCESS. A return value of SQL_NO_DATA_FOUND indicates that the last record has been retrieved; anything else is an error and treated accordingly. The program ends with the obligatory calls to SQLFreeStmt, SQLDisconnect, SQLFreeConnect, and SQLFreeEnv to free up resources and terminate the connection to the data source. If you run this program, it provides output similar to the following:
C:\>ages Record #1 Last Name: Record #2 Last Name: Record #3 Last Name: Record #4 Last Name: Successfully completed. C:\> Doe First Name: John Age: 29 Doe First Name: Jane Age: 26 Smith First Name: Jack Age: 33 Brown First Name: Joseph Age: 44

32
DATABASE PROGRAMMING THROUGH ODBC

Other ODBC Calls


The sample program in Listing 32.1 demonstrates some of the basic features of ODBC. Needless to say, there are many other ODBC function calls that applications can utilize. In addition to SQLConnect and SQLDriverConnect, the SQLBrowseConnect provides a third alternative for connecting to a data source. This function enables applications to iteratively browse data sources. Several connection options related to transaction processing, character set translation, timeouts, tracing, and other features can be set using SQLSetConnectOption. Current settings can be retrieved through SQLGetConnectOption. Information about drivers, data sources, and other options can be retrieved through a variety of functions, including SQLDataSources , SQLDrivers, SQLGetFunctions , SQLGetInfo, and SQLGetTypeInfo. Statement-level options can be specified by calling SQLSetStmtOption. As an alternative to calling SQLPrepare and SQLExecute, applications can utilize the SQLExecDirect function to execute SQL statements in a single step. The advantages of using SQLPrepare include the capability to execute a prepared statement more than once and to retrieve information about the result set before executing the statement.

590

Client/Server Solutions PART V

The drivers translated version of an SQL statement can be retrieved by calling SQLNativeSql. Some SQL statements require parameters. You can use SQLBindParameter to match variables in your program with question marks in an SQL statement. For example, you can use an SQL statement like this one:
INSERT INTO [Sheet1$] (LastName, FirstName, Age) VALUES (?, ?, ?)

Before executing this statement you can use three SQLBindParameter calls to match program variables to question marks in the SQL statement. This function is used in conjunction with SQLParamData and SQLPutData, which are used in response to an SQL_NEED_DATA return value from SQLExecute.
SQLParamOptions,

which is a Level 2 ODBC extension, enables an application to set multiple values. Another Level 2 extension, SQLExtendedFetch, can be used to return data on several rows in an array form. Information about a statements parameters can be retrieved by calling SQLDescribeParam and

SQLNumParams.

Information about a statements results can be obtained by calls to SQLNumResultCols, SQLColAttributes, and SQLDescribeCol. The SQLRowCount function returns the number of rows affected by an SQL UPDATE, INSERT, or DELETE operation. However, it is not guaranteed to return the number of rows in a result set, and few SQL drivers support that functionality. As an alternative to using SQLBindCol to bind columns, an application can rely on SQLGetData to retrieve data from unbound columns. ODBC supports positioning of SQL cursors. A Level 2 extension function, SQLSetPos, can be used to position the cursor to a specific row and to update, delete, or add data to the row set. Transaction processing is supported by the SQLTransact function. Information about a data source can be retrieved by calling the functions
SQLTables , SQLTablePrivileges, SQLColumns, SQLColumnPrivileges, SQLPrimaryKeys, SQLForeignKeys, SQLSpecialColumns, SQLStatistics, SQLProcedures, and SQLProcedureColumns. The information is returned by these functions as a result set, accessible by calling SQLBindCol and SQLFetch.

ODBC enables the asynchronous execution of functions. Asynchronous execution is enabled by calling SQLSetStmtOption or SQLSetConnectOption with SQL_ASYNC_ENABLE. When, afterward, a function that supports asynchronous execution is called, it returns immediately with the return value SQL_STILL_EXECUTING. Repeated calls to the same function (with parameters that must be valid but are ignored, except for the first, hStmt parameter) can be used to determine whether the functions execution has completed. Information about ODBC errors can be retrieved in a standardized form using SQLError.

Database Programming Through ODBC CHAPTER 32

591

The SQL Standard and ODBC


SQL, or Structured Query Language, is an official (ANSI) standard for relational database processing. Except for the simplest of cases, during most ODBC operations (including ODBC operations from within an MFC application) you will end up sending SQL statements to the driver; therefore, it is essential to have at least a basic familiarity with this language. This section presents a very brief overview of SQL, with special emphasis on the use of its statements in the context of ODBC. I hope this brief summary will be helpful in carrying out simple ODBC SQL operations in your applications without having to surround yourself with SQL reference works. At the heart of SQL are data manipulation statements and schema definition statements. Data manipulation statements retrieve, add, delete, or change data in a recordset (row set). Schema definition statements define the layout of a database.

32
DATABASE PROGRAMMING THROUGH ODBC

Data Manipulation Statements


There are four basic data manipulation statements: SELECT, INSERT, UPDATE, and DELETE.
SELECT operations have a general form of SELECT-FROM-WHERE. For example, a SELECT statement may look like the following: SELECT FirstName, LastName FROM EMPLOYEES WHERE EMPLOYEES.Age<30

Many other clauses and qualifiers can be used to refine a SELECT statement. One of the most distinguishing features of relational databases is the capability to perform join operations. Loosely speaking, join operations means combining two or more tables into a single result set. For example, consider the following statement:
SELECT EMPLOYEES.FirstName, EMPLOYEES.LastName, PLANS.Name FROM EMPLOYEES, PLANS WHERE EMPLOYEES.Age < PLANS.MaxAge

This statement operates on two tables, EMPLOYEES and PLANS; the former represents the employees of a corporation, the latter the set of benefit packages the corporation offers. This SELECT statement retrieves, for each employee, the name of the employee and the name of all the plans the employee qualifies for by age. Note that if the employee qualifies for more than one plan, his or her name will appear more than once in the result set. If you want to use a SELECT statement to retrieve all the fields in a row, you can use a single asterisk as a shorthand. For example, for an EMPLOYEES table that has three fields, FirstName, LastName, and Age, the following two statements are equivalent:
SELECT FirstName, LastName, Age FROM EMPLOYEE SELECT * FROM EMPLOYEE

592

Client/Server Solutions PART V

SQL also offers a series of aggregate functions. These functions include COUNT, MIN, MAX, AVG, and SUM. For example, to count in the EMPLOYEES table the number of employees whose last names are distinct, you would use the following statement:
SELECT COUNT (DISTINCT EMPLOYEES.LastName) FROM EMPLOYEES

Or, to calculate the combined life experience of the corporations work force, you would issue the following statement:
SELECT SUM (EMPLOYEES.AGE) FROM EMPLOYEES

Obviously, many forms of the SELECT statement operate on multiple rows and return row sets as results. The SQL standard defines the concept of a cursor that is used to iteratively fetch the rows from a result set. The ODBC SQLBindCol and SQLFetch functions are based on the same principle. The INSERT statement is used to add rows to a table. The UPDATE statement is used to modify existing rows. The DELETE statement is used to remove rows. The syntax of these commands is similar to the syntax of the SELECT command. In particular, these commands have cursor-based and noncursor variants. For example, consider the following command that you would execute on December 31 every year to update the corporate employee database (naturally, nobody ages after 30):
UPDATE EMPLOYEES SET EMPLOYEES.AGE = EMPLOYEES.AGE + 1 WHERE EMPLOYEES.AGE < 30

This searched update operates on all rows specified by the WHERE clause and does not require a cursor to execute. Other forms of these statements are cursor based; ODBC supports such operations via SQLBindParameter and related functions.

Views
A view, loosely speaking, is a kind of virtual table. Not backed by physical storage, a view represents a row set created dynamically using the CREATE VIEW statement. A view can be created with the help of a SELECT statement. For example, to create a view that contains all employees younger than 30, you would use the following statement:
CREATE VIEW YOUNGEMPLOYEES (LastName, FirstName, Age) AS SELECT EMPLOYEES.LastName, EMPLOYEES.FirstName, EMPLOYEES.Age FROM EMPLOYEES WHERE EMPLOYEES.Age < 30

In subsequent operations you can use this view just like you would use any other table. For example, you can use the following SELECT statement:
SELECT * FROM YOUNGEMPLOYEES

Database Programming Through ODBC CHAPTER 32

593

Data Definition Statements


Data definition statements are used to create and update tables and indexes in a database. The CREATE TABLE statement can be used to create a table, of course. To create the EMPLOYEES table that we used in the preceding sections, you could use the following statement:
CREATE TABLE EMPLOYEES ( LastName CHAR(30) NOT NULL, FirstName CHAR(30), Age INTEGER )

The CREATE TABLE statement supports constraint clauses. These include UNIQUE clauses (specifying that a fields value must be unique) and CHECK clauses (specifying a condition). For example, our EMPLOYEES table definition might look like this:
CREATE TABLE EMPLOYEES ( LastName CHAR(30) NOT NULL, FirstName CHAR(30) NOT NULL, Age INTEGER, UNIQUE (LastName, FirstName), CHECK (Age < 30) )

32
DATABASE PROGRAMMING THROUGH ODBC

Tables can also be created with indexes. For example, to create an EMPLOYEES table indexed by last name, use the following syntax:
CREATE TABLE EMPLOYEES ( LastName CHAR(30) NOT NULL, FirstName CHAR(30), Age INTEGER, PRIMARY KEY (LastName) )

The CREATE The ALTER

INDEX

statement can be used to create an index on an existing table.

TABLE

statement can be used to modify the structure of an existing table.

The DROP statement can be used to delete an existing table or index from the database. Finally, the GRANT and REVOKE commands can be used to grant and revoke security privileges on specific tables.

ODBC in MFC Applications


The use of ODBC is greatly simplified by the Microsoft Foundation Classes Library. Simple applications that access tables through ODBC can be created with only a few mouse clicks using the AppWizard and ClassWizard. Several MFC classes exist that support accessing databases and recordsets.

594

Client/Server Solutions PART V

Our discussion of ODBC-related features in the MFC Library starts with the construction of a simple example.

Setting Up a Data Source


Before an MFC ODBC application can be constructed using AppWizard, it is necessary to identify a data source on which the application will operate. The data source must be identified and set up through the ODBC setup applet. The data source used in our sample application is a text file. To access this file, we need the Microsoft Text ODBC driver. (If you did not install this driver when you set up Visual C++, rerun the Visual C++ setup program.) The data file, ages.txt, will contain a set of records with first names, last names, and ages. The first row in the file will be used as a header row. The file will be a comma-separated file with the following contents:
LastName,FirstName,Age Doe,John,29 Doe,Jane,26 Smith,Joe,44 Brown,Joseph,27

After creating this file, we must identify the data source through the 32-bit ODBC setup applet. Invoke this applet and click the Add button; select the Microsoft Text Driver in the dialog shown in Figure 32.6.

FIGURE 32.6.
Adding a text data source.

Clicking this dialogs Finish button invokes the ODBC Text Setup dialog (see Figure 32.7), which is a dialog specific to the selected driver. The Microsoft Text driver views disk directories as databases and individual text files as tables in the database. The driver can be set up to use either the current directory or a specific directory as the data source.

Database Programming Through ODBC CHAPTER 32

595

FIGURE 32.7.
ODBC Text Setup.

32
DATABASE PROGRAMMING THROUGH ODBC

If you select a specific directory, the driver enables, through the Options extension of its dialog, setting up individual tables (text files). For example, I specified C:\AMFC as the directory where the new application will be placed and created ages.txt in that directory. After specifying this directory name by clicking the Select Directory button, the Define Format button became active in the ODBC Text Setup dialog (see Figure 32.8).

FIGURE 32.8.
ODBC Text Setup options.

Clicking the Define Format button brings up yet another dialog (see Figure 32.9) where the format of individual tables (text files) can be specified. In the case of the ages.txt table, setting the Column Name Header check box enables the Guess button to work correctly and retrieve the names of fields and correctly guess their type.

596

Client/Server Solutions PART V

FIGURE 32.9.
Defining the format of a text table.

Dismiss this dialog by clicking OK. When the ODBC Text Setup dialog reappears, add a name to this data source. I decided to name this data source CSV Files In AMFC. Dismiss this dialog, too, by clicking OK, and dismiss the ODBC Data Source Administrator dialog by clicking OK. At this point, a look at the amfc directory where the ages.txt file resides reveals that the ODBC setup applet created another file, one named schema.ini. This file, shown in Listing 32.2, contains information on the characteristics of the ODBC data source that we just specified.

Listing 32.2. The schema.ini file created by the ODBC setup applet.
[ages.txt] ColNameHeader=True Format=CSVDelimited MaxScanRows=25 CharacterSet=OEM Col1=LASTNAME Char Width 255 Col2=FIRSTNAME Char Width 255 Col3=AGE Integer

Now that our data source has been set up and identified, we can turn to the AppWizard to construct a skeleton for our application.

Creating an ODBC Application Skeleton Through AppWizard


To begin creating the ODBC skeleton application, fire up AppWizard and create a project named AMFC. The project should be single-documentbased (AppWizard Step 1). Database options are specified in AppWizard Step 2 (see Figure 32.10), where you should specify the Database View Without File Support option.

Database Programming Through ODBC CHAPTER 32

597

FIGURE 32.10.
Specifying database support in AppWizard.

32
DATABASE PROGRAMMING THROUGH ODBC

After you have specified this option, you must click the Data Source button and define a data source for this application before proceeding. Specify the recently created data source, CSV Files In AMFC, as the ODBC data source in the Database Options dialog (see Figure 32.11).

FIGURE 32.11.
Specifying a data source.

You can also specify the recordset type. A snapshot-type recordset represents a static view of the underlying data. A dynaset-type recordset only captures an index of the underlying data; these recordsets, therefore, can reflect changes made to records after the recordset was created (but not any record additions or removals). A table-like recordset allows the most direct manipulation of tables in a relational database. Although snapshots represent the safest choice, dynasets also offer a performance advantage and should be considered. Our test application will be built with a dynaset-type recordset, so select this one in the Database Options dialog.

598

Client/Server Solutions PART V

When you click OK in the Database Options dialog, AppWizard responds by showing yet another dialog (see Figure 32.12) where you can select database tables. Select the ages.txt file as the database table and click OK.

FIGURE 32.12.
Specifying a table.

Clicking OK returns you to the AppWizard main dialog and enables you to proceed from Step 2. For our test application, we do not need to change any other options, so you might as well proceed by clicking the Finish button. The AMFC test application will be created by AppWizard at this time. Take a look at the classes created by AppWizard (see Figure 32.13). When compared with applications that have no database support, you might notice a new class and a few new member variables in the applications document and view classes. The new class, CAMFCSet, is a class derived from CRecordset. Looking at this classs declaration (shown in Listing 32.3), we can see that AppWizard not only created the class, it also added member variables that reflect the fields of the database table that we specified.

Listing 32.3. Declaration of CAMFCSet.


class CAMFCSet : public CRecordset { public: CAMFCSet(CDatabase* pDatabase = NULL); DECLARE_DYNAMIC(CAMFCSet) // Field/Param Data //{{AFX_FIELD(CAMFCSet, CRecordset) CString m_LastName; CString m_FirstName;

Database Programming Through ODBC CHAPTER 32


long m_Age; //}}AFX_FIELD // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CAMFCSet) public: virtual CString GetDefaultConnect(); // Default connection string virtual CString GetDefaultSQL(); // default SQL for Recordset virtual void DoFieldExchange(CFieldExchange* pFX); // RFX support //}}AFX_VIRTUAL // Implementation #ifdef _DEBUG virtual void AssertValid() const; virtual void Dump(CDumpContext& dc) const; #endif };

599

32
DATABASE PROGRAMMING THROUGH ODBC

FIGURE 32.13.
ODBC application classes.

These member variables are also reflected in the classs implementation file (see Listing 32.4), in the constructor function and also in the function DoFieldExchange. The latter is called by the MFC framework to exchange data between the recordsets member variables and corresponding columns in the database table.

600

Client/Server Solutions PART V

Listing 32.4. Implementation of CAMFCSet.


IMPLEMENT_DYNAMIC(CAMFCSet, CRecordset) CAMFCSet::CAMFCSet(CDatabase* pdb) : CRecordset(pdb) { //{{AFX_FIELD_INIT(CAMFCSet) m_LastName = _T(); m_FirstName = _T(); m_Age = 0; m_nFields = 3; //}}AFX_FIELD_INIT m_nDefaultType = dynaset; } CString CAMFCSet::GetDefaultConnect() { return _T(ODBC;DSN=CSV files in AMFC); } CString CAMFCSet::GetDefaultSQL() { return _T([AGES].[TXT]); } void CAMFCSet::DoFieldExchange(CFieldExchange* pFX) { //{{AFX_FIELD_MAP(CAMFCSet) pFX->SetFieldType(CFieldExchange::outputColumn); RFX_Text(pFX, _T([LastName]), m_LastName); RFX_Text(pFX, _T([FirstName]), m_FirstName); RFX_Long(pFX, _T([Age]), m_Age); //}}AFX_FIELD_MAP }

Before we proceed, let me call your attention to a subtle yet deadly bug present in many Visual C++ versions, including version 5. Notice that the GetDefault member function returns the string value [AGES].[TXT]. This, unfortunately, is wrong, and will result in an SQL syntax error if you attempt to run the application. The correct string should be [AGES.TXT]; you must change the implementation of this function to reflect the correct value. The changes in the applications document and view classes are minor. The document class acquired a new member variable, m_aMFCSet, which is of type CAMFCSet and, rather unsurprisingly, represents the table that the applications document is associated with. The view class also acquired a member variable, m_pSet, which is set by default to point to the document classs m_aMFCSet member. The view class also has a new function, OnGetRecordset; the default implementation simply returns the value of m_pSet. Although the skeleton application can be built at this time, it is not very useful in its present state. As shown in Figure 32.14, it merely displays a blank dialog; and although the record selection commands work, their only visible effect is the enabling and disabling of command

Database Programming Through ODBC CHAPTER 32

601

items and buttons as one end or the other of the table is reached. Clearly, we must modify the applications dialog before the application is of any practical use.

FIGURE 32.14.
The skeleton application in action.

32
DATABASE PROGRAMMING THROUGH ODBC

Customizing the ODBC Application


As it turns out, customizing our ODBC application is laughably simple. In fact, the customization that enables us to browse records in our table does not require adding a single line of code by hand. All that is required is the addition of controls to the applications main dialog and the use of ClassWizard to add the appropriate member variables. To begin, open the IDD_AMFC_FORM dialog for editing. Remove the static TODO control, then add three static controls and three edit controls as shown in Figure 32.15. Name the edit controls IDC_LASTNAME, IDC_FIRSTNAME, and IDC_AGE, respectively.

FIGURE 32.15.
Adding controls to AMFCs main dialog.

Now comes the tricky part. In order to have ClassWizard assign member variables, hold down the Control key and double-click one of the edit fields. The result is a ClassWizard Add Member Variable dialog that is already filled with values that represent ClassWizards guess as to the

602

Client/Server Solutions PART V

proper recordset member variable (see Figure 32.16). The ClassWizard guesses the proper variable name by looking at the static fields in the dialog.

FIGURE 32.16.
Assigning recordset member variables to dialog fields.

Repeat this action for the other two dialog fields. When youre finished, recompile the application. Surprise! This was all that needed to be done to turn AMFC into a functional application. In its present form it is a functional browser of records in the ages.txt file (see Figure 32.17).

FIGURE 32.17.
The AMFC application in action.

Needless to say, as a simple browser AMFC barely scratches the surface of the ODBC capabilities of MFC. Before you conclude this chapter youll take a look at what else is supported by the ODBC classes in the MFC Library.

ODBC Classes in MFC


The set of classes that the MFC Library offers in support of ODBC applications is shown in Figure 32.18. Among these classes, the two important ones are CDatabase and CRecordset.

Database Programming Through ODBC CHAPTER 32

603

FIGURE 32.18.
ODBC support classes in MFC.

CDatabase

CRecordset

CLongBinary

CFieldExchange

32
DATABASE PROGRAMMING THROUGH ODBC
CDBException

The CDatabase class represents a connection to a data source. Its member variable m_hdbc represents an ODBC connection handle. The member functions Open and Close can be used to establish a connection to the data source or to terminate the connection. Other member functions are used to set or retrieve connection settings. These functions include GetConnect (returns the ODBC connection string), IsOpen, GetDatabaseName, CanUpdate, CanTransact, InWaitForDataSource, SetLoginTimeout, SetQueryTimeout, and SetSynchronousMode. By default, the CDatabase class uses asynchronous mode for accessing the data source. An asynchronous operation that is in progress can be canceled by calling the Cancel member function. Transaction processing is supported through the member functions BeginTrans, CommitTrans, and Rollback. The CDatabase class also offers two overridable functions. OnSetOptions is used to set standard connection options. OnWaitForDataSource is called by the framework to yield processing time while performing a lengthy operation. The ExecuteSQL member function can be used to directly execute an SQL statement. This statement cant be used in conjunction with SQL statements that return data records. The CRecordset class encapsulates the functionality of an ODBC SQL statement and the row set returned by the statement. Member variables of this class identify the ODBC statement handle, the number of fields and parameters in the recordset, the CDatabase object through which this recordset is connected to the data source, and two strings that correspond to SQL WHERE and ORDER BY clauses. The two principal types of recordsets are dynasets and snapshots. The type of a recordset is specified when calling the CRecordset::Open member function. Snapshots represent a static view of the data as it existed at the time the snapshot was created. This is most useful for tasks such as report generation. Dynasets present a dynamic view of the data, reflecting changes to it made by other users or through other recordsets in your application.

604

Client/Server Solutions PART V

When the recordset is opened through its Open member function, the table is accessed and the query that the recordset represents is performed. The recordset and the associated statement handle can be closed by calling the Close member function. Attributes of the recordset can be retrieved by calling the member functions CanAppend , CanRestart, CanScroll, CanTransact, CanUpdate, GetRecordCount, GetStatus, GetTableName, GetSQL, IsOpen, IsBOF, IsEOF, and IsDeleted. The recordset can be navigated through the functions Move, MoveFirst, MoveLast, and MovePrev.
MoveNext,

Operations on the recordset can be carried out by calling AddNew, Delete, Edit, or Update. Other recordset functions carry out miscellaneous housekeeping functions. You never use an object of type CRecordset directly. Instead, you should derive a class from
CRecordset and add member variables that correspond to the fields (columns) of the table that

the recordset represents. Next, override the recordsets DoFieldExchange member function; this function should facilitate the exchange of data between member variables and fields in the database through RFX_ functions. These functions, similar in syntax and concept to the dialog data exchange (DDX_) functions, are summarized in Table 32.1.

Table 32.1. RFX_ functions. Function name Field type


RFX_Binary RFX_Bool RFX_Byte RFX_Date RFX_Double RFX_Int RFX_Long RFX_LongBinary RFX_Single RFX_Text CByteArray BOOL BYTE CTime double int LONG CLongBinary float CString

ODBC SQL type


SQL_BINARY, SQL_LONGVARBINARY, SQL_VARBINARY SQL_BIT SQL_TINYINT SQL_DATE, SQL_TIME, SQL_TIMESTAMP SQL_DOUBLE SQL_SMALLINT SQL_INTEGER SQL_LONGVARCHAR SQL_REAL SQL_CHAR, SQL_DECIMAL, SQL_LONGVARCHAR, SQL_NUMERIC, SQL_VARCHAR

Field exchange is facilitated through the CFieldExchange class. An object of this class contains information about the field that is being exchanged when the recordsets DoFieldExchange member function is called.

Database Programming Through ODBC CHAPTER 32

605

The CRecordView class is a view class derived from CFormView that is designed specifically to display database records in forms. Objects of type CRecordView utilize dialog data exchange (DDX) and record field exchange (RFX) functions to facilitate the movement of data between the form and the data source. CRecordView-derived objects are used in conjunction with CRecordset-derived objects. ODBC operations utilize the CDBException class for reporting errors via the MFC exception mechanism.

Summary
ODBC is a powerful, SQL-based, vendor-independent mechanism for accessing data in various data sources. At the heart of ODBC are ODBC drivers, which are often redistributable DLLs that implement access to data sources of various types. Single-tier drivers implement both the connection to the data source and the processing SQL statements. Multiple-tier drivers connect to data sources and pass on the SQL statements. ODBC data sources can be local files (for example, text files, dBase files, Excel files) and remote data servers (for example, SQL Server, Oracle). Data sources are usually specified through the ODBC setup applet (invoked through the Control Panel), although the SQLDriverConnect call makes it possible to connect to a data source that has not been set up this way. An ODBC session involves calls that build up a connection to the data source, construct and submit SQL statements, and process the results. The ODBC API defines a series of function calls that facilitate these sessions. The API defines a set of conformance levels (Core, Level 1, and Level 2); most drivers support at least Level 1 ODBC functions. ODBC supports a variation of the standard SQL syntax. Data manipulation statements such as SELECT, INSERT, UPDATE, and DELETEas well as data definition statements such as CREATE TABLE, DROP TABLE, CREATE INDEX, DROP INDEX, and ALTER TABLEare supported. ODBC also supports the CREATE VIEW SQL statement. The Microsoft Foundation Classes Library provides two classes for ODBC support. The class represents an ODBC connection; the CRecordset class represents an ODBC SQL statement and the row set the statement returns. Applications typically derive a class from CRecordset and add member variables corresponding to table columns. The CRecordset class offers member functions that facilitate browsing and editing the row set.
CDatabase

32
DATABASE PROGRAMMING THROUGH ODBC

Data Access Objects CHAPTER 33

607

Data Access Objects

33

IN THIS CHAPTER
s DAO Overview 608 s Building a DAO Application 608 s DAO Classes 619

33
DATA ACCESS OBJECTS

608

Client/Server Solutions PART V

Data Access Objects, or DAOs, are one of Microsofts latest inventions in database access technology. This technology is used for database access in Microsofts Visual Basic, Microsoft Access, and Visual Basic for Applications. Starting with MFC 4, with the help of a set of specialized MFC classes, it is also available to the C++ programmer. DAO is supplied in the form of redistributable components. For example, redistributable DAO files can be found in the DevStudio\VC\redist\dao directory on the Visual C++ CD-ROM. You may either utilize the setup program provided in this directory or incorporate the DAO components found here into your applications setup utility.

DAO Overview
Data Access Objects enable you to access and manipulate databases through the Microsoft Jet database engine. Through this engine, you can access data in Microsoft Access database files (MDB files). The technology also enables you to access local and remote databases through ODBC drivers. Data Access Object technology is based on OLE. Figure 33.1 depicts the hierarchy of Data Access Objects. This hierarchy is greatly simplified by the DAO classes in MFC. Many DAO functions utilize Structured Query Language (SQL) statements. You can use the SQL SELECT statement to retrieve data from a database, or the SQL UPDATE, INSERT, and DELETE statements to modify the contents of the database. An easy way to create SQL statements for use with DAO objects is to create the query from within Microsoft Access, save the query in the database, and access the query through a QueryDef object. Visual C++ provides extensive support for building DAO applications through the AppWizard. In addition to ODBC, AppWizard enables you to create applications that are based on DAO Classes. Our tour of DAO begins with the creation of a simple DAO application and exploration of its behavior.

Building a DAO Application


Building a DAO application is quite simple. First, if it does not exist yet, you must create a data source. For the application demonstrated here, the data source is a simple Access database of two tables. Next, the skeleton application must be created using AppWizard; and finally, you must customize this application as appropriate. The application is a simple browser; it browses a row set that is created as a relational join of two tables.

Data Access Objects CHAPTER 33

609

FIGURE 33.1.
The DAO object hierarchy.

DBEngine Workspace Database TableDef Field Index Field QueryDef Field Parameter Recordset Field Relation Field

33
DATA ACCESS OBJECTS

Container Field User Group Group User Error

The Database
The database used in this example contains two tables. One table contains the first names, last names, and ages of employees; the other table contains the names of benefit packages offered to employees and the maximum qualifying age for each package. The purpose of our application, which I decided to call ADAO, is simple: display for each employee all benefit packages for which he or she qualifies. The database, adao.mdb, is constructed using Microsoft Access. To construct the database, start Access and create a blank database named adao.mdb in the directory of your choice. Upon successful creation, select the Tables tab in the Database window, and click the New button.

610

Client/Server Solutions PART V

Select Design View. (Note that the database file used to create the sample code in this chapter was itself created using Access 95; if you are using a later version of Access, your database file, and hence the code generated by AppWizard, might differ slightly from what is shown here. The Access user interface might also differ slightly from what is presented here in the form of screen shots and textual descriptions.) Figure 33.2 shows the newly constructed Employees table just before it is saved. As you can see, three fields were added (LastName, FirstName, and Age). The first two are 50-characterwide text fields and the third is a number field. The tables primary key was set to the combination of the LastName and FirstName fields.

FIGURE 33.2.
Creating the Employees table.

Save the new table under the name Employees and repeat this procedure to create a second table (see Figure 33.3). This table contains information about benefit packages. It contains two fields, the first of which, Name, serves as the tables primary key. This table should be saved under the name Plans. When your work creating these tables is finished, the Database window should show two tables, as seen in Figure 33.4. The next step is to add data to these tables. You can do so by simply double-clicking the tables name in the Database window. Figure 33.5 shows the four records I added to Employees. Figure 33.6 shows the three records I added to the Plans table.

Data Access Objects CHAPTER 33

611

FIGURE 33.3.
Creating the Plans table.

FIGURE 33.4.
Tables in adao.mdb.

33
DATA ACCESS OBJECTS

FIGURE 33.5.
Records in the Employees table.

FIGURE 33.6.
Records in the Plans table.

612

Client/Server Solutions PART V

This is all you must do with Microsoft Access. Our MDB file is now ready for use from within a C++ DAO application.

Creating the Skeleton Application


To create the ADAO application, launch the AppWizard and start creating a single-document based project. Database support is specified in AppWizard Step 2; select the Database View Without File Support option (see Figure 33.7).

FIGURE 33.7.
Adding database support to a skeleton application.

Before you can proceed from this step, you must specify a data source. To do so, click the Data Source button. In the Database Options dialog that appears, select DAO as the data source (see Figure 33.8).

FIGURE 33.8.
Adding a DAO data source.

Data Access Objects CHAPTER 33

613

Clicking the ellipsis button next to the DAO field enables you to specify the actual database file. It brings up a standard File Open dialog in which you can select the file adao.mdb. Select this file and when the Database Options dialog reappears, click OK. This should display another dialog where you can select the tables of the database. Select both the Employees and the Plans tables and click OK (see Figure 33.9).

FIGURE 33.9.
Selecting tables.

33
DATA ACCESS OBJECTS

At this time, all dialogs should disappear except for the AppWizard Step 2 dialog; this dialog should now display our data source selection. All other AppWizard settings can remain at their default values; therefore, you can quickly complete creating the skeleton application by clicking the Finish button.

Exploring the DAO Application Skeleton


The classes of the skeleton application created by AppWizard are shown in Figure 33.10. When compared to an application with no database support, this application offers one extra class and a few additional member variables and functions in its document and view classes. Not evidently visible, but also a notable difference, is the fact that the view class is derived from CDaoRecordView. If you are experienced with MFC ODBC programming, you might note that the structure of this application is very similar to that of ODBC applications created by AppWizard. The new class, CADAOSet, is derived from CDaoRecordset and represents the row set that we will select from the join of the Employees and Plans table. Looking at this classs declaration in Listing 33.1, you can see that the AppWizard already inserted member variables that correspond to the columns (fields) in the two tables.

614

Client/Server Solutions PART V

FIGURE 33.10.
Skeleton application classes.

Listing 33.1. CDAOSet class declaration.


class CADAOSet : public CDaoRecordset { public: CADAOSet(CDaoDatabase* pDatabase = NULL); DECLARE_DYNAMIC(CADAOSet) // Field/Param Data //{{AFX_FIELD(CADAOSet, CDaoRecordset) CString m_LastName; CString m_FirstName; long m_Age; CString m_Name; long m_MaxAge; //}}AFX_FIELD // Overrides // ClassWizard generated virtual function overrides //{{AFX_VIRTUAL(CADAOSet) public: virtual CString GetDefaultDBName(); virtual CString GetDefaultSQL(); virtual void DoFieldExchange(CDaoFieldExchange* pFX); //}}AFX_VIRTUAL // Implementation #ifdef _DEBUG virtual void AssertValid() const; virtual void Dump(CDumpContext& dc) const; #endif };

Data Access Objects CHAPTER 33

615

A look at the implementation of CADAOSet in Listing 33.2 shows how these member variables are initialized in the classs constructor. The variables are also referred to in the AppWizardgenerated implementation of the DoFieldExchange member function. This function exchanges data between member variables in the class and fields in the database.

Listing 33.2. CADAOSet class implementation.


IMPLEMENT_DYNAMIC(CADAOSet, CDaoRecordset) CADAOSet::CADAOSet(CDaoDatabase* pdb) : CDaoRecordset(pdb) { //{{AFX_FIELD_INIT(CADAOSet) m_LastName = _T(); m_FirstName = _T(); m_Age = 0; m_Name = _T(); m_MaxAge = 0; m_nFields = 5; //}}AFX_FIELD_INIT m_nDefaultType = dbOpenDynaset; } CString CADAOSet::GetDefaultDBName() { return _T(C:\\ADAO\\adao.mdb); }

33
DATA ACCESS OBJECTS

CString CADAOSet::GetDefaultSQL() { return _T([Employees],[Plans]); } void CADAOSet::DoFieldExchange(CDaoFieldExchange* pFX) { //{{AFX_FIELD_MAP(CADAOSet) pFX->SetFieldType(CDaoFieldExchange::outputColumn); DFX_Text(pFX, _T([LastName]), m_LastName); DFX_Text(pFX, _T([FirstName]), m_FirstName); DFX_Long(pFX, _T([Age]), m_Age); DFX_Text(pFX, _T([Name]), m_Name); DFX_Long(pFX, _T([MaxAge]), m_MaxAge); //}}AFX_FIELD_MAP }

To do its work, DoFieldExchange uses DFX_ functions. These functions are the DAO cousins of the RFX_ functions used for ODBC field exchange. The set of DFX_ functions available for use in DoFieldExchange is summarized in Table 33.1.

616

Client/Server Solutions PART V

Table 33.1. DFX_ functions. Function Name


DFX_Binary DFX_Bool DFX_Byte DFX_Currency DFX_DateTime DFX_Double DFX_Long DFX_LongBinary DFX_Short DFX_Single DFX_Text

Field Type
CByteArray BOOL BYTE COleCurrency COleDateTime double long CLongBinary short float CString

ODBC SQL Type


DAO_BYTES DAO_BOOL DAO_BYTES DAO_CURRENCY DAO_DATE DAO_R8 DAO_I4 DAO_BYTES DAO_I2 DAO_R4 DAO_CHAR, DAO_WCHAR

NOTE
It is recommended that applications not use the DFX_LongBinary function but use DFX_Binary instead. DFX_LongBinary is provided for compatibility with ODBC.

The new member variables and functions in our applications view and document classes are a simple business. The document class, CADAODoc, contains a new member variable of type CADAOSet, m_aDAOSet. Obviously, this variable represents the recordset with which the document is associated. The view class contains a pointer of type CADAOSet (m_pSet); in the default implementation, this pointer is set to point to the document objects m_aDAOSet member. The view class also has a new member function, OnGetRecordset, which in the default implementation simply returns m_pSet. Although you can recompile the ADAO application at this time, as Figure 33.11 illustrates, it is not yet a very useful application. You must customize its dialog, and you also must add the necessary operations that will restrict the rows selected to rows that you want to see.

Data Access Objects CHAPTER 33

617

FIGURE 33.11.
Running the ADAO application skeleton.

Customizing the Application


The first step in customizing the ADAO application is changing its main dialog. Open the IDD_ADAO_FORM dialog for editing, remove the default TODO static control, and add static controls and edit controls as shown in Figure 33.12.

FIGURE 33.12.
Customizing the ADAO dialog.

33
DATA ACCESS OBJECTS

Name the five edit fields IDC_LASTNAME, IDC_FIRSTNAME, IDC_AGE, IDC_NAME, and IDC_MAXAGE as appropriate. Before dismissing this dialog, you can also use the ClassWizard to identify dialog fields with corresponding recordset member variables. To do so, hold down the Control key and double-click the IDC_LASTNAME edit field. The ClassWizard Add Member Variable dialog appears, with ClassWizards guess as to the name of the member variable (see Figure 33.13). ClassWizard derives its guess by looking at the static field in the dialog that, in the dialogs tab order, precedes the control associated with the member variable.

618

Client/Server Solutions PART V

FIGURE 33.13.
Adding a recordset member variable.

ClassWizards guess is appropriate for the first three fields in our dialog; however, for the plan name and plan maximum age fields, it is necessary to manually change ClassWizards selection. You can do this by selecting the proper m_pSet member variable from the drop-down list in the Add Member Variable dialog. After you have specified the member variables for all five edit fields, you can dismiss the dialog. However, you are not finished yet; you have not yet specified the selection criteria that would make the application display only the rows representing valid plans for each employee. To change the selection criteria, open the CADAOSet::GetDefaultSQL function for editing. The default implementation of this function simply returns the table names that form the recordset. What we want to do is add additional criteria that would restrict the selections from the tables to only those rows that we want to see. In SQL, our desired selection can be expressed in the form of a SELECT statement:
SELECT Employees.LastName, Employees.FirstName, Employees.Age, Plans.Name, Plans.MaxAge FROM Employees, Plans WHERE Employees.Age < Plans.MaxAge ORDER BY Employees.LastName, Employees.FirstName, Plans.Name

Indeed, one way to specify our selection would be to change GetDefaultSQL to return a string representing the preceding SQL SELECT statement. However, there is another way: utilize the member variables of the CDaoRecordset class. In particular, CDaoRecordset offers two member variables, one of which (m_strFilter) corresponds to the SQL WHERE clause and the other (m_strSort) to the SQL ORDER_BY clause. Our new version of CADAOSet::GetDefaultSQL (see Listing 33.3) utilizes these member variables to create the desired selection of rows.

Data Access Objects CHAPTER 33

619

Listing 33.3. Updated version of CADAOSet::GetDefaultSQL.


CString CADAOSet::GetDefaultSQL() { m_strFilter = _T([Employees].[Age] < [Plans].[MaxAge]); m_strSort = _T([Employees].[LastName],[Employees].[FirstName],[Plans].[Name]); return _T([Employees],[Plans]); }

This completes our work on the ADAO project. Recompiling and running the application shows that it indeed behaves as expected, displaying benefit plans that employees qualify for, ordered by the name of the employee and the name of the benefit package (see Figure 33.14).

FIGURE 33.14.
Running the ADAO application.

33
DATA ACCESS OBJECTS

DAO Classes
Although the ADAO application demonstrates how a simple DAO program can be created, it fails to demonstrate many DAO features. To remedy this deficiency, the rest of this chapter presents a brief tour of MFC DAO classes and their capabilities. The set of DAO classes offered by MFC is shown in Figure 33.15. In addition to the CDaoRecordset class that we have encountered while constructing ADAO, there are four other major classes and two helper classes related to DAO. Still, this is a significant improvement over the multitude of raw DAO objects (shown at the beginning of the chapter in Figure 33.1). Before you review the role and features of each of these classes one by one, this section presents a brief overview of how DAO works. For this, it might be helpful to take another look at Figure 33.1. All DAO objects are derived from the DBEngine object; furthermore, all database objects are derived from DAO workspace objects. However, unless you must manipulate secure databases, you typically do not need to reference either of these; instead, a default workspace object is assumed for all transactions.

620

Client/Server Solutions PART V

FIGURE 33.15.
DAO classes.

CDaoDatabase

CDaoFieldExchange

CDaoQueryDef CDaoException CDaoRecordset

CDaoTableDef

CDaoWorkspace

The database and recordset objects obviously represent databases and selection sets (tables, recordsets, or dynasets) in those databases. Query definition (QueryDef) objects are used to execute specific SQL queries against a database. Query definitions are normally used in conjunction with recordsets to access data in a database using a specific query. Table definition (TableDef) objects represent the structure of tables in the database. Through table definition objects, it is possible to create new tables and modify the structure and characteristics of existing tables. There are several other DAO object types. These object types (Field objects, Parameter objects, Index objects, User objects, Group objects, and Error objects) are not represented by specific MFC classes. Instead, DAO objects of this type are accessed through the other DAO MFC classes as appropriate.

The CDaoRecordset Class


CDaoRecordset objects represent recordsets. A recordset can represent records in a table, a dynaset, and a snapshot. A table-type recordset is updatable and represents the records in a single table. A dynaset-type recordset represents records from one or more tables as a result of a query; dynaset records are also updatable. A snapshot, on the other hand, can also contain fields from one or more tables, but these fields are not updatable; the snapshot is a static copy of records used to find data or generate reports.

A recordset is created by calling the CDaoRecordset::Open member function. The three forms of this function enable you to create a recordset using an SQL query string, a CDaoTableDef object, or a CDaoQueryDef object. The CDaoRecordset class offers a large number of member functions. Perhaps the most important among these are recordset navigation functions and data update functions. The navigation functions include Find, FindFirst, FindLast, FindNext, and FindPrev; and Move, MoveFirst, MoveLast, MoveNext, and MovePrev . Data update functions include AddNew , CancelUpdate , Delete, Edit, and Update.

Data Access Objects CHAPTER 33

621

Other navigation-related functions include G e t A b s o l u t e P o s i t i o n , G e t B o o k m a r k , GetPercentPosition, and SetAbsolutePosition, SetBookmark, and SetPercentPosition. The CDaoRecordset class offers a variety of attribute functions to set and retrieve recordset attributes. For example, the CanUpdate function can be used to determine whether a recordset is updatable; the SetCurrentIndex function can be used to set the current index on a table-type recordset. Normally, you use the CDaoRecordset class by deriving your own recordset class from it, adding member variables that represent fields, and overriding the DoFieldExchange member function to facilitate the exchange of data between the database and the member variables. However, several member functions exist that provide an alternative. These include GetFieldValue and SetFieldValue, which enable you to directly access the value of a field by name. This method is referred to as dynamic binding, as opposed to the static binding accomplished through DoFieldExchange. Other recordset operations can be used to control the locally maintained cache of records and to manipulate recordset indexes.

The CDaoDatabase Class


The CDaoDatabase class represents a connection to a database. A connection is created by calling CDaoDatabase::Open and terminated by calling CDaoDatabase::Close. A new database can be created by calling CDaoDatabase::Create. The CDaoDatabase class offers a series of attribute member functions; for example, the GetName member function can be used to retrieve the name of the database, or the IsOpen member function can be used to determine whether the connection represented by the CDaoDatabase object is open. Other member functions can be used to manipulate the collections of table definition and query definition objects that are defined for this database. In particular, you can use the DeleteTableDef member function to delete not only a DAO TableDef object but also the underlying table and all its data from the database.

33
DATA ACCESS OBJECTS

The CDaoWorkspace Class


The CDaoWorkspace class represents database sessions. Typically, you do not need to create objects of type CDaoWorkspace, unless you want to utilize specific functionality available through this class or to access password-protected databases. A DAO workspace can be created by calling CDaoWorkspace::Create. Arguments to this function specify the name of the workspace, the username, and password. An existing workspace object can be opened by calling CDaoWorkspace::Open; by passing a NULL parameter to this function, you can explicitly open the default workspace.

622

Client/Server Solutions PART V

Several member functions exist that manipulate databases and the database engine itself. For example, you can compact or repair a database by calling the CompactDatabase or RepairDatabase member functions. Other functions can be used to manipulate usernames, passwords, and other database attributes.

The CDaoQueryDef Class


The CDaoQueryDef class represents query definitions. To create a new query definition, use the CQueryDef::Create member function; to access a query definition that was saved into a database, use CQueryDef::Open. A newly created query can be added to the database by calling the CQueryDef::Append member function.
CQueryDef objects can be used in conjunction with CRecordSet objects to retrieve data from the database. CQueryDef objects can also be used directly; to execute an action query that modifies the data in the database, use the CQueryDef::Execute member function.

Other CQueryDef member functions can be used to set and retrieve query definition attributes and to manipulate query fields and parameters.

The CDaoTableDef Class


The CDaoTableDef class represents table definitions. A table definition describes the structure and attributes of a table in a database. You can open an existing table definition in a database by calling CDaoTableDef::Open. A new table definition can be created by calling CDaoTableDef::Create. To add a table corresponding to a new definition to the database, call the Append member function. Fields can be created and deleted by calling the CreateField and DeleteField member functions. Indexes for the table can be created or deleted by calling CreateIndex and DeleteIndex. Other member functions can be used to set or retrieve various table attributes; for example, GetFieldCount returns the number of fields in the table, and SetValidationRule can be used to assign a validation rule to a field.

Miscellaneous DAO Classes


In addition to the five fundamental DAO classes, DAO operations make use of two additional classes: CDaoFieldExchange and CDaoException. is used in calls to CDaoRecordset::DoFieldExchange. An object of type CDaoFieldExchange defines the field that is affected by the field exchange operation and provides other parameters that characterize the field exchange.
CDaoFieldExchange

All DAO classes utilize exception objects of type CDaoException to report errors.

Data Access Objects CHAPTER 33

623

Summary
Data Access Objects represent an OLE-based technology used in Visual Basic, Visual Basic for Applications, and Microsoft Access to access databases through the Microsoft Jet database engine. The Microsoft Foundation Classes Library and the Visual C++ AppWizard and ClassWizard provide extensive support for developing DAO-based applications in Visual C++. DAO libraries are supplied in the form of redistributable components that you can freely distribute with your Visual C++ applications. Constructing a DAO application through AppWizard is simple. The steps to construct a simple application include specifying the data source, modifying the applications main dialog, and adding recordset member variables to the dialog. The DAO object hierarchy is a complex hierarchy of several objects. The MFC provides a greatly simplified view of DAO, through a set of five core DAO classes and two supplementary classes. Of the core classes, CDaoQueryDef and CDaoRecordset represent queries against a database and the query results; databases themselves are represented by CDaoDatabase. A class used more rarely is CDaoWorkspace; unless you must access secure databases, you can normally rely on the implied default workspace rather than create an object of type CDaoWorkspace explicitly. Finally, CDaoTableDef is used to represent table structures; through this class you can add tables to your database and manipulate existing tables.

33
DATA ACCESS OBJECTS

624

Client/Server Solutions PART V

OLE DB and ADO CHAPTER 34

625

OLE DB and ADO

34

IN THIS CHAPTER
s OLE DB 626 s ActiveX Data Objects 633

34
OLE DB AND ADO

626

Client/Server Solutions PART V

An immortal saying made by one of the giants from the golden age of physics in the 1930s is a question attributed to I. I. Rabi upon the discovery of the muon: Who ordered that? He was, of course, expressing the frustration felt by many of his colleagues when they learned of a new elementary particle that wrecked existing theories. Many Windows programmers probably feel the same way as new technologies appear to arrive from Redmond, Washington on an almost daily basis. Yet just as a place was eventually found for the muon in new physical theories that provided a better, deeper understanding of the forces of nature, a better understanding of Microsofts development strategy also helps us understand the role of many of its new development tools and components. OLE DB and ADO are no exception. The uninformed programmer might have wondered what need exists for these two new specifications, when we already had two competing data access technologies available for use by the C++ programmer: ODBC and DAO. And now there is more: ActiveX template support for OLE DB. It is important to realize that neither OLE DB nor ADO are replacements for existing DBMS technologies. No, they venture instead to a place where no DBMS programmer has gone before: OLE DB is an attempt to create a COM-based interface specification that unites traditional (DBMS) data sources and other sources of data ranging from plain file systems to electronic mail. ADO stands for ActiveX Data Objects, and represents a small-footprint technology (based, in part, on OLE DB) for data access. By small footprint I mean, of course, that the resulting code objects (executable or library files) are small, which makes them best suited for Internet-based content. This chapter provides a brief overview of these two technologies. More reading material, including up-to-the-minute information on these rapidly changing technologies, is available at Microsofts Web site.

OLE DB
OLE DB is a set of Component Object Model (COM) interface specifications. These interfaces are exposed by providers; they are used by consumers. Note that this distinction is purely conceptual; a single application or library can act in both of these capacities. Indeed, this is what OLE DB service providers do; instead of owning data, these components work as proxies by providing an encapsulation for data provided by another service.

OLE DB and ADO CHAPTER 34

627

The OLE DB SDK


I should emphasize again that OLE DB is a specification, not a library or a program. The product that you receive on your Visual C++ CD-ROM is the OLE DB SDK; this software development kit contains header files, libraries, and examples that demonstrate both OLE DB providers and consumers. Part of the OLE DB SDK is the Microsoft OLE DB Provider for ODBC data. This provider is installed along with other OLE DB SDK components, and can be readily used by OLE DB clients that you yourself write. Before you can use the OLE DB SDK you must install it. You might also need to update your Windows 95/98 or Windows NT environment as well as your development system directory settings to ensure that the OLE DB library and header files are found by the compiler.

Basic Concepts
The OLE DB specification defines a series of component objects. These are briefly reviewed in the following sections.

Data Source Objects


The session of an OLE DB client begins by creating a data source object using the CoCreateInstance function. This activates the corresponding OLE DB provider and prepares it for a session. Data source objects expose the IDBProperties interface, through which connection and authentication information (such as a data source name or a user password) can be communicated. They also expose the IDBInitialize interface, which can be used to actually connect to the data source.

Sessions
Sessions are created through the IDBCreateSession interface, also exposed by data source objects. A session object acts as a factory for rowset, command, and transaction objects.

34
OLE DB AND ADO

Commands
Commands are created using the IDBCreateCommand interface of a session. The command text is provider-specific and is set using the ICommandText interface exposed by the command object. The text is often ANSI SQL or a superset of it; however, this is not prescribed in the OLE DB specification.

Rowsets
Rowsets are created when commands are executed through ICommand::Execute. Note that command objects do not always return an object when this method is called. For example, a rowset

628

Client/Server Solutions PART V

is returned if the command is an SQL SELECT statement, but not if the command represents an update or delete query. Rowsets can also be created using the IOpenRowset interface exposed by session objects. This is the preferred method if the provider does not support commands.

Transactions
Transactions provide a means for coordinated processing on multiuser systems. If a provider supports transactions, the session exposes the ITransactionLocal interface. Through its StartTransaction and Commit methods, transaction processing can be implemented.

Enumerators
Enumerator objects can be used to enumerate other objects. Enumerators are created by calling CoCreateInstance with the enumerators class ID. A root enumerator, shipped with the OLE DB SDK, enumerates data sources and other enumerators. Using this enumerator is the preferred alternative to searching the Registry when trying to establish the set of available data sources. Other enumerators can be provider-specific; for example, enumerators can be used to traverse a file system.

Errors
OLE DB error objects extend the capabilities of Automation error objects with the capabilities to return multiple error records and to return provider-specific errors. An error object is accessed by calling GetErrorInfo from the Automation DLL and retrieving an IErrorInfo interface on this object. Providers can also create error objects using CoCreateInstance.

A Working Example
Now to put it all together in the form of a simple Win32 console application. This example does not use commands, transactions, enumerators, or error objects. It is an OLE DB consumer application that accesses a Microsoft Access database through the Microsoft OLE DB Provider for ODBC and dumps the contents of a table from that database to the console in a tab-delimited format. In order to keep the code simple, it is also assumed that all fields in the specified table are text fields. The complete program is provided in Listing 34.1. Because it consists of only a single main function, it is a fairly straightforward example of the basic OLE DB calls and functionality. This program can be compiled using the following command line: cl od.cpp oledb.lib ole32.lib oleaut32.lib. Note that you must have the OLE DB SDK installed in order to compile this example successfully.

OLE DB and ADO CHAPTER 34

629

Listing 34.1. An OLE DB example.


#define DBINITCONSTANTS #include #include #include #include #include #include <iostream.h> <assert.h> <stddef.h> <ole2ver.h> <oledb.h> <msdasql.h>

const WCHAR szDSNtemplate[] = LDRIVER={Microsoft Access Driver (*.mdb)};DBQ=; void main(int argc, char *argv[]) { IMalloc* pIMalloc; IDBInitialize *pIDBInit; IDBProperties *pIDBProperties; IDBCreateSession *pIDBCreateSession; IOpenRowset *pIOpenRowset; IRowset *pIRowset; IColumnsInfo *pIColumnsInfo; IAccessor *pIAccessor; HACCESSOR hAccessor; HROW *rghRows = NULL; DBPROP dbProp; DBPROPSET dbPropSet; DBID dbID; DBCOLUMNINFO *pColumnInfo; WCHAR *pColumnNames; ULONG cColumns; ULONG cBindings; ULONG cRowsObtained; DBBINDING *rgBindings; BYTE *pData; struct COLUMNDATA { DWORD dwLength; DWORD dwStatus; BYTE bData[1]; } *pColumn; DWORD dwOffset; int i, j, l; char *p; WCHAR *pwszDSN; WCHAR *pwszTable; if (argc != 3) { cerr << Usage: << argv[0] << database-name table-name << endl; exit(1);

34
OLE DB AND ADO

continues

630

Client/Server Solutions PART V

Listing 34.1. continued


} l = strlen(argv[1]) + 1; pwszDSN = new WCHAR[wcslen(szDSNtemplate) + l]; wcscpy(pwszDSN, szDSNtemplate); MultiByteToWideChar(CP_ACP, 0, argv[1], l, pwszDSN + wcslen(pwszDSN), l); l = strlen(argv[2]) + 1; pwszTable = new WCHAR[l]; MultiByteToWideChar(CP_ACP, 0, argv[2], l, pwszTable, l); CoInitialize(NULL); CoCreateInstance(CLSID_MSDASQL, 0, CLSCTX_INPROC_SERVER, IID_IDBInitialize, (void**) &pIDBInit); CoGetMalloc(MEMCTX_TASK, &pIMalloc); VariantInit(&dbProp.vValue); dbProp.dwOptions = DBPROPOPTIONS_REQUIRED; dbProp.dwPropertyID = DBPROP_INIT_PROVIDERSTRING; dbProp.colid = DB_NULLID; dbProp.vValue.vt = VT_BSTR; dbProp.vValue.bstrVal = SysAllocString(pwszDSN); delete[] pwszDSN; dbPropSet.rgProperties = &dbProp; dbPropSet.cProperties = 1; dbPropSet.guidPropertySet = DBPROPSET_DBINIT; pIDBInit->QueryInterface(IID_IDBProperties, (void **)&pIDBProperties); pIDBProperties->SetProperties(1, &dbPropSet); if (pIDBInit->Initialize() == E_FAIL) { cerr << Invalid database name << endl; goto NODB; } pIDBInit->QueryInterface(IID_IDBCreateSession, (void **)&pIDBCreateSession); pIDBCreateSession->CreateSession(NULL, IID_IOpenRowset, (LPUNKNOWN *)&pIOpenRowset); dbID.eKind = DBKIND_NAME; dbID.uName.pwszName = pwszTable; pIOpenRowset->OpenRowset(NULL, &dbID, NULL, IID_IRowset, 0, NULL, (LPUNKNOWN*) &pIRowset); if (pIRowset == NULL) { cerr << Invalid table name << endl; goto NOTABLE; } pIRowset->QueryInterface(IID_IColumnsInfo, (void**) &pIColumnsInfo); pIColumnsInfo->GetColumnInfo(&cColumns, &pColumnInfo, &pColumnNames); pIColumnsInfo->Release(); for (i = 0, cBindings = 0; i < cColumns; i++) { if (pColumnInfo[i].wType != DBTYPE_VECTOR) cBindings++; } rgBindings = new DBBINDING[cBindings];

OLE DB and ADO CHAPTER 34


dwOffset = 0; for (i = 0, j = 0; i < cColumns; i++) { if (pColumnInfo[i].wType == DBTYPE_VECTOR) continue; assert(j < cBindings); if (j > 0) cout << \t; l = wcslen(pColumnInfo[i].pwszName) + 1; p = new char[l]; WideCharToMultiByte(CP_ACP, 0, pColumnInfo[i].pwszName, l, p, l, NULL, NULL); cout << p; delete[] p; rgBindings[j].dwPart = DBPART_VALUE | DBPART_LENGTH | DBPART_STATUS; rgBindings[j].eParamIO = DBPARAMIO_NOTPARAM; rgBindings[j].iOrdinal = pColumnInfo[i].iOrdinal; rgBindings[j].wType = DBTYPE_STR; rgBindings[j].pTypeInfo = NULL; rgBindings[j].obValue = dwOffset + offsetof(COLUMNDATA, bData); rgBindings[j].obLength = dwOffset + offsetof(COLUMNDATA, dwLength); rgBindings[j].obStatus = dwOffset + offsetof(COLUMNDATA, dwStatus); rgBindings[j].cbMaxLen = pColumnInfo[i].wType == DBTYPE_STR ? pColumnInfo[i].ulColumnSize + sizeof(char) : 100; rgBindings[j].pObject = NULL; dwOffset += rgBindings[j].cbMaxLen + offsetof(COLUMNDATA, bData); dwOffset = (dwOffset + 7) & ~7; j++; } cout << endl; pData = new BYTE[dwOffset]; pIRowset->QueryInterface(IID_IAccessor, (void**)&pIAccessor); pIAccessor->CreateAccessor(DBACCESSOR_ROWDATA, cBindings, rgBindings, 0, &hAccessor, NULL); while ( SUCCEEDED(pIRowset->GetNextRows(NULL, 0, 100, &cRowsObtained, &rghRows)) && cRowsObtained) { for (i = 0; i < cRowsObtained; i++) { pIRowset->GetData(rghRows[i], hAccessor, pData); for (j = 0; j < cBindings; j++) { pColumn = (COLUMNDATA *)((BYTE *)pData + rgBindings[j].obLength); if (j > 0) cout << \t; cout << (char *)(pColumn->bData); } cout << endl; } pIRowset->ReleaseRows(cRowsObtained, rghRows, NULL, NULL, NULL); } pIAccessor->ReleaseAccessor(hAccessor, NULL); pIAccessor->Release(); pIRowset->Release(); pIMalloc->Free(pColumnInfo); pIMalloc->Free(pColumnNames);

631

34
OLE DB AND ADO

continues

632

Client/Server Solutions PART V

Listing 34.1. continued.


pIMalloc->Free(rghRows); delete[] rgBindings; delete[] pData; NOTABLE: pIOpenRowset->Release(); pIDBCreateSession->Release(); NODB: pIDBProperties->Release(); pIDBInit->Release(); pIMalloc->Release(); CoUninitialize(); delete[] pwszTable; }

The program starts with an analysis of the command line; it should be invoked with two parameters, the first representing the filename of the Microsoft Access database that you want to access, and the second should be the name of the table that is to be dumped. For example, if the file is called BOOKS.MDB, located in C:\BOOKS, and the table to be dumped is called Authors, the program (named OD.EXE) would be invoked as follows:
OD C:\BOOKS\BOOKS.MDB Authors

The database string is then combined into an SQL connection string and converted into Unicode, as required by 32-bit OLE. Using an SQL connection string helps you avoid having to install an ODBC data source through the ODBC Manager. Next the COM libraries are initialized and a data source object is created with its IDBInitialize interface exposed. In order to initialize the data source, you must communicate the OLE connection string to it as a property. This will be the provider-specific DBPROP_INIT_PROVIDERSTRING property; using DBPROP_INIT_DATASOURCE would be more standard, but it wouldnt allow you to use a connection string, only a data source name installed through the ODBC Manager. After a session is successfully created, you proceed directly to creating a rowset using the IOpenRowset interface. More complex applications can use a command object instead, which would allow, for example, submitting an SQL query string to the provider. After a rowset is obtained, columns are enumerated and column headings are printed. Finally, the program retrieves data and iterates through records, printing the contents of each record. When its task is finished the program releases all COM objects and frees up memory allocated locally or by COM methods.

OLE DB and ADO CHAPTER 34

633

ActiveX Data Objects


ActiveX Data Objects is a technology specifically developed for client applications that are to access and manipulate data in a database. ADO is a high-speed, small-footprint technology that is well suited for Web-based content development. ADO is a specification, not a product. However, a specific ADO implementation, ADODB, is supplied as part of the OLE DB SDK. ADODB is an ADO implementation that uses OLE DB providers, including the Microsoft OLE DB Provider for ODBC.

ADO Objects Overview


ADO specifies a set of seven object types and four collection types. These are depicted in Figure 34.1 and briefly reviewed in the following paragraphs.

FIGURE 34.1.
The ADO object model.

Connection

Errors collection

Error

Command

Parameters collection Parameter

34
Recordset

OLE DB AND ADO

Fields collection

Field

Properties collection

Property

634

Client/Server Solutions PART V

The Connection Object


A Connection object represents a connection to a data source. Connection objects are created independently of previously created objects; in ADO programs, creating a Connection object is usually the first step. Connection objects offer methods for managing connections and methods for transaction processing.

The Properties Collection and Property Objects


Property objects represent dynamic characteristics that are not implemented by the methods and properties of ADO objects. The meaning of Property objects is defined by the provider. Property objects are organized into Properties collections. Connection objects, Command objects, Recordset objects, and Field objects can each have a Properties collection.

The Errors Collection and Error Objects


When an error occurs during an ADO operation, one or more Error objects are placed into the Errors collection of the Connection object. Error objects have several properties such as Description, Number, Source, or SQL State that provide an accurate description of the error that occurred.

The Command Object


Command objects are used to create and execute a command (such as an SQL statement) on a provider. Command objects can be created by themselves, in which case Connection objects are created implicitly as needed; or, if you use several Command objects, it is best if a Connection object is created first and the ActiveConnection property of any new command is set to reference it. Command objects can be used to obtain records from the data source in the form of Recordset objects.

The Parameters Collection and Parameter Objects


Command objects can reference stored procedures or parameterized queries, in which case it might be necessary to pass parameters to the Command object before execution. This is accomplished through the use of the Command objects Parameters collection. Parameter objects support the eters can be defined.
Name

and

Value

properties, through which individual param-

The Recordset Object


A Recordset object represents a collection of records. Such a collection can be, for example, the result set of a SELECT query.

OLE DB and ADO CHAPTER 34

635

Recordset objects can be created independently, in which case a Connection object is created implicitly. Alternatively, Recordset objects can be opened using a previously defined Connection object. Recordset objects support methods for traversing records and for retrieving data. The GetRows method, in particular, can be used to retrieve information efficiently by returning the contents of multiple rows at once.

The Fields Collection and Field Objects


A Fields collection is associated with each Recordset object. Through the Fields collection, columns of a table or result set can be enumerated and information about them, such as the column name, can be retrieved. Columns are represented by Field objects in a Fields collection. Field objects have properties such as Name, Type, or Precision that describe the column with which they are associated.

A Working Example
ADO is best suited for use from within scripting languages, such as Visual Basic. However, it is also relatively easy to write simple ADO applications using C++. The sample program in this section is one such application; in fewer than 90 lines of code it accesses an OLE DB data source, executes an SQL SELECT query, and displays the results in a tab-delimited form. This program specifically references a Microsoft Access database called BOOKS.MDB. This database contains two tables: the Authors table lists the names and nationalities of authors; the Books table lists the ISBN numbers, titles, and authors of books. The query executed in the sample program shown in Listing 34.2 lists all books that were written by American authors.

34
Listing 34.2. An ADO example. OLE DB AND ADO
#define INITGUID #include #include #include #include <iostream.h> <objbase.h> <adoid.h> <adoint.h>

const WCHAR wszConnection[] = LDRIVER={Microsoft Access Driver (*.mdb)};DBQ=BOOKS.MDB; const WCHAR wszSQL[] = LSELECT Books.*, Authors.Nationality LFROM { oj Books LEFT OUTER JOIN Authors ON Books.Author = Authors.Name } LWHERE (Authors.Nationality = USA); void main(void) {

continues

636

Client/Server Solutions PART V

Listing 34.2. continued


ADORecordset *pRecordset; ADOFields *pFields; ADOField *pField; VARIANT varSQL; VARIANT varConnection; VARIANT varStart; VARIANT varField; VARIANT varRows; VARIANT varItem; BSTR strColumn; LONG lColumns, lRows; LONG lIndex[2]; LONG l; char *p; VariantInit(&varConnection); VariantInit(&varSQL); VariantInit(&varStart); VariantInit(&varField); VariantInit(&varItem); varConnection.vt = VT_BSTR; varConnection.bstrVal = SysAllocString(wszConnection); varSQL.vt = VT_BSTR; varSQL.bstrVal = SysAllocString(wszSQL); varStart.vt = varField.vt = VT_ERROR; varStart.scode = varField.scode = DISP_E_PARAMNOTFOUND; CoInitialize(NULL); CoCreateInstance(CLSID_CADORecordset, NULL, CLSCTX_INPROC_SERVER, IID_IADORecordset, (void **)&pRecordset); pRecordset->Open(varSQL, varConnection, adOpenForwardOnly, adLockReadOnly, adCmdText); pRecordset->GetRows(adGetRowsRest, varStart, varField, &varRows); SafeArrayGetUBound(varRows.parray, 1, &lColumns); SafeArrayGetUBound(varRows.parray, 2, &lRows); pRecordset->get_Fields(&pFields); for (lIndex[0] = 0; lIndex[0] <= lColumns; lIndex[0]++) { varItem.vt = VT_INT; varItem.intVal = lIndex[0]; pFields->get_Item(varItem, &pField); pField->get_Name(&strColumn); if (lIndex[0] > 0) cout << \t; l = wcslen(strColumn) + 1; p = new char[l]; WideCharToMultiByte(CP_ACP, 0, strColumn, l, p, l, NULL, NULL); cout << p; delete[] p; SysFreeString(strColumn); pField->Release(); } cout << endl; pFields->Release();

OLE DB and ADO CHAPTER 34


for (lIndex[1] = 0; lIndex[1] <= lRows; lIndex[1]++) { for (lIndex[0] = 0; lIndex[0] <= lColumns; lIndex[0]++) { SafeArrayGetElement(varRows.parray, &lIndex[0], &varField); if (lIndex[0] > 0) cout << \t; l = wcslen(varField.bstrVal) + 1; p = new char[l]; WideCharToMultiByte(CP_ACP, 0, varField.bstrVal, l, p, l, NULL, NULL); cout << p; delete[] p; } cout << endl; } pRecordset->MoveFirst(); pRecordset->Release(); SysFreeString(varConnection.bstrVal); SysFreeString(varSQL.bstrVal); }

637

This program is a Win32 console application that can easily be compiled from the command line using the following command:
cl ad.cpp ole32.lib oleaut32.lib

In order to compile and run this program successfully, you must have the OLE DB SDK installed on your system. Execution of this program is very straightforward and requires little explanation. After some variable initialization, you begin the real work by creating an ADO Recordset object. This object is used to establish a connection using an ODBC connection string; using a connection string instead of a data source name helps you avoid having to install the data source through the ODBC Manager. The query is executed and the results are retrieved by the call to the GetRows method of the Recordset object. The rest of the application is little more than pretty printing. First, field information for each column is retrieved so that a header row can be printed; next, an iteration is made through all rows and data is printed to the standard output. Note that it is assumed that all fields in this query return a text value; if you use this program as a basis for your own applications, you might need to add code that examines the field type rather than blindly assuming a type that might be incorrect.

34
OLE DB AND ADO

638

Client/Server Solutions PART V

Summary
OLE DB is an interface specification that allows access to diverse data sources using the Component Object Model (COM). These data sources can be traditional (DBMS) but can also include other sources such as file systems or electronic mail. OLE DB applications can be providers or consumers. Providers expose interfaces; consumers utilize interfaces to perform database tasks. The OLE DB specification defines a series of objects: data source objects, session objects, command objects, rowset objects, transaction objects, enumerator objects, and error objects. Consumers create a data source object by calling CoCreateInstance and use interfaces on this object to create a session. Sessions, in turn, are used to create commands and rowsets for accessing data. ActiveX Data Objects provides a high-speed, small-footprint technology for accessing data. As such, it is ideal for Web-based applications, but can also be used from ordinary C++ applications with ease. ADO defines a set of seven objects and four collections. Connections to data sources are represented by Connection objects. The results of queries executed on the data source are represented by Recordset objects. Other object types include Field objects (in Fields collections), Command objects and corresponding Parameter objects (in Parameters collections), Errors collections containing Error objects, and Properties collections containing Property objects.

Writing a Windows NT Service CHAPTER 35

639

Writing a Windows NT Service

35

IN THIS CHAPTER
s Services in the Windows NT Environment 640 s Creating a Windows NT Service Application 642

35
WRITING A WINDOWS NT SERVICE

640

Client/Server Solutions PART V

One of the more arcane tasks a Windows NT programmer has to face occasionally is the need to write a Windows NT service application. The need for this special class of applications becomes clear if you consider the Windows NT security model. All applications launched by a user run under that users account and privileges; when the user logs off from Windows NT, all applications the user started will be stopped. Clearly, this is not good enough for many server-side applications. Take, for instance, a database server; it is a piece of software that is expected to run 24 hours a day, independent of any currently logged-on user. Furthermore, many such server-side applications must run under special privileges. The answer that Windows NT provides to this dilemma is in the form of service applications.

Services in the Windows NT Environment


What is a Windows NT service? It is typically an application that performs some system task independently of any currently logged-on user. Service applications tend to run all the time in the background, waiting for specific events or listening for incoming service requests.

The Service Control Manager


Service applications are controlled by the Service Control Manager. This Windows NT system component is started by Windows NT when the system is booted. It maintains a database of installed services and starts services as appropriate. It also provides a Remote Procedure Call (RPC) server interface so that services can be started or stopped on remote machines. Typically, you would use the Services applet (Figure 35.1) of the Control Panel to interact with the Service Control Manager. This applet can be used to start or stop a service, or change its runtime characteristics. However, you cannot install new services or remove existing services this way. That is a task performed by the service application itself, an associated utility program, or perhaps an installation program.

FIGURE 35.1.
The Services applet.

Writing a Windows NT Service CHAPTER 35

641

Starting and Stopping Services


Many services are started by the Service Control Manager automatically at system boot time. Other services are started manually if needed. To see details about how a service is started, select the service in the Services applet and click the Startup button. This invokes the Service dialog (Figure 35.2).

FIGURE 35.2.
The Service dialog.

A service that is marked Automatic will be started by the Service Control Manager automatically. A service that is marked Manual is either started by a user or by another application (or service). Services marked Disabled cannot be started. (There are actually two additional startup types, Boot and System, started respectively at the time the operating system is loaded or I/O is initialized. These settings are valid only for driver services.) In all cases, the Service dialog allows you to specify the account under which the service is run. Most services are run under the local system account; some services (notably, services that require special access privileges) are run under other accounts. Most services run in the background with no user interaction at all. However, services that are run under the system account can also be permitted to interact with the desktop of the currently logged on user. This, of course, can represent a security risk, because you dont know who might be logged on to the system at the time the service presents its user interface.

35
WRITING A WINDOWS NT SERVICE

Services and the Win32 API


The Win32 API provides a number of function calls for interacting with the service control manager. Some of these functions are meant for use by service control programs (programs that communicate with the Service Control Manager for the purposes of starting, stopping, or otherwise manipulating services). Others are meant for use by the services themselves.

642

Client/Server Solutions PART V

A service control program obtains a handle to the Service Control Managers database by calling OpenSCManager. The Service Control Manager can be local or it can reside on another computer. The handle that is returned can be used in calls to CreateService or OpenService, in order to obtain a handle to a particular service. That handle, in turn, can be used in calls such as ControlService, StartService, or DeleteService. A service program, when started by the Service Control Manager, calls
StartServiceCtrlDispatcher to connect itself to the Service Control Manager. The Service Con-

trol Manager, in turn, calls the service applications S e r v i c e M a i n function. The RegisterServiceCtrlHandler function is used to register a function in the service application that will handle requests from the Service Control Manager. At any time during startup, shutdown, or while handling other requests, the service application reports its status to the Service Control Manager using SetServiceStatus. Information on services is stored by the Service Control Manager in the Windows NT Registry. Services are enumerated under the key HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\ Services. It is best, however, if you avoid manipulating Registry entries directly, as they may change on future versions of the Windows NT operating system. Finally, I want to mention the curious fact that a very limited subset of the Service API can also be found on the Windows 95 operating system. It basically allows programs to start in the background prior to the user logging on to the system.

Creating a Windows NT Service Application


In the rest of this chapter, I present a simple, yet useful, service application example. This example implements a multiuser Internet chat service under Windows NT. It allows an arbitrary number of users to connect to the service and communicate with other connected users. It demonstrates the use of multiple threads in a service, as well as the use of the WinSock library.

Using the Windows NT SDK Service Sample


Unfortunately, there is no Visual C++ AppWizard that would create a nice, customized application skeleton for a Windows NT service. However, this does not mean that you need to start from scratch, either. There is a Windows NT service example on your Visual C++ CD-ROM that contains an extremely useful generic component that can be reused in your own applications. I am, of course, referring to the SDK Service example that you can find among the many samples supplied on the Visual C++ CD-ROM (try searching with the quoted phrase Simple Service in the MSDN documentation). Part of this example is the file SERVICE.C. This file provides an implementation of a generic main function for a service, as well as a series of convenience functions. It also handles command-line switches that can be used to install or remove the service, or run the service in a special debug mode.

Writing a Windows NT Service CHAPTER 35


SERVICE.C can be used as is, without modification, in your own service applications. The only changes that need to be made are in the header file SERVICE.H; this file contains a series of definitions that identify the services name, application name, and display name.

643

Copying the files SERVICE.C and SERVICE.H to a new, empty directory and modifying SERVICE.H as needed is actually a good way to start developing our own service application. Here are the lines in SERVICE.H that need changing:
/////////////////////////////////////////////////////////////////// //// todo: change to desired strings //// // name of the executable #define SZAPPNAME CHATSVC // internal name of the service #define SZSERVICENAME SimpleChatService // displayed name of the service #define SZSERVICEDISPLAYNAME Simple Chat Service // list of service dependencies - dep1\0dep2\0\0 #define SZDEPENDENCIES

At this point, you may also want to create a Visual C++ Console Application project for the new chat service. I called my version CHATSVC; the project will work fine with default settings. If you wish to speed up compilation, you may specify precompiled header files and remove unnecessary libraries using the Settings dialog. To facilitate this, I moved all #include directives to a separate file, stdhdr.h (Listing 35.1). There is also a corredsponding C++ file, stdhdr.cpp, containing the single line #include stdhdr.h, which is used to generate the precompile headers file.

Listing 35.1. The precompiled headers file stdhdr.h.


#include <winsock.h> #include <assert.h> #include <iostream.h>

Chat Service Application Architecture


The service implementation in SERVICE.C assumes that you provide two functions: ServiceStart and ServiceStop . ServiceStart has a somewhat misleading name because this function doesnt merely start the service; it is the service, inasmuch as when this function returns, the service is considered to have terminated. ServiceStop is also called from SERVICE.C, but from another thread; this function is called as an indication that the Service Control Manager has requested shutdown of the service. Building around the architecture provided by these functions, here is what our service will look like: Upon startup, it will open a TCP socket, listening for incoming connections. When a connection is requested, a new thread will be created, handling input on that connection. A C++ object will be associated with the service; objects of another C++ class will be associated with each connection to the service. Precautions will be taken to ensure that the application behaves correctly in a multiuser environment.

35
WRITING A WINDOWS NT SERVICE

644

Client/Server Solutions PART V

The application will consist of three modules in addition to SERVICE.C. The first, MAIN.CPP, will contain the ServiceStart, ServiceStop, and service thread functions. The second and third modules, SVC.CPP and SLOT.CPP, provide the implementation of service and slot classes.

The Main Service Module


The main service module, MAIN.CPP, is shown in Listing 35.2. This file contains the definition of three functions: ServiceStart, ServiceStop, and the service thread function.

Listing 35.2. The main service module: MAIN.CPP.


#include #include #include #include stdhdr.h service.h svc.h slot.h // NT (29806)

#define SVCPORT 0x6E74

volatile BOOL g_bStop; volatile BOOL g_bStopped; extern C BOOL bDebug; // defined in SERVICE.C // This is the main thread routine. It receives a user slot pointer. int thread(CSlot *pSlot) { int i = 0; pSlot->Send(Welcome! /QUIT or /BYE are the commands to quit.\n\n); while (1) { char buf[1001]; int n; n = pSlot->Recv(buf + i, 999 - i); if (g_bStop) return 0; if (n > 0) { if (i + n == 999) buf[i + (n++)] = \n; for (int j = i; j < i + n; j++) { if (buf[j] == \n) { buf[j] = \0; if (j > 0 && buf[j - 1] == \r) buf[j - 1] = \0; if (!stricmp(buf, /quit) || !stricmp(buf, /bye)) { delete pSlot; return 0; } else pSlot->m_pSvc->SendAll(buf, pSlot, false); if (g_bStop) return 0; memmove(buf, buf + j + 1, i + n - j - 1); i -= j + 1; break; } } i += n; } } // Neverreach }

Writing a Windows NT Service CHAPTER 35


// // // // Main service entry point. We wait for incoming connections, spawn threads. We shut down if the global (volatile) variable nStop is true. We assume that threads will stop within a finite amount of time.

645

void ServiceStart(unsigned long argc, char *argv[]) { WSADATA wsaData; SOCKET sockfd; SOCKADDR_IN sin; CSvc *pSvc; int id; // We are under Windows NT so I am not worried about too LOW a // version number. if (WSAStartup(MAKEWORD(1, 1), &wsaData)) return; pSvc = new CSvc; if (!ReportStatusToSCMgr(SERVICE_START_PENDING, NO_ERROR, 3000)) goto cleanup; id = 1; sockfd = socket(AF_INET, SOCK_STREAM, 0); if (sockfd == INVALID_SOCKET) goto cleanup; sin.sin_family = AF_INET; sin.sin_port = SVCPORT; sin.sin_addr.s_addr = 0; if (bind(sockfd, (LPSOCKADDR)&sin, sizeof(sin))) goto cleanup; if (listen(sockfd, 5) < 0) goto cleanup; if (!ReportStatusToSCMgr(SERVICE_RUNNING, NO_ERROR, 0)) goto cleanup; while (1) { unsigned long ul; SOCKET newsockfd = accept(sockfd, NULL, NULL); if (g_bStop) { if (newsockfd != INVALID_SOCKET) closesocket(newsockfd); break; } if (newsockfd == INVALID_SOCKET) { if (bDebug) cout << Socket error << WSAGetLastError() << endl; } else { struct linger l; l.l_onoff = TRUE; l.l_linger = 1; setsockopt(newsockfd, SOL_SOCKET, SO_LINGER, (LPSTR)&l, sizeof(l)); CSlot *pSlot = new CSlot(pSvc, newsockfd, id++); CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)thread, pSlot, 0, &ul); } } cleanup: delete pSvc; if (sockfd != INVALID_SOCKET) closesocket(sockfd);

35
WRITING A WINDOWS NT SERVICE

continues

646

Client/Server Solutions PART V

Listing 35.2. continued


WSACleanup(); g_bStopped = TRUE; } // When the service is stopped, we must interrupt any blocking calls. // We do that by setting g_bStop and then connecting to ourselves. void ServiceStop() { WSADATA wsaData; if (WSAStartup(MAKEWORD(1, 1), &wsaData)) return; g_bStop = TRUE; SOCKET s = socket(AF_INET, SOCK_STREAM, 0); SOCKADDR_IN sin; sin.sin_family = AF_INET; sin.sin_port = SVCPORT; sin.sin_addr.s_addr = inet_addr(127.0.0.1); connect(s, (LPSOCKADDR)&sin, sizeof(sin)); while (!g_bStopped); closesocket(s); WSACleanup(); }

Lets begin our review of this code with the ServiceStart function. This function can be divided into three distinct parts: initialization, the main service loop, and termination and cleanup. Initialization consists of startup of the Windows Sockets library and creation of a service object of class CSvc. If both these steps are completed successfully, a loop is entered; in this loop, the application waits for incoming connections on TCP port 29806 (which happens to be the numerical equivalent of the ASCII characters NT). If an incoming connection is detected, the socket is opened, an object of type CSlot representing the connection is created, and a new thread is started that handles incoming data on the new socket. The loop is interrupted if, at the time when an incoming connection is detected, the value of the global variable g_bStop is nonzero. This variable is manipulated from another thread, which is the reason why it is declared volatile, to ensure that compiler optimizations do not produce incorrect code.
ServiceStart uses the ReportStatusToSCMgr convenience function (defined in SERVICE.C) to report service startup status. Failing to call this function would cause the Service Control Manager to assume that the service did not initialize itself properly.

When ServiceStart completes execution, it sets the value of another global variable, g_bStopped, to nonzero. This indicates to other threads that the service is no longer running. Input on a connection is handled by the thread function thread. This function also uses a loop in its implementation, which is interrupted if the user of the service enters one of the quit commands, or if the service itself is being stopped. Most of this function is simply about manipulating a circular buffer, with handling of line-termination characters and the /QUIT and /BYE end user commands.

Writing a Windows NT Service CHAPTER 35

647

The third main module function is ServiceStop. This function first sets the value of g_bStop to nonzero, then makes an attempt to connect to the service itself. This will cause the blocking call to accept in ServiceStart to return, so ServiceStart can terminate. ServiceStop will wait until the value of the global variable g_bStopped changes to nonzero, indicating that ServiceStart completed its execution; at this time, ServiceStop also returns. Note that in this implementation, we assume that ServiceStop will be able to complete its task rapidly. If it were to take three seconds or more to shut down the service, it may be necessary to delegate the shutdown functionality to another thread, to ensure that ServiceStop returns before a time-out occurs. The main module has an embarrassingly simple header file, consisting of a single line only (Listing 35.3). This line declares the g_bStop variable with external linkage, to ensure that it can be referenced from other modules.

Listing 35.3. The one-liner header file: MAIN.H.


extern volatile BOOL g_bStop;

The Service Class


The class CSvc is essentially a simple collection class that maintains a collection of CSlot objects. During the lifetime of our service, a single object of type CSvc is created. All incoming connections are then associated with CSlot objects which are added to, or removed from, the collection represented by CSvc. The declaration of the CSvc class is shown in Listing 35.4. In addition to the constructor, destructor, and the variable m_pHead that marks the head of a linked list of CSlot objects, the class contains three more functions for adding and removing CSlot objects, and for sending messages to all CSlot objects in the collection. Lastly, a critical section object is also created; this object will be used for synchronization when the list of CSlot objects is being manipulated.

Listing 35.4. Declaration of the CSvc class in SVC.H.


class CSlot; class CSvc { public: CSlot *m_pHead; CSvc(); ~CSvc(); void AddSlot(CSlot *pSlot); void RemoveSlot(CSlot *pSlot); void SendAll(char *pBuf, CSlot *pSlot = NULL, bool bSystem = true); CRITICAL_SECTION cs; };

35
WRITING A WINDOWS NT SERVICE

648

Client/Server Solutions PART V

The implementation of the CSvc function (Listing 35.5) is straightforward. Note the calls to EnterCriticalSection and LeaveCriticalSection, used to protect segments of code that directly manipulate the linked list.

Listing 35.5. CSvc class implementation in SVC.CPP.


#include #include #include #include stdhdr.h main.h svc.h slot.h // defined in SERVICE.C

extern C BOOL bDebug;

CSvc::CSvc() { m_pHead = NULL; InitializeCriticalSection(&cs); } CSvc::~CSvc() { SendAll(Service shutdown in progress.); while (m_pHead != NULL) delete m_pHead; } void CSvc::AddSlot(CSlot *pSlot) { EnterCriticalSection(&cs); CSlot **ppSlot = &m_pHead; while (*ppSlot != NULL) ppSlot = &((*ppSlot)->m_pNext); pSlot->m_pNext = NULL; *ppSlot = pSlot; pSlot->m_pSvc = this; LeaveCriticalSection(&cs); if (bDebug) cout << Created slot # << pSlot->GetId() << . << endl; } void CSvc::RemoveSlot(CSlot *pSlot) { EnterCriticalSection(&cs); CSlot **ppSlot = &m_pHead; while (*ppSlot != pSlot) { assert(*ppSlot != NULL); ppSlot = &((*ppSlot)->m_pNext); } *ppSlot = pSlot->m_pNext; pSlot->m_pNext = NULL; LeaveCriticalSection(&cs); if (bDebug) cout << Destroyed slot # << pSlot->GetId() << . << endl; } void CSvc::SendAll(char *pBuf, CSlot *pSender, bool bSystem) { char buf[1020], num[17]; strcpy(buf, [);

Writing a Windows NT Service CHAPTER 35


if (!bSystem) { strcat(buf, itoa((pSender != NULL ? pSender->GetId() : 0), num, 10)); strcat(buf, ] ); } strcat(buf, pBuf); if (bSystem) strcat(buf, ]); if (bDebug) cout << buf << endl; strcat(buf, \n); EnterCriticalSection(&cs); CSlot *pSlot = m_pHead; while (pSlot != NULL) { if (pSlot != pSender) { pSlot->Send(buf); } pSlot = pSlot->m_pNext; } LeaveCriticalSection(&cs); }

649

The SendAll member function is a broadcast-type function that is used to send a string of text to all connected users. The string is preceded by a numeric identifier of the sender, unless the message is marked as a system message, in which case the entire message is enclosed in square brackets. This function also uses the variable bDebug, declared at the top of the file. This variable is defined in SERVICE.C and is used to indicate whether the service has been started up in debug mode. If not (that is, if it is being run by the Service Control Manager), no data is emitted to the standard output because standard output may not exist at all.

The CSlot Class


The last of the modules of our service application is the module that defines the behavior of the CSlot class. The declaration of this class is shown in Listing 35.6.

Listing 35.6. Declaration of the CSlot class in SLOT.H.


class CSvc; class CSlot { private: int m_id; SOCKET m_sockfd; public: CSlot *m_pNext; CSvc *m_pSvc; CSlot(CSvc *pSvc, SOCKET sockfd, int id);

35
WRITING A WINDOWS NT SERVICE

continues

650

Client/Server Solutions PART V

Listing 35.6. continued


~CSlot(); void Send(char *pszSend); int Recv(char *pszRecv, int n); int GetId() { return m_id; }; };

The CSlot implementation file, SLOT.CPP (Listing 35.7) implements the CSlot constructor, destructor, and two member functions for sending and receiving data. The constructor and the destructor use the AddSlot and RemoveSlot member functions to add the slot to the list in the CSvc service object, or to remove the slot. The linked list of slots is implemented through the m_pNext variable.

Listing 35.7. CSlot class implementation in SLOT.CPP.


#include #include #include #include stdhdr.h main.h svc.h slot.h

CSlot::CSlot(CSvc *pSvc, SOCKET sockfd, int id) { char buf[80], num[17]; m_id = id; m_sockfd = sockfd; pSvc->AddSlot(this); CSlot *pSlot = pSvc->m_pHead; strcpy(buf, User #); strcat(buf, itoa(m_id, num, 10)); strcat(buf, has arrived.); pSvc->SendAll(buf, this); } CSlot::~CSlot() { char buf[80], num[17]; CSlot *pSlot = m_pSvc->m_pHead; strcpy(buf, User #); strcat(buf, itoa(m_id, num, 10)); strcat(buf, has just left.); m_pSvc->SendAll(buf, this); m_pSvc->RemoveSlot(this); closesocket(m_sockfd); } void CSlot::Send(char *pszSend) { send(m_sockfd, pszSend, strlen(pszSend), 0); }

Writing a Windows NT Service CHAPTER 35


int CSlot::Recv(char *pszRecv, int n) { return recv(m_sockfd, pszRecv, n, 0); }

651

Compiling and Running the Service


As mentioned earlier, the service can be compiled as a standard Windows NT console application. The only libraries that the application requires are the wsock32.lib and advapi32.lib libraries; all other libraries files can be safely removed from the build settings. After the service application is successfully compiled, you can run it in debug mode using the command-line option. This option is most useful if you wish to use the Visual C++ debugger, as it allows you to execute the service application under the debuggers control.
-debug

When you are ready to install the service, run it once using the -install command-line option. If run this way, the service application registers itself with the Service Control Manager (thanks to the magic performed in SERVICE.C). The service can be deinstalled using the -remove command-line option. After the service is up and running (either through the Service Control Manager or in debug mode) you can connect to it using a Telnet application. (Note that you may need to turn on local echo or line mode in your Telnet client.) Simply telnet to the IP address of your Windows NT system using port number 29806. When you do so, all existing connections receive a notification string; notifications are also sent when a connection is terminated. After its connected, anything you type will be broadcast to all other connected users the moment you hit the Enter key.

Summary
Services are Windows NT applications that run in the background, independent of the desktop of the user currently logged on to the system. Services can run under a specific users account or under the local system account and may or may not interact with the current users desktop. Services are controlled by the Windows NT Service Control Manager. The Service Control Manager maintains a database of installed services in the Registry. It starts and stops services as needed and provides a Remote Procedure Call (RPC) interface to permit control of services by other applications. The Win32 API defines a number of functions related to services. These can be used to register and delete a service, modify the parameters of a service, or start and stop a service. Other Win32 API functions are used by the services themselves to communicate with the Service Control Manager.

35
WRITING A WINDOWS NT SERVICE

652

Client/Server Solutions PART V

The Win32 SDK contains a service sample with a flexible reusable component. This component, in the form of a C source file SERVICE.C, provides a simple yet robust implementation of a services main function and a number of convenience functions. It can be used virtually unaltered (with a few cosmetic changes in SERVICE.H) in most service projects.

MTS and the Three-Tiered Model CHAPTER 36

653

36

MTS and the Three-Tiered Model


IN THIS CHAPTER
s Dynamic HTML 654 s The Three-Tiered Client/Server Model 657 s The Microsoft Transaction Server 659

36
MTS AND THE THREE-TIERED MODEL

654

Client/Server Solutions PART V

With the emergence of the Internet and the increasing importance of corporate intranet solutions, Microsoft is busy positioning itself as a supplier of state-of-the-art client/server solutions. Accordingly, there is growing emphasis on client/server development in all Microsofts development products. Visual C++, and in particular the Microsoft Foundation Classes, have supported database solutions for some time. Through database access mechanisms such as ODBC and ADO, it has always been possible to implement the client side of simple client/server database systems. Now, however, there is more. Several of Microsofts technologies are rapidly converging toward an all-encompassing model of component-based client implementations operating in a three-tiered client/server environment. These technologies include s s s s Database access solutions ActiveX and specifically, downloadable components Dynamic HTML The Microsoft Transaction Server

These technologies, along with Microsofts Internet Information Server and other components, also form the basis of Microsofts new strategy for Web computing: the Distributed interNet Application Architecture, or DNA. Database programming and ActiveX have been discussed extensively elsewhere in this book. The present chapters focus is DHTML and MTS.

Dynamic HTML
Dynamic HTML is a significant extension to HTML technology. In the simplest terms, it extends programmatic control of a Web page to every component in that Web page. Listing 36.1 shows one of the simplest DHTML examples.

Listing 36.1. Dynamic HTML.


<HTML> <BODY> <H3 id=mytitle>Dynamic Content Example</H3> <P><input type=text size=20' value=Enter new text here id=newtitle> <input type=button value=Change Title onclick=(document.all.mytitle.innerText=document.all.newtitle.value)> </BODY> </HTML>

MTS and the Three-Tiered Model CHAPTER 36

655

Now you might say, this is great for Web page authors that they can create cool HTML pages like this, but what does this have to do with Windows programming? The answer is simple: DHTML is really object technology. A DHTML documents components are exposed and can be accessed through that trusty old object model, COM. This means literally complete control over a Web documents content and appearance. Listing 36.2 shows an example. This program invokes Internet Explorer through Automation and instructs it to open the HTML document shown in Listing 36.1. (Note that the location of the document, C:\TEMP\DHTML.HTM, is hard-coded and might have to be changed before you can run this program.) It then changes the title string much the same way as it was changed before when you clicked the Change Title button.

36
MTS AND THE THREE-TIERED MODEL

Listing 36.2. Controlling Dynamic HTML documents from C++.


#include <windows.h> #include <mshtml.h> #include <exdisp.h> void main(void) { CLSID clsid; LPUNKNOWN punk; IWebBrowser2 *pWB; IHTMLDocument2 *pHTML; IHTMLElementCollection *pElements; IHTMLElement *pItem; LPDISPATCH pdisp; DISPID dispid, dispidNamed = DISPID_PROPERTYPUT; DISPPARAMS dispparams; UINT uArgErr; VARIANTARG vArg; VARIANT_BOOL bBusy; VARIANT varName, varIndex, var; BSTR bstr; OleInitialize(NULL); CLSIDFromProgID(OLESTR(InternetExplorer.Application), &clsid); CoCreateInstance(clsid, NULL, CLSCTX_SERVER, IID_IUnknown, (LPVOID *)&punk); punk->QueryInterface(IID_IWebBrowser2, (LPVOID *)&pWB); punk->Release(); pWB->put_Visible(TRUE); BSTR bstrVal = SysAllocString(LC:\\TEMP\\DHTML.HTM); var.vt = VT_I4; var.lVal = 0; pWB->Navigate(bstrVal, &var, &var, &var, &var); SysFreeString(bstrVal); do { Sleep(500); pWB->get_Busy(&bBusy);

continues

656

Client/Server Solutions PART V

Listing 36.2. continued


} while (bBusy); pWB->get_Document(&pdisp); pWB->Release(); pdisp->QueryInterface(IID_IHTMLDocument2, (LPVOID *)&pHTML); pdisp->Release(); pHTML->get_all(&pElements); varName.vt = VT_BSTR; varName.bstrVal = SysAllocString(Lmytitle); varIndex.vt = VT_I4; varIndex.lVal = 0; pElements->item(varName, varIndex, &pdisp); SysFreeString(varName.bstrVal); pdisp->QueryInterface(IID_IHTMLElement, (LPVOID *)&pItem); pdisp->Release(); bstr = SysAllocString(LHello from C++!); pItem->put_innerText(bstr); SysFreeString(bstr); pItem->Release(); OleUninitialize(); }

How does this program work? It actually performs a sequence of steps that walks a series of interfaces, until the interface to the desired object is obtained. Specifically, it first obtains an IWebBrowser2 interface, which allows it to make the main Internet Explorer window visible (as any decent Automation server, Internet Explorer also starts out invisible when started through Automation.) Next, it uses the IWebBrowser2 interface to navigate to the desired URL (in this case, the filename). It waits until the navigation is complete (it would be more elegant to use events for this, but for our present purposes, that little do loop will suffice.) Finally, before it is released, the IWebBrowser2 interface is used to obtain an IDispatch interface to the HTML document object itself through get_Document. The HTML document object has a method, get_all, which obtains a collection of all elements in the document. You can then use the item method of this collection to find a specific element by name; in our case, we are seeking an element with the name mytitle.
IHTMLElement

After you have an IDispatch interface to the desired element, you can obtain a pointer to the interface, and through this interface you can change the elements text.

You see, theres no black magic here. Most of the apparent complexity in this example is just the usual paraphernalia associated with using COM from C/C++: a confusing mess of strange types and multiple interfaces. But in the end, all you really did can best be expressed with the following single line of pseudocode:
document.all.mytitle.innerText=Hello from C++!

Did I say pseudocode? But this is almost exactly the same line that was included in the HTML file itself, as the action associated with the Change Title button.

MTS and the Three-Tiered Model CHAPTER 36

657

This application can be compiled from the command line by typing cl dhtml.cpp ole32.lib oleaut32.lib. Note that you must have Internet Explorer 4 and the Internet Client SDK (distributed, for example, with recent versions of the Microsoft Platform SDK) installed. If you run this application, Internet Explorer will open the file DHTML.HTM and will immediately change its title text (see Figure 36.1).

36
MTS AND THE THREE-TIERED MODEL

FIGURE 36.1.
Displaying a DHTML document under Automation control.

Needless to say, this example merely scratched the surface of the possibilities that DHTML provides. Sophisticated applications can control all aspects of HTML elements including their text, style, or appearance. They can insert, remove, and reposition elements, completely rearranging the appearance of the page if necessary. The DHTML client program does not need to be a separate executable, and it does not have to use Automation to invoke a new copy of Internet Explorer. It could be an ActiveX control, for example, already resigning in a Web page and controlling the appearance of other elements of the same page. The control could remain completely hidden, while exercising control over every HTML document element.

The Three-Tiered Client/Server Model


A traditional client/server implementation involves two systems: the client where user interaction takes place and the server where shared resources are stored and managed and multiuser services are provided. The software components that comprise the client are assumed to be installed on the client computer; the model does not make provisions for storing or delivering these components. Unlike this traditional model, the three-tiered client/server model introduces an extra tier, the middle tier. This tier hosts software components (or, if you prefer analyst-speak, it hosts components that encapsulate business rules). These components can be in any form: scripts, controls, or executables. The middle tier is responsible for delivering these components to the client, but can also provide a conduit to data sources.

658

Client/Server Solutions PART V

The Tiers
The three-tiered model is flexible, and the roles of the three tiers are overlapping. For example, the middle tier can both run software components and deliver components to client workstations. The client might be limited in functionality to a Web browser, or it can run compiled executables or other components. Here is my attempt to provide a useful set of definitions for these tiers: s Client tier: Typically an end-user workstation that runs software components either stored locally or obtained from the middle tier, to access data sources and other services from the server tier either directly or through the middle tier. s Middle tier (also called application server tier): A server that hosts software components that can be downloaded by clients or executed on the middle tier server. s Server tier (also called data source tier): A server that provides data sources and other multiuser services that can be accessed by clients either directly or through the services of a middle tier. Needless to say, the three tiers do not have to be implemented as physically separate computers. All three can reside on the same computer, for example. So whats so special about having a middle tier? Isnt it just a glorified name to something that has been around since the first LANs, such as a file server where applications are installed? Well, such a file server could indeed be viewed as a middle tier implementation, but in modern architectures the middle tier can do a lot more. Consider, for example, a complex business transaction that involves several data sources. A properly constructed middle tier might ensure that multiple clients can access multiple data source servers simultaneously, while maintaining the integrity of all data sets. Another important distinction is that the middle tier delivers not just any software component; it manages and delivers components that specifically encapsulate the business rules of the application. In other words, its significance is not that it can deliver standard components, such as the Web browser executable, but that it can deliver components specifically tailored to the business task, such as a DHTML Web page or a Microsoft Transaction Server application component.

Sample Implementations
Lets not be too abstract here. The three-tiered model is not the pipe dream of some West Coast programmer intoxicated by the fair weather there. It is actually a fairly accurate description of how many Web-based systems work today. Figure 36.2 shows just such a system.

MTS and the Three-Tiered Model CHAPTER 36

659

FIGURE 36.2.
Web technology and the three-tiered model.

36
MTS AND THE THREE-TIERED MODEL

Client (Internet Explorer)

Application Server (IIS)

Data Source Server (SQL Server)

Client (Internet Explorer)

Application Server (IIS)

Data Source Server (SQL Server)

Client (Internet Explorer)

In this scenario, clients are personal computers running a Web browser (Internet Explorer). The software components that run on these clients are Web pages; these are obtained (perhaps even dynamically generated) from application servers that run a Web server, such as Microsofts Internet Information Server. After the components are downloaded to clients they are used to access data that resides on remote data source servers (running a multiuser RDBMS, such as SQL Server). The access to data sources can be facilitated by components such as data-enabled ActiveX controls. In this example, the middle tier plays a relatively passive role: IIS merely delivers Web pages to clients. In contrast, a much more active role can be played by an application component that runs, for example, in the context of the Microsoft Transaction Server.

The Microsoft Transaction Server


The Microsoft Transaction Server, or MTS, is a Windows NT Server component specifically designed to support scalable applications. What follows is a short review of some of the basic concepts behind this technology.

660

Client/Server Solutions PART V

Scalability Issues
Writing client/server applications is simple, as long as only a single server is involved. However, when size or performance considerations require that multiple servers be installed, the complexity of the task increases greatly. Specifically, the following issues must be considered: s Atomicity: Many business transactions must be either fully completed or rolled back; otherwise, the system might be left in an inconsistent state. s Consistency: The result of a transaction must be a valid state for the system. s Isolation: A transaction shall not see the partial, incomplete results of another transaction in progress. s Durability: After a transaction is committed its results shall be protected from typical failures (for example, a server process crash or a power failure). These issues are often difficult to address even when only a single server is involved. When multiple servers are used, the situation is a lot more complex because consistency must be maintained across servers. For example, if a transaction updates a table of payments on one server and a table of delivery schedules on another, these two updates together must form an atomic transaction. Using multiple redundant servers also provides unique challenges that are not encountered with single-server implementations. These include s Fault isolation: When a server fails, the effects of that failure shall be contained without bringing down the rest of the system. s Load balancing: When multiple servers perform the same function, processing load shall be distributed among them as evenly as possible to increase overall system performance. Clients shall not be required to be aware of the identities of individual servers. The Microsoft Transaction Server is designed to address most of these issues.

MTS Concepts
The operation of the Microsoft Transaction Server is based on the Component Object Model. The basic units of operation in MTS are application components. Application components are in-process COM servers that run within the environment of a server process. Such a server process is the Microsoft Transaction Server Executive, but other server processes can also host transaction server components. Server applications use resource managers to maintain durable data. Multiple resource managers are coordinated by the Microsoft Distributed Transaction Coordinator, or DTC. Examples for resource managers include ODBC 3.0 or SQL Server 6.5.

MTS and the Three-Tiered Model CHAPTER 36

661

Finally, resource dispensers manage nondurable, shared state. For example, the Shared Property Manager provides synchronized access to process-wide variables. Figure 36.3 provides a schematic illustration of the relationships of MTS components.

36
MTS AND THE THREE-TIERED MODEL

FIGURE 36.3.
MTS conceptual components.

Client

Client

Client

Client

Server Process

Server Process

Application Component

Application Application Component Component

Application Component

Application Application Component Component

Resource Dispenser

Resource Dispenser

Resource Dispenser

Resource Dispenser

Resource Manager

Resource Manager

Resource Manager

Resource Manager

When you install the MTS you actually install the DTC, the MTS Executive, and helper components such as the Microsoft Transaction Server Explorer, which lets you install and manage application components.

MTS Programming: An Overview


Simple MTS application components contain no MTS-specific code. Instances of these components are created and managed by MTS through standard COM interfaces. The MTS can automatically create a transaction or assign an existing transaction when an object is created, depending on the components transaction attribute. This attribute can be encoded in the components type library (recommended) or can be set at runtime through the MTS Explorer. The MTS API also provides a series of COM interfaces. For example, through the GetObjectContext function an application component can obtain an IObjectContext interface, through which it can examine its context. For each MTS object, MTS automatically creates a context object that encapsulates information about the current context. For example, the context object might contain information about the current client or identify the current transaction. Another series of COM interfaces can be used to manipulate shared properties. Shared properties are organized into groups which, in turn, are managed by the SharedPropertyGroupManager objects.

662

Client/Server Solutions PART V

Some of the COM interfaces are used by MTS client applications. For example, a client application can combine the work of several MTS objects into an atomic transaction. MTS application components are developed using any language capable of implementing COM interfaces: for example, Visual C++ or Visual Basic. The final step in the development process is creating a package. Packages are created through the MTS Explorer, and represent a collection of related components that are deployed together. Whereas the MTS API is used to develop MTS clients and application components, the far more complex MTS SDK can be used to create resource managers and resource dispensers.

Summary
Several new technologies assist the developer in creating large, scalable multiuser database applications on the Win32 platform. The basis of these technologies is the three-tier client/server model. This model separates the roles of the application server, which provides software components that encapsulate business rules, and the data source server, which provides multiuser access to shared data. An important technology used in three-tier implementations is Dynamic HTML. Dynamic HTML places the hitherto static content of Web pages under program control. It exposes all components of a Web page through COM interfaces. The controlling program can be a script that is part of the Web page, an ActiveX control, or even an external program that controls the Web browser through Automation. Another important technology is the Microsoft Transaction Server. This component of Windows NT Server is specifically designed to support scalable applications in large-scale environments. It provides solutions to the most common problems encountered when client/server applications are scaled to multiple servers. Specifically, it addresses the issues of atomicity, consistency, isolation, and durability, without an increase in complexity or robustness problems that are often encountered in these situations. MTS also uses COM as its underlying technology. MTS application components are inprocess COM servers that execute transactions. The MTS creates instances of these components and manages their context in accordance with configuration settings that are established through the MTS Explorer.

IN THIS PART
s Writing Messaging Applications with MAPI 665 s TCP/IP Programming with WinSock 679 s Using the WinInet API 699

VI
PART

s Telephony Applications with TAPI 711 s Named Pipes and Remote Procedure Calls 731

Networks and Communications

Writing Messaging Applications with MAPI CHAPTER 37

665

37

Writing Messaging Applications with MAPI


37
WRITING MESSAGING APPLICATIONS

IN THIS CHAPTER
s The MAPI Architecture 666 s MAPI APIs 670 s MAPI Support in MFC 677

666

Networks and Communications PART VI

Writing messaging-enabled applications is no longer the task of a few corporate programmers working on enterprise-wide system integration projects. The Microsoft Exchange client and MAPI services are now available on most Windows 95 desktops; and with the recent phenomenal growth of the Internet and intranets, users in increasing numbers demand messaging features in end-user applications. Solutions that were developed for corporate networks yesterday are used on home computers today. Microsoft recognized this fact when it decided to add the requirement for some level of MAPI support to its Windows 95 Logo Program. For applications to qualify under this program, it is now necessary for them to support, at the very minimum, a Send command that enables them to send a document to a MAPI recipient. (Obviously, if your application is conceptually different, such as a game program, MAPI support is not required.) The MAPI acronym stands for Messaging Applications Programming Interface. But what exactly is MAPI? How is MAPI implemented? And, most importantly, in what way can applications utilize MAPI services most efficiently? These are the questions that I attempt to answer in the present chapter. Be aware that the purpose of this chapter is not to provide an introduction to MAPI at the level required to develop MAPI providers. Such discussion would itself warrant a book. Instead, I am trying to present an overview of MAPI as an application interface, with special emphasis on MAPI usage from within MFC framework applications.

NOTE
Throughout this chapter, the MAPI architecture I describe corresponds to MAPI in Windows 95 or later, or in Windows NT version 3.51 or later. Earlier versions of Microsoft Mail, although they provide limited support for Simple MAPI and CMC, do not provide full Extended MAPI support, nor are they compatible with the architecture described in this chapter.

The MAPI Architecture


What exactly is MAPI? Is it the Microsoft Exchange client, later renamed Windows Messaging client (to eliminate the suggestion that it may only be a client to Microsofts Exchange Server mail system) that is supplied with Windows 95 or Windows NT 4.0? Is it the much discussed Exchange Server? Is it the MAPI spooler that is quietly running in the background whenever Exchange is started? The answer is none of the above. MAPI is not an application, a DLL, or a system service; rather, it is a series of specifications. Although Microsoft is the prime supplier of MAPI components, it is by no means necessary to have a single Microsoft component in order to use MAPI (nor is

Writing Messaging Applications with MAPI CHAPTER 37

667

it necessary to use a Windows or Win32 platform, actually). You can have a MAPI-compliant system with third-party message store, address book, and transport providers on a nonWindows operating system. MAPI is part of what Microsoft calls WOSA, the Windows Open Services Architecture, that consists of a common set of APIs for distributed computing. MAPI, in fact, can be viewed as two independent sets of APIs that link client applications on the one hand with service providers on the other (Figure 37.1).

FIGURE 37.1.
The MAPI architecture.

Messaging-aware applications

Messagingenabled

Messaging-based applications

37
WRITING MESSAGING APPLICATIONS

Simple MAPI

CMC

Extended MAPI

OLE messaging

Messaging Subsystem

Extended MAPI

Message store provider

Address book provider

Transport provider

Why several APIs? The reasons are partly historical. First, Microsoft released the specifications for Simple MAPI, which included about a dozen fundamental calls that enabled applications to use messaging. The Common Messaging Calls (CMC) API has been developed as a platform-independent replacement for Simple MAPI. It also contains about 10 fundamental calls that provide access to basic messaging services. In contrast, Extended MAPI is a large, complex, evolving specification that provides an application programming interface not only for client applications but for service providers as well. Active Messaging, an API specifically developed for Visual Basic and Visual C++ applications, uses the OLE Automation architecture. The next section offers a closer look at the MAPI architecture itself.

668

Networks and Communications PART VI

Types of MAPI Support


There are three distinct levels of MAPI support that applications can provide. The simplest MAPI applications are messaging-aware applications. These applications do not depend on the presence of MAPI to perform their functions; they merely provide some simple MAPI functionality. The Windows 95 WordPad, which provides a Send command in its File menu, is a good example of a messaging-aware application. In contrast, messaging-enabled applications require MAPI to be present in order to offer full functionality. While these applications may function when MAPI is not present, they cannot offer the full range of their services under these conditions. A perfect example for a messaging-enabled application is Microsoft Schedule+; while this program can be used as a standalone personal information manager, many of its capabilities do not make sense unless it is running on a system with MAPI installed. At the top of the hierarchy are messaging-based applications. Running these applications requires that the full range of MAPI services (message store, address book, transport) be present and available. These applications are also often referred to as workgroup applications. A good example of a workgroup application is the Microsoft Exchange client; I will spare you the explanation of why this application cannot be run on a standalone system with no MAPI support. The next section presents a brief tour of MAPI service providers; then we can examine how the various APIs are used in applications that provide these different levels of support.

Service Providers
MAPI service providers are the components that collectively implement MAPI service on a system. There are three distinct types of service providers: message stores, address books, and transports. Address books contain one or more lists of recipients. An address book provider implements a specific set of interfaces through which messaging applications or other providers can gain access to address entries or lists of addresses. Message stores are hierarchical depositories of MAPI messages. A message consists of a multitude of properties that include the sender of the message, the recipient, the date, the subject, the message body, and many other items. Message store providers implement some form of persistent storage for messages (for example, a local disk file) and provide a set of interfaces through which messaging applications and other providers can enumerate messages in the message store, retrieve specific message properties, or retrieve a set of messages. Transport providers represent the link between the local system and remote systems. Transport providers take outgoing messages, establish connections to remote systems, and transmit

Writing Messaging Applications with MAPI CHAPTER 37

669

messages in a format that the remote system can comprehend. Transport providers also accept incoming messages from the remote system, translate these into MAPI message objects as necessary, and place them into the local message store. A good example of a transport provider is the Microsoft Internet transport provider, which connects to remote systems using Windows Sockets and uses SMTP (Simple Mail Transfer Protocol) for outgoing and POP (Post Office Protocol) for incoming messages. Outgoing messages are translated into a readable ASCII form; incoming messages are translated from such form into MAPI message objects. All providers are represented in the form of DLLs. The set of DLLs that are to be loaded for a MAPI session is determined by MAPI profiles.

37
WRITING MESSAGING APPLICATIONS

MAPI Profiles
The glue that holds MAPI components together is the MAPI Spooler. The MAPI Spooler is a separate process that facilitates message receipt and delivery. It is typically the MAPI Spooler that passes submitted messages to transport providers and accepts incoming messages from them. The MAPI Spooler implements a store-and-forward architecture; that is, messages submitted are spooled by the spooler to the appropriate transport provider when it becomes available. This capability is vital on systems with large message volumes and many remote connections. Actually, it is also the MAPI Spooler that is responsible for loading and initializing service providers. But how does the spooler know which service providers to load? The answer is MAPI profiles. MAPI profiles are Registry entries that specify the current MAPI configuration. These Registry entries can be found under the following Registry key:
HKEY_CURRENT_USER\ Software\ Microsoft\ Windows Messaging Subsystem\ Profiles

Note that the MAPI profile Registry entries are generally not readable by human beings and should not be edited manually. There can be several profiles defined for a user. Each profile describes a series of service providers and their operating parameters. For example, a typical profile may include the Microsoft Personal Information Store (message store provider), the Microsoft Personal Address Book (address book provider), Microsoft Fax (transport provider), and the Microsoft Network Online Service (address book and transport). MAPI profiles can be edited using the Microsoft Exchange client or Microsoft Outlook; select the Services command from its Tools menu (Figure 37.2).

670

Networks and Communications PART VI

FIGURE 37.2.
Editing a MAPI profile.

MAPI APIs
The different APIs serve different types of MAPI applications. Messaging-aware and messaging-enabled applications can use both Simple MAPI and CMC. The main advantage of using CMC is complete platform independence. It is recommended that newer applications use CMC in place of the older, platform-dependent Simple MAPI. Messaging-based applications (not to mention service providers) require access to the full MAPI API set and thus should use Extended MAPI. Active Messaging is used by Visual Basic and Visual C++ applications. Active Messaging provides a richer API than CMC or Simple MAPI but falls short of the functionality of the full Extended MAPI set.

Simple MAPI
Simple MAPI provides a series of functions that enable applications to establish a MAPI session, perform messaging functions, and shut down the session. The simplest way to use Simple MAPI is via the MAPISendDocuments function. This function can be used to create a standard MAPI message with one or more file attachments. MAPISendDocuments always displays a dialog where the user can specify recipients, sending options, and the message text. This is demonstrated by the program in Listing 37.1. This program sends a message using MAPISendDocuments with the file c:\autoexec.bat embedded in it (note that the program fails if that file does not exist). You can compile this program from the command line by typing cl sendmsg.c user32.lib.

Listing 37.1. Using MAPISendDocuments.


#include <windows.h> #include <mapi.h> LPMAPISENDDOCUMENTS lpfnMAPISendDocuments;

Writing Messaging Applications with MAPI CHAPTER 37


void SendMsg(HWND hwnd) { (*lpfnMAPISendDocuments)((ULONG)hwnd, ;, C:\\AUTOEXEC.BAT, AUTOEXEC.BAT, 0); MessageBox(hwnd, Message sent, , MB_OK); } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_LBUTTONDOWN: SendMsg(hwnd); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; HANDLE hMAPILib; hMAPILib = LoadLibrary(MAPI32.DLL); lpfnMAPISendDocuments = (LPMAPISENDDOCUMENTS)GetProcAddress( hMAPILib, MAPISendDocuments); if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); FreeLibrary(hMAPILib); return msg.wParam; }

671

37
WRITING MESSAGING APPLICATIONS

672

Networks and Communications PART VI

A much more flexible way of sending messages is through MAPISendMail. Through a series of structures, you can specify the recipient and contents of the message. MAPISendMail makes it possible to send mail without presenting a user interface. Thus, MAPISendMail can also be used, for example, from command-line applications.
MAPISendMail is typically used within a MAPI session that is started by a call to MAPILogon and terminated by calling MAPILogoff. This usage of MAPISendMail is demonstrated by the program in Listing 37.2, which sends a simple text message to president@whitehouse.gov without displaying a user interface. (For the sake of the sanity of Mr. Clintons staff, do change the mail address before you start experimenting with this program!)

Listing 37.2. Using Simple MAPI in a console application.


#include <windows.h> #include <stdio.h> #include <mapi.h> LPMAPILOGON lpfnMAPILogon; LPMAPISENDMAIL lpfnMAPISendMail; LPMAPILOGOFF lpfnMAPILogoff; MapiRecipDesc recipient = { 0, MAPI_TO, Bill Clinton, SMTP:president@whitehouse.gov, 0, NULL }; MapiMessage message = { 0, Greetings, Hello, Mr. President!\n, NULL, NULL, NULL, 0, NULL, 1, &recipient, 0, NULL }; void main(void) { LHANDLE lhSession; HANDLE hMAPILib; hMAPILib = LoadLibrary(MAPI32.DLL); lpfnMAPILogon = (LPMAPILOGON)GetProcAddress(hMAPILib, MAPILogon); lpfnMAPISendMail = (LPMAPISENDMAIL)GetProcAddress(hMAPILib, MAPISendMail); lpfnMAPILogoff = (LPMAPILOGOFF)GetProcAddress(hMAPILib, MAPILogoff); (*lpfnMAPILogon)(0, NULL, NULL, 0, 0, &lhSession); (*lpfnMAPISendMail)(lhSession, 0, &message, 0, 0); (*lpfnMAPILogoff)(lhSession, 0, 0, 0); printf(Message to the White House sent.\n); FreeLibrary(hMAPILib); }

Writing Messaging Applications with MAPI CHAPTER 37

673

This program can be compiled from the command line without specifying any flags or libraries: cl cmdmsg.c. Simple MAPI can also be used to process incoming messages. Calls like MAPIFindNext and MAPIReadMail can be used, for example, to examine new messages in the users Inbox. Simple MAPI also offers a user interface for entering addresses. Instead of using MAPISendDocuments, you can utilize the MAPIAddress call to enable the user to select and enter addresses.

Common Messaging Calls


Common Messaging Calls is an implementation of the X.400 API Associations Common Messaging Call API. It provides a series of high-level messaging functions that can be used in a fashion similar to Simple MAPI. Table 37.1 compares CMC functions with Simple MAPI functions.

37
WRITING MESSAGING APPLICATIONS

Table 37.1. CMC and Simple MAPI. CMC Simple MAPI


cmc_logon cmc_logoff cmc_free cmc_send cmc_send_documents cmc_list cmc_read cmc_act_on MAPILogon MAPILogoff MAPIFreeBuffer MAPISendMail MAPISendDocuments MAPIFindNext MAPIReadMail MAPISaveMail MAPIDeleteMail cmc_look_up MAPIAddress MAPIDetails MAPIResolveName cmc_query_configuration

Description Log on to the service Log off from the service Free allocated memory Send a message Send files in a message Find messages Retrieve messages Save or delete messages

N/A

Addressing CMC configuration data

The program in Listing 37.3 demonstrates the use of CMC calls from within a console application. You can compile and run this program from the command line. To compile, use cl cmcmsg.c.

674

Networks and Communications PART VI

Listing 37.3. CMC in a console application.


#include <windows.h> #include <xcmc.h> typedef CMC_return_code (FAR PASCAL *LPFNCMCLOGON)(CMC_string, CMC_string, CMC_string, CMC_enum, CMC_ui_id, CMC_uint16, CMC_flags, CMC_session_id FAR *,CMC_extension FAR *); typedef CMC_return_code (FAR PASCAL *LPFNCMCSEND)(CMC_session_id, CMC_message FAR *, CMC_flags, CMC_ui_id,CMC_extension FAR *); typedef CMC_return_code (FAR PASCAL *LPFNCMCLOGOFF)(CMC_session_id, CMC_ui_id, CMC_flags, CMC_extension FAR *); LPFNCMCLOGON lpfnCMCLogon; LPFNCMCSEND lpfnCMCSend; LPFNCMCLOGOFF lpfnCMCLogoff; CMC_recipient recipient = { Bill Clinton, CMC_TYPE_INDIVIDUAL, SMTP:president@whitehouse.gov, CMC_ROLE_TO, CMC_RECIP_LAST_ELEMENT, NULL }; CMC_message message = { NULL, CMC: IPM, Greetings, {0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, Hello, Mr. President!\n, &recipient, NULL, CMC_MSG_LAST_ELEMENT, NULL }; void main(void) { char msg[1000]; CMC_session_id session; CMC_return_code retcode; HANDLE hMAPILib; hMAPILib = LoadLibrary(MAPI32.DLL); lpfnCMCLogon = (LPFNCMCLOGON)GetProcAddress(hMAPILib, cmc_logon); lpfnCMCSend = (LPFNCMCSEND)GetProcAddress(hMAPILib, cmc_send); lpfnCMCLogoff = (LPFNCMCLOGOFF)GetProcAddress(hMAPILib, cmc_logoff); (*lpfnCMCLogon)(NULL, NULL, NULL, (CMC_enum)0, 0, 100, CMC_ERROR_UI_ALLOWED | CMC_LOGON_UI_ALLOWED, &session, NULL); (*lpfnCMCSend)(session, &message, 0, 0, NULL); (*lpfnCMCLogoff)(session, 0, CMC_ERROR_UI_ALLOWED | CMC_LOGOFF_UI_ALLOWED, NULL); printf(Message to the White House sent.\n); FreeLibrary(hMAPILib); }

Writing Messaging Applications with MAPI CHAPTER 37

675

Extended MAPI
Extended MAPI is a large, complex object-oriented programming interface to all aspects of MAPI. Its programming model is based on the Component Object Model (COM). Extended MAPI defines a series of object types, including messages, message stores, folders, attachments, mail users, address books, providers, sessions, and many more supplementary objects. MAPI objects are characterized by a set of properties, themselves implemented as objects. MAPI objects are accessed through interfaces derived from the COM IUnknown interface. For example, property objects implement the IMAPIProp interface, while message objects implement the IMessage interface. Extended MAPI programming involves manipulating MAPI objects through their properties and methods and through event notifications. MAPI properties can be manipulated through property objects and the IMAPIProp interface, or through property tables. Methods are the IUnknown-derived interfaces that MAPI objects provide. Event notification is the mechanism of communication between MAPI objects. Extended MAPI can be used to implement messaging-based applications that require finer control over messaging services than what is available through Simple MAPI or CMC. Extended MAPI can also be used to perform administrative functions, such as selecting and configuring services and creating user profiles. Another use of Extended MAPI is to program with MAPI forms. MAPI forms represent special types of messages, implemented through a user interface and a form server, which manage the users interaction with the forms message. Extended MAPI can also be used to extend the Microsoft Exchange client. Customized versions of the Microsoft Exchange client can provide application-specific views on address books, message stores, and message objects, and can add new searching, addressing, and workgroup features to that application. Lastly, Extended MAPI also represents the programming interface for creating MAPI service providers. These include address books, message stores, and transports.

37
WRITING MESSAGING APPLICATIONS

Active Messaging
Active Messaging exposes MAPI objects through the automation interface. As such, it works best with automation clients, such as Visual Basic applications or programs written in Visual Basic for Applications (that is, Microsoft Excel, Access, or Project programs). Figure 37.3 illustrates the hierarchy of automation objects exposed by Active Messaging.

676

Networks and Communications PART VI

FIGURE 37.3.
Active Messaging objects.

Session object

Folder

Messages Collection Message

Recipients Collection Recipient

Address

Attachments Collection Attachment

Fields Collection

Field Folders Collection

Folder

Demonstrating the use of Active Messaging from within Visual C++ is difficult, as it requires constructing an automation client application. In order to demonstrate a simple Active Messaging session, I resorted to Visual Basic. The example shown in Listing 37.4 shows how a Visual Basic application can send a simple message without displaying a user interface.

Listing 37.4. Active Messaging example in Visual Basic.


Sub Command1_Click () Dim objSession As Object Dim objMessage As Object Dim objRecip As Object Set objSession = CreateObject(MAPI.Session) objSession.Logon , , False, False Set objMessage = objSession.Outbox.Messages.Add objMessage.Subject = Greetings objMessage.Text = Hello again, Mr. President!

Writing Messaging Applications with MAPI CHAPTER 37


Set objRecip = objMessage.Recipients.Add objRecip.Name = Bill Clinton objRecip.Address = SMTP:president@whitehouse.gov objRecip.Type = 1 objRecip.Resolve (False) objMessage.Send True, False MsgBox Message to the White House sent objSession.Logoff Exit Sub End Sub

677

To test this simple example, attach this code to a button in a Visual Basic form (you can also use disptest.exe, the automation test version of Visual Basic that is part of some Visual C++ installations), run the program, and click on the button. Oh, and dont forget to change the address before exercising this code! Note that this test will only work if you have Active Messaging installed on your computer. Active Messaging components are only available at present as part of the MAPI SDK, distributed through the Microsoft Developer Network Level 2 subscription, Professional level or above.

37
WRITING MESSAGING APPLICATIONS

MAPI Support in MFC


The Microsoft Foundation Classes Library provides MAPI support through a series of member functions in CDocument. This support, and how this support is incorporated into MFC framework applications, is the subject of the last part of this chapter.

MAPI Support in CDocument


The CDocument class supports MAPI through the OnFileSendMail member function.
CDocument::OnFileSendMail serializes the document into a temporary file, and then calls Simple MAPI functions to prepare and send a message with this file as an attachment.

A variant of CDocument::OnFileSendMail is COleDocument::OnFileSendMail; this version handles compound files correctly. The CDocument::OnUpdateFileSendMail member function determines whether MAPI support is available on the system and updates command items accordingly through a CCmdUI object.

MAPI and AppWizard


The AppWizard can generate skeleton applications with minimal MAPI support. If requested, the AppWizard will add a Send menu item to the new applications File menu. This item will invoke the OnFileSendMail member function of the applications document class. AppWizard will also install a command update handler, referencing the OnUpdateFileSendMail member function.

678

Networks and Communications PART VI

In other words, if you only wish to make your application messaging-aware, you need not do anything other than enabling MAPI support when creating the applications skeleton with AppWizard. This is all you need to do in order to satisfy the new Windows 95 Logo requirements.

Summary
MAPI, the Messaging Application Programming Interface, represents a comprehensive set of specifications that link messaging applications on the one hand and messaging service providers on the other, forming a messaging architecture. Messaging applications include messaging-aware, messaging-enabled, and messaging-based programs. Messaging-aware applications such as the Windows 95 WordPad, although they provide some level of MAPI functionality, do not depend on the presence of MAPI for their functionality. Messaging-enabled applications such as Microsoft Schedule+ require the presence of MAPI to perform many of their functions. Messaging-based applications such as the Microsoft Exchange client require MAPI in order to function. On the service provider side, MAPI recognizes message stores, address books, and transport providers. These providers, acting under the control of the MAPI Spooler, perform the services that together form the messaging system on your computer. The MAPI configuration is stored in the form of MAPI profiles. MAPI profiles are Registry entries that identify active MAPI services and their configuration. A user can have several MAPI profiles, with an active profile selected at the start of the MAPI session. When developing MAPI applications, you can pick one of several APIs. Simple MAPI provides a set of functions essential for addressing and sending simple messages, as well as retrieving incoming messages. Similar functionality is offered by CMC, the Common Messaging Call API; however, CMC is completely platform-independent. CMC is an implementation of the X.400 API Associations Common Messaging Call API. Visual Basic and Visual C++ applications can also use Active Messaging for an object-oriented interface to MAPI functions. The full interface to MAPI, Extended MAPI, is used for developing messaging-based applications, service providers, new message types (MAPI forms), administrative applications, and extensions to the Microsoft Exchange client application.

TCP/IP Programming with WinSock CHAPTER 38

679

TCP/IP Programming with WinSock

38

IN THIS CHAPTER
s TCP/IP Networks and OSI s The WinSock API 685 s A Simple WinSock Example 690 s Socket Programming and the Microsoft Foundation Classes 692 s Further Information 695 680

38
PROGRAMMING WITH WINSOCK TCP/IP

680

Networks and Communications PART VI

Until the recent information explosion that began a few years ago, not too many programmers had heard of TCP/IP, the protocol suite that is used throughout the Internet for internetwork communication (that is, communication between subnetworks). It was an obscure protocol a few years back, but no new operating system can claim to be complete without it nowadays; Windows is no exception. The most widely used application programming interface for TCP/IP programming is the Berkeley sockets library. Berkeley sockets have been used throughout the Internet for implementing TCP/IP applications. It was only natural that when the time came, Microsoft picked this API for TCP/IP under Windows. The use of Berkeley sockets is not limited to TCP/IP programming. Berkeley sockets represent an intertask communication mechanism that may use TCP/IP or other network protocols for the actual transmission of data. However, current WinSock implementations limit the use of the Berkeley socket interface to TCP/IP. First, heres a review of the fundamentals of TCP/IP networking.

TCP/IP Networks and OSI


I will spare you yet another brief account of the history of TCP/IP. The tale of how it evolved as the protocol suite of the ARPANET and later the NSFNET has been told and retold many times.

The Internet Protocol Suite


TCP/IP is somewhat of a misnomer; the internet protocol suite that it usually refers to consists of several components other than TCP and IP. Figure 38.1 provides a brief overview of these in the context of the seven layers of the Open Systems Interconnect (OSI) standard.

NOTE
In the phrase internet protocol, the term internet is used as a generic term and is written in lowercase. There can be many internets (for example, the internetwork at a large organization, often called an intranet nowadays) of which the Internet is but one, albeit the biggest by far.

What is behind these strange acronyms? TCP is the Transmission Control Protocol. TCP provides a reliable byte stream between two processes. Reliability in this context means that applications are not required to monitor for lost or corrupted packets or implement error correction. TCP guarantees delivery or notification of non-delivery. TCP is also a connection-oriented protocol. Applications that communicate via TCP establish a logical connection before any exchange of data can take place and terminate the connection when the data exchange

TCP/IP Programming with WinSock CHAPTER 38

681

has been completed. The term virtual circuit is sometimes used for this kind of a logical connection. UDP is the User Datagram Protocol. UDP is a connectionless protocol. Data packets are addressed and delivered individually. UDP is not a reliable protocol; applications must be prepared to handle the loss of packets. ICMP is the Internet Control Message Protocol. This protocol is often used to deliver error and control information on TCP/IP networks. ICMP packets are rarely used by user processes. A notable exception is the ping utility that is used to verify the accessibility of a foreign host by sending an ICMP echo packet and monitoring its return. IP stands for (what else?) Internet Protocol. IP provides the packet delivery service on which TCP, UDP, and ICMP rely. ARP is the Address Resolution Protocol. ARP is used to translate a network address to a hardware address. RARP is the Reverse Address Resolution Protocol. RARP is used to translate a hardware address into a network address.

38
PROGRAMMING WITH WINSOCK TCP/IP

FIGURE 38.1.
The internet protocol suite and the OSI layers.

7 Application

6 Presentation User program 5 Session User program

4 Transport TCP UDP

3 Network ICMP IP ARP RARP

2 Data Link Hardware 1 Physical

682

Networks and Communications PART VI

IP Datagrams
At the heart of TCP/IP data transmissions is the IP datagram. An IP datagram is a packet that encapsulates source and destination addresses, type of service information, user data, and error correction information. An IP datagram consists of a header and a block of data. The data can be anything depending on the type of service and the user requirements. The header, on the other hand, contains a set of well-defined fields; the next section covers them.

IP Headers
The header of an IP datagram, or IP header, usually consists of 20 bytes. Figure 38.2 shows the fields comprising this IP header.

FIGURE 38.2.
The IP header.
Version (4 bits) Header Length (4 bits) Type of Service (8 bits)

Packet Length (16 bits)

Packet Identifier (16 bits)

Fragmentation Data (16 bits)

Time to Live (8 bits) Header Checksum (16 bits)

Protocol (bits)

Source Address (32 bits)

Destination Address (32 bits)

Of the fields shown in Figure 38.2, the ones that are of considerable interest to us include the Protocol, Source Address, and Destination Address fields. The Protocol field determines how the rest of the IP packet is interpreted. Several dozen values have been defined for this field. For example, a value of 6 implies a TCP packet; a value of 17 implies a UDP packet. The addresses in a packet represent host addresses. Host addresses uniquely identify hosts on an internet.

TCP/IP Programming with WinSock CHAPTER 38

683

IP Host Addresses and Routing


An IP host address is a 32-bit number uniquely identifying an internet host. Or, to be more precise, an IP address uniquely identifies a particular interface on an internet host; gateway machines (those which have interfaces on more than one network) routinely have several host addresses. Actually, even ordinary desktop computers often do; for example, your desktop system might have a host address for its Ethernet interface and another for its dial-up connection to its Internet service provider. According to convention, the 4-byte internet host address is usually written down as a set of four decimal numbers: for example, 127.0.0.1. The internet host address is usually divided into two parts: the network address and the actual host address. The respective lengths of these two portions of the address vary depending on the most significant byte in the address. Class A network addresses are those with the most significant byte between 0 and 127. A class A address consists of an 8-bit network address and a 24-bit host address. Because addresses beginning with 0 and 127 are reserved, there can be a maximum of 126 class A subnets on an internet. A class A subnet can contain up to 16,777,214 hosts. (Addresses in the form of nnn.0.0.0 and nnn.255.255.255 are reserved.) Class B network addresses are those with the most significant byte between 128 and 191. Class B addresses consist of a 16-bit network address and a 16-bit host address. There can be a maximum of 16,383 class B subnets. A class B subnet may contain up to 65,534 hosts. (Again, addresses in the form of nnn.mmm.0.0 and nnn.mmm.255.255 are reserved.) A class C network address is one with the most significant byte between 192 and 223. This allows a total of 2,097,152 class C subnets. A class C subnet may contain up to 254 hosts. (Addresses in the form of nnn.mmm.kkk.0 and nnn.mmm.kkk.255 are reserved.) Class D addresses (most significant byte between 224 and 255) are reserved for IP multicasting. This limited form of IP broadcasting is of no concern for most WinSock programmers. Once the destination address of a packet is known, hosts on an internet can route the packet as appropriate. For most hosts, routing involves either forwarding a packet to a destination that is known to the host or passing it along a default route. For example, if you have a class C subnet that is connected to an Internet service provider via a SLIP or PPP connection, this connection is your default route; any packets that are not addressed to a host on your subnet will be automatically forwarded to your provider. Larger hosts maintain dynamically updated routing tables. Through these tables, the correct route of packets can be identified and in the case of a network error, alternate routes can be found. Inasmuch as the WinSock programmer is concerned, routing is a transparent activity. You need not be concerned how the Internet accomplishes its magic when delivering a packet to a World Wide Web host in New Zealand or an FTP site in Alaska.

38
PROGRAMMING WITH WINSOCK TCP/IP

684

Networks and Communications PART VI

Hostnames
It was recognized early during the evolution of the Internet that numeric host addresses may not always be adequate. For one thing, they are difficult to remember; also, if a hosts address changes for any reason, the likely result would be endless confusion. Therefore, a naming system was introduced that mapped numeric IP addresses to memorizable hostnames and vice versa. In those early days of few hosts on the Internet, every host maintained a file (the hosts file) with a list of all Internet hosts and their addresses. However, as the Internet began its phenomenal growth, this solution quickly proved to be inadequate. First, the naming of Internet hosts had to be standardized; second, a mechanism needed to be found that would eliminate the need to maintain a copy of the hosts file on every Internet-connected computer. The answer to these problems is the Domain Name System (DNS), development of which began in the early 1980s. DNS is a hierarchical naming system. The format of DNS hostnames is familiar to everyone who has ever used the Internet: a typical Internet hostname is in the form of host.subdomain.domain. On top of the hierarchy is the root domain, denoted by a single period. Next are the top-level domains. Top-level domain names have been assigned either by organization (mostly in the United States) or by country name. The most widely used organizational top-level domains include the following:
GOV:

Government bodies EDU: Educational institutions COM: Commercial enterprises MIL: Military organizations ORG: Other organizations Top-level domains by country usually follow the ISO 3166 standard for two-letter country name abbreviations. Examples include US (USA), CA (Canada), DE (Germany). How is a domain name translated into a numeric IP address and vice versa? DNS defines a distributed name service mechanism whereby hosts can act as name servers for various domains. Name servers constantly communicate with each other, supplying each other with information on specific hosts. Without going into too much detail, suffice it to say that when your application requests the IP address for a hostname, this request is sent to your default name server, which, in turn, may query other name servers on the Internet to get the requested information. Again, the mechanism is completely transparent to the application programmer.

TCP and UDP Packets, Port Numbers, and Sockets


Clearly, knowing the destination host address of an IP packet is not sufficient for most applications. After all, a host can maintain several open connections, be engaged in several

TCP/IP Programming with WinSock CHAPTER 38

685

TCP/IP conversations at once. When an IP packet is received, how does the host determine which particular conversation this packet would be part of? In the case of both TCP and UDP packets, in addition to the IP header, the packets also contain additional header information. The first 4 bytes of both TCP and UDP headers contain a 2-byte source and a 2-byte destination port number. The port number, together with the IP number, identifies a socket that is unique throughout an internet. A pair of sockets uniquely identifies a connection (at least this is the case of TCP, which is a connection-oriented protocol).

NOTE
TCP and UDP ports are not equivalent. For example, TCP port 25 and UDP port 25 refer to two different entities on the same computer.

Internet Services
There are many types of services on IP networks that are in widespread use. Examples include FTP, Telnet, Gopher, the WWW, archie, DNS name service, whois, finger, and many others. The protocols used for these services are defined in the Internet Requests For Comment documents, or RFCs (some of the most relevant RFCs are listed later in this chapter). The services themselves are usually available on well-known port numbers. For example, to connect to a Telnet server on any host on the Internet, you would make a connection attempt to TCP port 23 on that machine. Well-known ports that a host recognizes are typically identified in the systems services file. This file is used in UNIX as well as Windows TCP/IP implementations. On most systems, TCP and UDP port numbers 0-1023 are reserved for privileged processes. For example, on a multiuser server an ordinary user cant start a process that listens to TCP port 23, and thus hijack all incoming telnet sessions.

38
PROGRAMMING WITH WINSOCK TCP/IP

The WinSock API


The Berkeley sockets interface can be used for communicating using both connection-oriented protocols (TCP) and connectionless protocols (UDP). The programming model is the clientserver model; servers wait for incoming requests, while clients initiate sessions. There are some implementation differences between WinSock and the UNIX version of Berkeley sockets. Of these, perhaps the most significant is the fact that socket descriptors and file descriptors cannot be used interchangeably. This has a notable effect when porting applications that make an assumption of this equivalence. Another difference is that the WinSock library must be initialized. Applications that intend to use WinSock functions must first call the WSAStartup function; when their work with the WinSock library is finished, they should call WSACleanup for proper termination.

686

Networks and Communications PART VI

The WinSock API also introduces several WinSock-specific functions for performing asynchronous I/O on sockets in a manner consistent with the messaging architecture of Windows. This assists in the development of responsive GUI WinSock applications.

WinSock Initialization
The WinSock library is initialized by a call to WSAStartup. The application calling this function provides the address to a WSADATA structure, which will hold initialization information. During the call to WSAStartup, applications and the WinSock library negotiate a version number. The initialization request fails if there is no overlap between the version number supported by the application and the version number supported by the WinSock library. If an error occurs, WSAStartup returns a nonzero value. Applications may retrieve extended error information through the function WSAGetLastError.

Creating and Using Sockets


A socket is created by a call to the socket function. Parameters to this function indicate the type of the socket, the type of the network address, and the protocol being used. For example, the call
socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)

creates a TCP socket. To associate a socket with an actual host address and port number, applications typically call the bind function. In addition to the socket identifier, or socket descriptor (which is returned by the socket call), bind takes a parameter that is a pointer to a structure describing the socket address. This structure is defined as follows:
struct sockaddr { u_short sa_family; char sa_data[14]; };

The sa_family member of this structure specifies the type of the address. For internet addresses, this value is set to AF_INET. The sa_data contains the actual address. Applications can easily access the various components of the address by referring to it through a sockaddr_in structure (instead of using sockaddr). Here is the definition of this structure:
struct sockaddr_in { short sin_family; u_short sin_port; struct in_addr sin_addr; char sin_zero[8]; };

Of these members, sin_port is the 16-bit port number, while sin_addr is the 32-bit host address.

TCP/IP Programming with WinSock CHAPTER 38

687

Name Service
In order to assign meaningful values to the host address field in the sockaddr_in structure, you must first obtain a 32-bit host address. To obtain an address when the symbolic name of a host is known, use the gethostbyname function. When calling gethostbyname, applications pass the symbolic name of the host and receive a pointer to a hostent structure in return. The hostent structure is defined as follows:
struct hostent { char FAR * h_name; char FAR * FAR * h_aliases; short h_addrtype; short h_length; char FAR * FAR * h_addr_list; };

This structure is necessary because a hostname may be associated with several host addresses (the reverse is also true). In most cases, applications just take the first (often only) address in h_addr_list; to make this easier, the symbol h_addr is defined as h_addr_list[0]. If an application wants to use a numeric address instead, it can use the inet_addr function to convert a string containing a numeric address into a 32-bit address value. Once that value has been obtained, gethostbyaddr can be used to return a hostent structure for the specified host.

38
PROGRAMMING WITH WINSOCK TCP/IP

Byte Ordering
A problem that is particularly important when it comes to designing applications that are expected to work on hybrid networks is the issue of byte ordering, or the order in which individual bytes, representing a multibyte value, are stored in memory. Some system architectures are big-endian (most significant byte comes first); others are little-endian (most significant byte comes last). Examples of the latter include the Intel family of processors and DEC CPUs; examples of the former include the Motorola 68000 processor family. Internet numbers (for example, host addresses) are always big-endian. To ensure correct conversion between machine-independent internet numbers and their machine-dependent representation, you can use the following set of functions: htonl, htons, ntohl, and ntohs. These functions connect short or long integers from network to host format or vice versa. Note that these functions may be implemented as macros.

Communication Through Sockets


In the case of the connection-oriented TCP protocol, the server application binds to a specific TCP port and then uses the listen function to indicate its willingness to accept incoming connection requests. Immediately after the call to listen, the server calls accept to wait for incoming connections. When accept returns, it provides the address of the peer process.

688

Networks and Communications PART VI

The client, after creating a socket using the socket call, can immediately initiate a connection using the connect call. It is not necessary to bind the socket to a specific port using bind prior to calling connect. After the connection has been successfully initiated, both the client and the server can use the call to transmit data and the recv call to receive data. The semantics of send and recv are similar to the semantics of the read and write calls for low-level file I/O. Indeed, on UNIX systems it is possible to use the latter pair of functions to perform I/O on sockets. Unfortunately, as I mentioned earlier, this is not possible with WinSock, due to the differences between a file descriptor and a socket descriptor. For the same reason, it is not possible to use the close system call to close a socket; applications must use closesocket when a socket connection is about to be terminated. Either the client or the server can terminate the connection using closesocket.
send

Figure 38.3 provides an overview of setting up and using a TCP connection.

FIGURE 38.3.
Socket communications for connection-oriented protocols.
Creates a socket Binds socket to a specific port

Server
socket ()

Client

bind ()

Indicates willingness to accept incoming connection requests Waits for incoming requests

listen ()

accept ()

socket ()

Creates a socket

connect ()

Initiates a connection

Receives data

recv ()

send ()

Sends data

Sends data

send ()

recv ()

Receives data

Terminates the connection

closesocket ()

closesocket

Terminates the connection

In the case of the connectionless UDP protocol, the sequence of events is somewhat different; the activities of the client and server application are more symmetrical. In this case, both the client and the server create their respective sockets and bind those to specific port numbers. The server then makes a call to the recvfrom function, which waits for any incoming data. The client, in turn, uses the sendto call to send data to a specific address. When this data is received by the server, the recvfrom call returns, and the server also obtains the address where the data

TCP/IP Programming with WinSock CHAPTER 38

689

has been received from. It can then use this address in a subsequent call to sendto, as it replies to the client. Figure 38.4 presents an overview of this process.

FIGURE 38.4.
Socket communications for connectionless protocols.
Creates a socket

Server
socket ()

Client

Binds socket to a specific port Receives data and gets peer address

bind ()

recvfrom ()

socket ()

Creates a socket

bind ()

Binds socket to a specific port Sends data to a specific address Receives data and gets peer address

38
PROGRAMMING WITH WINSOCK TCP/IP

send to ()

recvfrom ()

Sends data to a specific address

sendto ()

The Blocking Problem and the select Call


In the simplistic models shown in Figures 38.3 and 38.4, both the client and the server use blocking calls when waiting for data. A blocking call does not return to the calling function until the requested data becomes available. In other words, the application that makes such a call becomes suspended until the call is completed. While this model will suffice in many simple situations, it is clearly unacceptable for interactive applications. Such an application (for example, a telnet client) cannot simply freeze until data becomes available from the server. The solution employed by most UNIX TCP/IP applications relies on the select system call. This call makes it possible to wait on multiple (file or socket) descriptors. This way, a UNIX process can easily wait for data on both a socket or the standard input and spring into action whenever data is received on either of them. Unfortunately, things are not this easy with WinSock. Remember that socket and file descriptors are not interchangeable? The select function is no exception; it can only wait on multiple socket descriptors, not on a mix of socket and file descriptors.

690

Networks and Communications PART VI

While it is possible to monitor a socket while polling, this is not a very efficient solution. Fortunately, Win32s multithreading capability comes to our rescue. A process can easily start additional threads and have a separate thread for each input source. This mechanism works well for both command-line and graphics TCP/IP utilities. Nevertheless, the WinSock library offers yet another family of functions that assist in writing well-behaved TCP/IP applications without having to resort to multithreaded trickery. These asynchronous socket calls were presumably developed for the benefit of those who worked with 16-bit programs and programs intended for Win32s, but they can be put to good use in Win32 applications as well.

Asynchronous Socket Calls


Asynchronous socket calls rely on the Windows message-passing mechanism to communicate socket events to Windows applications. At the center of this mechanism is the WSAAsyncSelect function call. Through this function, an application can wait for a combination of socket events. Applications may receive notifications indicating readiness for reading, writing, incoming and completed connections, and socket closure. (This call can also be used for notifications regarding out of band data, something that has not been discussed in this section.) The notification takes place in the form of a userdefined message that is posted to a window, also defined in the call to WSAAsyncSelect. posts a single message for every event the application expressed an interest in. Once the message has been posted, no further messages will be posted for the same event until the application implicitly resets the event by calling the appropriate socket library function. For example, if a notification for incoming data was posted, no further such notifications will be posted for the given socket until the application retrieves that data with a call to recv or recvfrom.
WSAAsyncSelect

Other asynchronous socket functions include, for example, asynchronous versions of the standard Berkeley gethostbyname and gethostbyaddr calls: WSAAsyncGetHostByName and WSAAsyncGetHostByAddr. WinSock applications can also influence the blocking mechanism used in the standard Berkeley-style calls by using the WSASetBlockingHook function.

A Simple WinSock Example


It is time to put all this theory into practice. I decided to include a simple WinSock-based command-line utility. This program connects to a well-known service (the time service on TCP port 37) on the specified host and retrieves the current time in machine readable form, which it then formats and displays. With little modification, this application could be used to set the system time from a server that is known to keep reliable time. The program shown in Listing 38.1 can be compiled from the command line by typing cl
gettime.cpp wsock32.lib.

TCP/IP Programming with WinSock CHAPTER 38

691

Listing 38.1. The gettime command-line utility.


#include <iostream.h> #include <time.h> #include <winsock.h> void main(int argc, char *argv[]) { time_t t; int s; struct sockaddr_in a; struct hostent *h; WSADATA wsaData; if (argc != 2) { cout << Usage: << argv[0] << host\n; exit(1); } if(WSAStartup(0x101, &wsaData)) { cout << Unable to initialize WinSock library.\n; exit(1); } h = gethostbyname(argv[1]); if (h == NULL) { cout << Cannot resolve hostname\n; WSACleanup(); exit(1); } a.sin_family = AF_INET; a.sin_port = htons(37); memcpy(&(a.sin_addr.s_addr), h->h_addr, sizeof(int)); s = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (s == 0) { cout << Cannot establish connection: << WSAGetLastError() << \n; WSACleanup(); exit(1); } if (connect(s, (struct sockaddr *)&a, sizeof(a))) { cout << Cannot establish connection: << WSAGetLastError() << \n; WSACleanup(); exit(1); } if (recv(s, (char *)&t, 4, 0) != 4) cout << Unable to obtain time.\n; else { t = ntohl(t) - 2208988800; cout << asctime(localtime(&t)); } closesocket(s); WSACleanup(); }

38
PROGRAMMING WITH WINSOCK TCP/IP

692

Networks and Communications PART VI

The time service is described in RFC868. An Internet time server listens to TCP port 37; when it receives a connection request, it sends the current time (GMT) as a 4-byte number representing the number of seconds elapsed since January 1, 1900. The gettime application in Listing 38.1 requires a single command-line parameter, the name of the host to which it should make a connection attempt. After verifying that this parameter has been supplied, it makes an attempt to initialize the WinSock library. Next, the hostname supplied by the user is converted to a host address. (No attempt is made to resolve numeric addresses that the user might supply.) Next, a socket is created, and a connection attempt is made to the designated host on TCP port 37. If the connection is successful, the program tries to receive exactly four bytes of data. When the receive operation is complete, the data is converted (ntohl); the time base is corrected (Win32 time functions use January 1, 1970 as the base date for time variablesthe number of seconds from January 1, 1900 and January 1, 1970 is 2,208,988,800); and finally, the time is displayed as a human-readable local time on standard output:
C:\>gettime mud2.com Mon Feb 24 02:48:18 1997 C:\>

Before exiting, the program closes the socket and terminates the session with the WinSock library.

Socket Programming and the Microsoft Foundation Classes


The Microsoft Foundation Classes library encapsulates socket functionality in the CAsyncSocket class. Member functions of CAsyncSocket correspond to many of the standard and asynchronous functions in the WinSock library. An application wanting to use CAsyncSocket must first initialize the socket library via a call to AfxSocketInit; there is no need to call a corresponding cleanup function. Typically, applications would derive a class from CAsyncSocket and override several callback functions that facilitate asynchronous operations.

CAsyncSocket Example
To see how CAsyncSocket works, we can rewrite our gettime application using MFC. This simple program communicates with users using the dialog shown in Figure 38.5.

TCP/IP Programming with WinSock CHAPTER 38

693

FIGURE 38.5.
MFC implementation of the gettime program.

The skeleton for this application can be created through the MFC AppWizard. Create a new project with the name GT. Specify that its user interface be dialog-based. In Step 2, make sure that you check Windows Sockets support. Set the dialogs title to a meaningful text string, such as TCP/IP GetTime. After AppWizard has finished creating the skeleton, the first step is to update the applications dialog. Add the fields shown in Figure 38.5. The name of the text field holding the hostname should be IDC_HOST; the name of the text field holding the time should be IDC_TIME. The readonly attribute for IDC_TIME should be set. The Connect buttons identifier should be IDC_CONNECT; the Close button can keep the IDCANCEL identifier. Next, add three member variables using the ClassWizard. Two string variables (m_sHost and m_sTime) will correspond to the two text fields; the third variable, m_cConnect, will be of type CButton and correspond to the Connect button. While using the ClassWizard, add a message handler for the IDC_CONNECT button. This handler, CGTDlg::OnConnect (see Listing 38.2), should process BN_CLICKED messages. It is in this handler function where we add most socket-related code.

38
PROGRAMMING WITH WINSOCK TCP/IP

Listing 38.2. MFC gettime: The CGTDlg::OnConnect function.


void CGTDlg::OnConnect() { CGTSocket *pSocket; pSocket = new CGTSocket(this); UpdateData(TRUE); m_cConnect.EnableWindow(FALSE); pSocket->Create(); pSocket->AsyncSelect(FD_READ | FD_CLOSE); pSocket->Connect(m_sHost, 37); }

To implement our time retrieval functionality, we need to derive a class from CAsyncSocket and create an object of this new type when the user clicks on the Connect button. The code that you need to add to the dialog implementation file GTDlg.cpp is shown in Listing 38.3. Note that this code must precede the implementation of CGTDlg::OnConnect; otherwise, the compilation will fail, as CGTDlg::OnConnect attempts to declare an object of type CGTSocket that has not yet been declared.

694

Networks and Communications PART VI

Listing 38.3. MFC gettime: The CGTSocket class.


class CGTSocket : public CAsyncSocket { public: CGTSocket(CGTDlg *pDlg) {m_pDlg = pDlg;}; virtual void OnReceive(int nErrorCode); virtual void OnClose(int nErrorCode); CGTDlg *m_pDlg; time_t t; }; void CGTSocket::OnReceive(int nErrorCode) { Receive(&t, 4); t = ntohl(t) - 2208988800; m_pDlg->m_sTime = asctime(localtime(&t)); m_pDlg->UpdateData(FALSE); } void CGTSocket::OnClose(int nErrorCode) { m_pDlg->m_cConnect.EnableWindow(TRUE); delete this; }

In the function CGTDlg::OnConnect, a new object of type CGTSocket is created. This type is derived from CAsyncSocket. The actual socket is created through a call to the Create member function, which is followed by a call to AsyncSelect. This call enables the callback functions OnReceive and OnClose. At this time, the Connect button is also disabled. Finally, the Connect member function is called to initiate a connection to the destination. When the connection attempt succeeds, the time server immediately sends the time in the form of four data bytes. This triggers execution of the OnReceive function, which reads this data using Receive and uses the result to update the time field in the dialog. After sending the data, the server closes the connection; this results in a call to OnClose, which re-enables the Connect button in the dialog and destroys the CGTSocket object. Note that in its present form, this program does not perform any error checking and is likely to fail if a network error occurs.

Synchronous Operations and Serialization


The purpose of the CAsyncSocket class is to provide a low-level interface to the WinSock library. In contrast, the CSocket class, which is derived from CAsyncSocket, provides somewhat higher-level functionality. Unlike CAsyncSocket, CSocket provides blocking. Its member functions do not return until a requested operation has been completed.

TCP/IP Programming with WinSock CHAPTER 38

695

NOTE
The callback functions OnConnect and OnSend are never called for CSocket objects.

One particular use of CSocket objects is in conjunction with CFileSocket objects to enable the MFC serialization functions to work on sockets. A CFileSocket object can be attached to a CArchive and a CSocket; afterwards, data can be sent and received by simply using MFC serialization.

Further Information
TCP/IP programming is a very broad subject with a tremendous amount of literature. This chapter is certainly not sufficient to do this complex topic full justice. Because of this, I have included some references that I hope will assist you in finding the information you need. For WinSock, the authoritative reference is Microsofts WinSock specification, which is published on the Microsoft Developer Network Library CD-ROM. There are several good books on the subject. One particularly useful book is Uyless Blacks TCP/IP & Related Protocols (McGraw Hill, 1994). For more information on TCP/IP programming (and, in particular, programming with Berkeley sockets), see UNIX Network Programming by W. Richard Stevens (Prentice Hall, 1990). Although the book (as its title implies) is aimed at the UNIX programmer, much of the discussion is not UNIX-specific, simply because the respective APIs are vendor-independent. The official definitions of most Internet-related standards and protocols are found in the form of RFCs (Requests For Comment). These RFCs are available online. On the World Wide Web, use the following URL: http://www.cis.ohio-state.edu/htbin/rfc/rfc-index.html. RFCs can also be requested by Internet e-mail from the InterNIC Directory and Database Services mail server. Send a message to mailserv@ds.internic.net with the following command in the message body:
document-by-name rfcNNNN

38
PROGRAMMING WITH WINSOCK TCP/IP

To obtain the RFC index, use this command:


document-by-name rfc-index

Several documents can be requested in a single mail if you separate their names with commas or include several document-by-name commands in the message body in separate lines. The RFCs have also been published in CD-ROM format by a number of CD-ROM publishers (the one I use often is the Standards CD-ROM from InfoMagic). If you intend to do serious TCP/IP programming, I strongly recommend that you acquire a low-cost CD-ROM like this one.

696

Networks and Communications PART VI

Internet RFCs go back a long time. The earliest RFC on my CD-ROM, RFC0003, dates back to April 1969! Of course, many of the RFCs have long become obsolete. RFCs that define the standards most commonly encountered by the WinSock programmer are as follows: RFC0768, User Datagram Protocol (J. Postel, 1980) RFC0791, Internet Protocol (J. Postel, 1981) RFC0792, Internet Control Message Protocol (J. Postel, 1981) RFC0793, Transmission Control Protocol (J. Postel, 1981) RFC0826, Ethernet Address Resolution Protocol: Or Converting Network Protocol Addresses to 48.bit Ethernet Address for Transmission on Ethernet Hardware (D. Plummer, 1982) RFC0903, Reverse Address Resolution Protocol (R. Finlayson, T. Mann, J. Mogul, M. Theimer, 1984) RFC1034, Domain namesConcepts and Facilities (P. Mockapetris, 1987) RFC1035, Domain namesImplementation and Specification (P. Mockapetris, 1987) Two additional RFCs deserve mentioning. RFC1700 (Assigned Numbers, J. Reynolds, J. Postel, 1994) contains all well-known numbers including protocol identifiers, port numbers, and the like. RFC2200 (Internet Official Protocol Standards, J. Postel, 1997) is an invaluable reference to all the other RFC standards.

NOTE:
A popular misconception is that any material published in the form of an Internet RFC represents a standard. This is not so. In addition to standards, Internet RFCs may contain Informational, Experimental, Standards Track (Proposed Standard, Draft Standard, Internet Standard), or Historic material. The nature of an RFC is usually noted on the front page.

By the time you read this, the RFCs listed here might already have been replaced by newer versions. When you query an up-to-date RFC index, it usually indicates when an RFC has become obsolete and, if so, what newer RFCs might serve as its replacement.

Summary
TCP/IP is the name most often used to refer to the internet protocol suite. This protocol suite, which consists of protocols such as TCP, UDP, ICMP, IP, ARP, and RARP, provides the fundamentals for internetwork communication on the global Internet.

TCP/IP Programming with WinSock CHAPTER 38

697

Under Windows, TCP/IP programming is accomplished using the WinSock library. WinSock is an API that closely mimics the Berkeley sockets API used on many UNIX systems. However, there are some differences arising from the fact that Windows and UNIX are very dissimilar architecturally. In particular, the Berkeley sockets library uses socket descriptors that are interchangeable with file descriptors used in low-level I/O operations. For this reason, standard C low-level I/O functions can be used on Berkeley sockets, something that is not true under Windows. Berkeley sockets can be used to communicate using both connection-oriented protocols, such as TCP, and connectionless protocols, such as TCP/IP. In both cases, fundamental operations include the creation of a socket, binding a socket to a host address and port number, and sending and receiving data. Host addresses can be obtained using the gethostbyname function that resolves symbolic system names into 4-byte TCP/IP addresses. To ensure machine-independent representation of host addresses and other numeric quantities, a set of functions is provided for the translation to and from host byte ordering and internet byte ordering. Writing responsive applications requires that an application does not remain suspended indefinitely while waiting for socket input. The most generally used Berkeley sockets mechanism, namely the use of the select function, has limited utility under Windows because socket and file descriptors are not interchangeable; hence select cannot be used to wait for input on stdin, for example. However, Win32 applications can use multiple threads to furnish data exchange on multiple I/O channels simultaneously. To facilitate the development of responsive applications without relying on the multithreaded capabilities of Windows 95 or Windows NT, applications can use asynchronous socket calls. These calls rely on Windows messaging to deliver information about incoming data or other socket events to the application. Socket functionality is encapsulated in the CAsyncSocket class in the Microsoft Foundation Classes library. With the help of the CSocket class, which is derived from CAsyncSocket and provides synchronous socket I/O, and the CSocketFile class, applications can use MFC serialization functions on a socket interface.

38
PROGRAMMING WITH WINSOCK TCP/IP

698

Networks and Communications PART VI

Using the WinInet API CHAPTER 39

699

Using the WinInet API

39

IN THIS CHAPTER
s Internet Protocols 700 s The WinInet Library 705

39
USING THE WININET API

700

Networks and Communications PART VI

The new MFC classes that provide extensive support for HTTP, FTP, and Gopher operations over the Internet all rely on a recent extension to the Win32 API. This extension, referred to either as Win32 Internet Functions or the WinInet library, contains a series of functions that greatly simplify the task of connecting to a WWW, Gopher, or FTP server and transferring data. The WinInet library is initially distributed by Microsoft as an add-on component to Windows 95 and Windows NT. The library is implemented in the form of the file WININET.DLL, which is redistributable by developers. Eventually, this library of functions will become an integral part of later versions of Windows such as Windows NT 5 or Windows 98. So when do you need WinInet? The answer is simple: Every time you need to connect to a WWW, Gopher, or FTP server to perform a file transfer, you should use the WinInet library. The library makes it possible to write applications that do not embed detailed knowledge of the underlying protocols and do not require the manipulation of TCP/IP connections directly through the WinSock library. It also handles the idiosyncrasies of various servers; for instance, your application need not have detailed knowledge of the ASCII directory listing formats of various FTP servers. You can use WinInet as is, or you can use the MFC Internet classes. You would presumably use naked WinInet functions in situations in which the use of MFC is not advantageous. WinInet functions can also be called from pure C code; they can also be used from other programming languages, such as Visual Basic, that provide an interface for calling functions in external DLLs. The MFC Internet classes are quite obviously specific to C++. Although the purpose of the WinInet library is to free the programmer from having to deal with the details of protocol implementations, understanding how they work is still easier if you at least have a superficial understanding of how WWW, Gopher, and FTP work. Therefore, this chapter begins with a brief review of these protocols, followed by a discussion of how WinInet provides an easy-to-use, robust wrapper for them.

Internet Protocols
As mentioned before, the WinInet library provides an API for three of the most popular Internet protocols: FTP, Gopher, and HTTP. Of these three protocols, FTP is the oldest, and provides a means for bidirectional file transfer. Gopher can be viewed as the immediate predecessor of the WWW; it provided an efficient menu-based navigational system for finding and retrieving text and other documents. HTTP, of course, is the protocol behind the World Wide Web. Both HTTP and Gopher are unidirectional; they can be used for retrieving material from a server, but not for uploading.

Using the WinInet API CHAPTER 39

701

The File Transfer Protocol (FTP)


The oldest of the three protocols, FTP has been used on the Internet for many years at file repositories. Usually, you interact with an FTP server using an FTP client application, such as the Windows 95 FTP.EXE program. Here is an example of a typical FTP session:
C:\>ftp vtt1 Connected to vtt1. 220 vtt1 FTP server (Version wu-2.4(1) Sat Feb 18 13:40:36 CST 1995) ready. User (vtt1:(none)): ftp 331 Guest login ok, send your complete e-mail address as password. Password: 230-Welcome, archive user! This is an experimental FTP server. If have any 230-unusual problems, please report them via e-mail to root@vtt1 230-If you do have problems, please try using a dash (-) as the first character 230-of your password -- this will turn off the continuation messages that may 230-be confusing your ftp client. 230230 Guest login ok, access restrictions apply. ftp> cd pub 250 CWD command successful. ftp> ls 200 PORT command successful. 150 Opening ASCII mode data connection for /bin/ls. total 2 drwxr-xr-x 2 root wheel 1024 Apr 9 1996 . drwxr-xr-x 8 root wheel 1024 Apr 9 1996 .. -rwxrwxr-x 1 root wheel 0 Jul 10 1993 dummy_test_file 226 Transfer complete. 198 bytes received in 0.00 seconds (198000.00 Kbytes/sec) ftp> get dummy_test_file 200 PORT command successful. 150 Opening ASCII mode data connection for dummy_test_file (0 bytes). 226 Transfer complete. ftp> bye 221 Goodbye. C:\>

39
USING THE WININET API

Needless to say, what you see here is not the protocol itself, just the user interface of the FTP client. Most commands that can be issued are straightforward and easy to understand; others, such as the ls command, may appear a little cryptic for anyone not familiar with UNIX (ls is the UNIX equivalent of the MS-DOS DIR command, used to list the contents of directories). But how does this all work? How do FTP clients and servers actually communicate? Instead of getting into a meaningless and tedious discussion of the protocols intricacies, we can do a handson experiment by connecting to the FTP server using a telnet client. Most telnet programs, including the Windows 95/98 and Windows NT versions of TELNET.EXE, accept an optional parameter that specifies the TCP port number to connect to. Well-known port numbers also have symbolic names, stored in a file named services (located in the main Windows directory or, under UNIX, in /etc). The port number for FTP, port 21, has the symbolic identifier (no surprise here) ftp, thus a telnet command line similar to the following is valid:
C:\>telnet vtt1 ftp

702

Networks and Communications PART VI

So lets see what happens if we attempt to manually simulate an FTP session:


Connected to vtt1. Escape character is ^]. 220 vtt1 FTP server (Version wu-2.4(1) Sat Feb 18 13:40:36 CST 1995) ready. USER anonymous 331 Guest login ok, send your complete e-mail address as password. PASS user@host.net 230-Welcome, archive user! This is an experimental FTP server. If have any 230-unusual problems, please report them via e-mail to root@vtt1 230-If you do have problems, please try using a dash (-) as the first character 230-of your password -- this will turn off the continuation messages that may 230-be confusing your ftp client. 230230 Guest login ok, access restrictions apply. CWD pub 250 CWD command successful. LIST 425 Cant build data connection: Connection refused.

Oops. We can go no further. To actually transfer a file (directory listings are treated as text files by FTP), we would have to build a second connection because FTP uses two connections: one for control, the other for data transfer. Still, this little telnet session should prove to be educational. As you can see, most commands and responses are human-readable; data is transmitted separately. When directory listings are transmitted, FTP clients have two choices: either display the listing as is, or parse and interpret the contents. In the second case, the client must know how different servers present directory contents. The FTP protocol can be used for many tasks in addition to retrieving files. First, it is bidirectional; via FTP, files can also be uploaded to the server. The protocol also offers several file system manipulation commands that can be used to delete files and create, remove, and traverse directories.

The Gopher Protocol


Gopher, an Internet protocol still popular today, presents the Internet in the form of a series of menus and documents. A typical Gopher session begins by connecting to a root Gopher server somewhere, which presents a series of menu items to the user; selecting these items connects the user to other Gopher servers, either to retrieve specific documents or to navigate additional menus. In the past, Gopher client software was typically text-based. Nowadays, most Web clients such as Netscape or Internet Explorer can also connect to Gopher servers, eliminating the need for separate programs. Web browsers can also display multimedia content, whereas many Gopher clients of the past could only store them on disk for later retrieval. Gopher was originally developed at the University of Minnesota, which still operates the Gopher server most commonly accessed today as the default Gopher server of client programs. On most computer systems that sport a Gopher client, starting it will produce the following menu:

Using the WinInet API CHAPTER 39


Internet Gopher Information Client v2.1.-1 Home Gopher server: gopher2.tc.umn.edu --> 1. 2. 3. 4. 5. 6. 7. 8. 9. 10. 11. 12. Information About Gopher/ Computer Information/ Discussion Groups/ Fun & Games/ Internet file server (ftp) sites/ Libraries/ News/ Other Gopher and Information Servers/ Phone Books/ Search Gopher Titles at the University of Minnesota <?> Search lots of places at the University of Minnesota <?> University of Minnesota Campus Information/ Page: 1/1

703

Press ? for Help, q to Quit

The user can move the selection arrow up and down using cursor keys; an item can be selected by pressing Enter. So how does the protocol actually deliver this information? Lets try to connect to the University of Minnesota server manually (that is, using Telnet) and retrieve the same menu. To do so, all we need to do is send a Carriage Return character (Control+M) after the connection is established:
$ telnet gopher2.tc.umn.edu gopher Trying 134.84.132.29... Connected to gopher.tc.umn.edu. Escape character is ^]. 1Information About Gopher<tab>1/Information About Gopher<tab>gopher.tc.umn.edu<tab>70<tab>+ 1Computer Information<tab>1/computer<tab>spinaltap.micro.umn.edu<tab>70 1Discussion Groups<tab>1/Mailing Lists<tab>gopher.tc.umn.edu<tab>70<tab>+ 1Fun & Games<tab>1/fun<tab>spinaltap.micro.umn.edu<tab>70 1Internet file server (ftp) sites<tab>1/FTP Searches<tab>gopher.tc.umn.edu<tab>70<tab>+ 1Libraries<tab>1/Libraries<tab>gopher.tc.umn.edu<tab>70<tab>+ 1News<tab>1/News<tab>gopher.tc.umn.edu<tab>70<tab>+ 1Other Gopher and Information Servers<tab>1/Other Gopher and Information Servers<tab>gopher.tc.umn.edu<tab>70<tab>+ 1Phone Books<tab>1/Phone Books<tab>gopher.tc.umn.edu<tab>70<tab>+ 7Search Gopher Titles at the University of Minnesota<tab><tab>mudhoney.micro.umn.edu<tab>4325 7Search lots of places at the University of Minnesota<tab><tab>mindex:/lotsoplaces spinaltap.micro.umn.edu<tab>70 1University of Minnesota Campus Information<tab>1/ uofm<tab>gopher.tc.umn.edu<tab>70<tab>+ . Connection closed by foreign host. $

39
USING THE WININET API

As you can see, the contents of the menu that Gopher presents on the screen arrives in a highly structured format. Each line begins with a single character that identifies the type of information that the menu item represents. Type 1, that we have the most of in the preceding menu,

704

Networks and Communications PART VI

is a Gopher directory (that is, another menu), while type 7 represents a search service. Needless to say, there are many other types. The type character is followed by several strings, separated by the tab character. The first is the selector string, the human-readable title of the item. The second, third, and fourth strings identify the items internal identifier, the host, and the port number where it can be retrieved from. Gophers output is simpler when an actual document is being retrieved. The client requests the document by sending its identifier, followed by a Carriage Return; the server responds by dumping the documents contents on the client. Needless to say, a simplistic overview like this does not do the protocol justice, but I hope it was sufficient to establish a basic understanding of how Gopher delivers information.

The Hypertext Transfer Protocol (HTTP)


By far the most popular protocol on the Internet these days is HTTP. This protocol is used throughout the World Wide Web to deliver multimedia information in the form of Hypertext Markup Language (HTML) documents: Web pages, that is. Because of the popularity of the World Wide Web, I dont think there is any need to introduce Web pages from a users perspective. I seriously doubt that there is a single Windows developer, especially not one interested in C++ programming, left on this planet who has not seen yet seen a Web browser in operation. So lets jump right ahead and see how the protocol works. Suppose you are trying to retrieve Web content from the following URL: http://localhost/blank.html. A simple telnet session shows how this can be accomplished manually:
$telnet localhost http Trying 127.0.0.1... Connected to localhost. Escape character is ^]. GET /index.html <HTML> <HEAD> <TITLE>Microsoft Internet Explorer</TITLE> </HEAD> <BODY> </BODY> </HTML> Connection closed by foreign host. $

Not much mystery here. The URL is obviously split into halves by the browser; the first part, http://localhost, identifies both the host that supplies the requested material and the protocol used to retrieve it; the second part, /index.html, is sent to the server in the form of a GET command, to which the server responds with the documents contents. Again, the actual protocol is of course much more complex than this simple session would suggest. For instance, an HTTP client can identify its type, pass on authorization information

Using the WinInet API CHAPTER 39

705

for accessing password-protected Web pages, and so on. However, this example should take some of the mystery away just by showing what goes on behind the scenes when a simple request is serviced by a Web server.

The WinInet Library


The previous examples illustrate how simple files are retrieved from servers using the FTP, Gopher, and HTTP protocols. Carrying out the actual conversations with servers from C or C++ programs, however, would be a tedious task. The application would have to open and manage one or several sockets. It would have to embed detailed knowledge of the protocols involved, including server- or version-specific idiosyncrasies. Because these types of connections are used frequently in Internet programming, creating a reusable library is an obvious stepand this is exactly what the WinInet library delivers. The WinInet programming model should be familiar to Windows programmers who have experience with ODBC or TAPI, for example. Like layers of an onion, a WinInet transaction consists of establishing a session, a connection within that session, and a file transfer within that connection, as illustrated in Figure 39.1.

FIGURE 39.1.
Typical WinInet call sequence.

InternetOpen(); InternetConnect(); FtpOpenFile(); GopherOpenFile(); HttpOpenRequest();

Initialize the WinInet library Connect to an Internet server Open an FTP or Gopher file, or an HTTP request

39
USING THE WININET API

// Execute WinInet calls InternetReadFile(); HttpSendRequest(); ... InternetCloseHandle(); InternetCloseHandle(); InternetCloseHandle(); Close the Internet file Terminate server connection Deinitialize WinInet library

All WinInet sessions begin by establishing a WinInet session handle through a call to the InternetOpen function. Most applications will need to make only one call to this function, as the handle is reusable. However, applications can conceivably use several handles, for instance, when different proxy servers are utilized. Equipped with a session handle, applications establish Internet connections through a call to This function is used to establish connections to all three types of servers. InternetConnect also returns a handle, which can be used in subsequent calls to functions such as FtpOpenFile, GopherOpenFile, or HttpOpenRequest calls.
InternetConnect.

706

Networks and Communications PART VI

The following sections review some simple examples that retrieve files from FTP, Gopher, and HTTP servers.

Retrieving Files from an FTP Server


Listing 39.1 contains a sample program that retrieves a specific file from an FTP server across the Internet.

Listing 39.1. Simple FTP application.


#include <windows.h> #include <wininet.h> #include <stdio.h> void main(void) { HINTERNET hIS, hIC, hIF; DWORD dwBytes; char c; hIS = InternetOpen(FTPGET, INTERNET_OPEN_TYPE_DIRECT, NULL, NULL, 0); hIC = InternetConnect(hIS, ftp.microsoft.com, INTERNET_DEFAULT_FTP_PORT, NULL, NULL, INTERNET_SERVICE_FTP, 0, 0); hIF = FtpOpenFile(hIC, disclaimer.txt, GENERIC_READ, FTP_TRANSFER_TYPE_ASCII, 0); while (InternetReadFile(hIF, &c, 1, &dwBytes)) { if (dwBytes != 1) break; putchar(c); } InternetCloseHandle(hIF); InternetCloseHandle(hIC); InternetCloseHandle(hIS); }

wininet.lib.

To compile this example, enter the following command at the command line: cl ftpget.c The resulting executable, ftpget.exe, should successfully access the Microsoft FTP server and display the contents of the file disclaimer.txt, found at that server. The program contains no surprises. As discussed earlier, three calls are used to establish the connection and open the desired file: a call to InternetOpen initializes the WinInet library, a call to InternetConnect establishes an anonymous FTP connection to the server, and a call to FtpOpenFile actually accesses the file. After the file is successfully opened, repeated calls to InternetReadFile are used to retrieve its contents. In this example, retrieval takes place one character at a time; however, in practical applications, one would typically use a much larger buffer for more efficient transfer.

Using the WinInet API CHAPTER 39

707

With a few changes, this application can be modified to access any server and retrieve the file of your choice. Additional modifications could add new features to the application, such as the ability to toggle ASCII or binary mode file transfers, or implement file uploading. The WinInet library also contains functions that facilitate the browsing of FTP directories. The function FtpFindFirstFile can be used to retrieve directory information from the FTP server and parse it. The major advantage of this approach is that your application does not need to know about the format in which directory information is presented by various FTP servers.

Retrieving Files from a Gopher Server


Retrieving files from Gopher servers is not any more difficult than retrieving them using FTP, as illustrated in Listing 39.2.

Listing 39.2. Simple Gopher application.


#include <windows.h> #include <wininet.h> #include <stdio.h> void main(void) { HINTERNET hIS, hIC, hIF; DWORD dwBytes; char c, *p; hIS = InternetOpen(GOPHRGET, INTERNET_OPEN_TYPE_DIRECT, NULL, NULL, 0); hIC = InternetConnect(hIS, localhost, 0, NULL, NULL, INTERNET_SERVICE_GOPHER, 0, 0); hIF = GopherOpenFile(hIC, 0\t\tgopher2.tc.umn.edu\t70, NULL, 0, 0); while (InternetReadFile(hIF, &c, 1, &dwBytes)) { if (dwBytes != 1) break; putchar(c); } InternetCloseHandle(hIF); InternetCloseHandle(hIC); InternetCloseHandle(hIS); }

39
USING THE WININET API

This application can also be compiled with a simple command, cl entered at the command line.

gophrget.c wininet.lib,

The first two WinInet calls in this program are the same as the calls in the FTP example. However, note that the parameters passed to InternetConnect do not actually specify the Internet host that will be accessed later. This is because the Internet host and TCP port number will be

708

Networks and Communications PART VI

specified in the call to GopherOpenFile, and the host name and port number parameters passed to InternetConnect are apparently ignored. However, I found no explicit statement to this effect in the WinInet documentation; in other words, this behavior may be subject to change in later versions of the WinInet library. The call to GopherOpenFile differs from its FTP counterpart in that its second parameter does not specify a filename but a Gopher locator string. A Gopher locator string is similar in format to the strings contained in Gopher menus. The string contains tab-delimited fields that specify the documents type and title, identifier, and the host and port number that the document can be retrieved from. In our example, the string 0\t\tgopher2.tc.umn.edu\t70 is used. This translates as follows:
0

<empty string>
gopher2.tc.umn.edu 70

Type is 0; title is blank Identifier of the root menu Host name Default Gopher TCP port number

Although in this simple example a manually created locator string has been used, typically this would not be the case in real-life applications. Far more likely, you would be using a locator string returned by the function GopherFindFirstFile. This function can retrieve and parse a Gopher menu and present its contents in the form of locator strings embedded in structures of type GOPHER_FIND_DATA.

Retrieving Files from an HTTP Server


The sample program in Listing 39.3 performs the now-familiar function of retrieving a single file, this time from an HTTP server.

Listing 39.3. Simple Gopher application.


#include <windows.h> #include <wininet.h> #include <stdio.h> void main(void) { HINTERNET hIS, hIC, hIR; DWORD dwBytes; char c; hIS = InternetOpen(HTTPGET, INTERNET_OPEN_TYPE_DIRECT, NULL, NULL, 0); hIC = InternetConnect(hIS, www.microsoft.com, INTERNET_DEFAULT_HTTP_PORT, NULL, NULL, INTERNET_SERVICE_HTTP, 0, 0); hIR = HttpOpenRequest(hIC, NULL, /default.asp, NULL, NULL, NULL, 0, 0);

Using the WinInet API CHAPTER 39


if (HttpSendRequest(hIR, NULL, 0, NULL, 0)) { while (InternetReadFile(hIR, &c, 1, &dwBytes)) { if (dwBytes != 1) break; putchar(c); } } InternetCloseHandle(hIR); InternetCloseHandle(hIC); InternetCloseHandle(hIS); }

709

To compile this program from the command line, type cl

httpget.c wininet.lib.

Like its FTP and Gopher counterparts, this application also begins with calls to InternetOpen and InternetConnect; this time, an HTTP connection is specified to Microsofts main Web server. Opening an HTTP request with HttpOpenRequest does not automatically communicate the request to the server. For this, a separate call to HttpSendRequest is needed. After the request is successfully sent, the servers response can be read using InternetReadFile just as with FTP and Gopher.

Other WinInet Features and Functions


The examples presented in this chapter all use WinInet functions in synchronous (blocking) mode. In other words, the functions do not return until the specified request is completed or an error occurs. The WinInet library also supports asynchronous mode, in which functions return immediately and the request is processed in the background. Asynchronous mode can be specified using the INTERNET_FLAG_ASYNC flag in the fifth parameter to InternetOpen. After this flag is specified, all subsequent calls utilizing the session handle returned by the InternetOpen call operate in asynchronous mode. Asynchronous mode requires that a status callback function be registered using InternetSetStatusCallback. When a WinInet function is called in asynchronous mode, it may return the error code ERROR_IO_PENDING. When the operation is completed, the WinInet library calls the status callback function with the value INTERNET_STATUS_REQUEST_COMPLETED. Other features of the WinInet library include the capability to manage the cache. The cache is used to save local copies of data transferred from the Internet, thereby speeding up subsequent access to the same information by eliminating repeated downloads. Cache functions in WinInet allow an application to explicitly identify locally stored data with specific URLs and later reference this stored data using URLs.

39
USING THE WININET API

710

Networks and Communications PART VI

Summary
The WinInet library is one of Microsofts latest additions to the Win32 API to improve Internet support. The library provides support for communicating with three types of Internet servers: FTP, HTTP, and Gopher. These three protocols are the most widely used on the Internet for providing access to file-based content. The typical calling sequence consists of a call to InternetOpen, followed by InternetConnect, and then functions specific to the protocol in use. Many WinInet functions return Internet handles of type HINTERNET, which can subsequently be closed via calls to InternetCloseHandle. In addition to file transfers, the WinInet library supports many other types of operations specific to the three types of servers. In particular, it supports file and directory manipulation and uploading of files to FTP servers; parsing of FTP directory listings; interpretation of Gopher menus; and various HTTP operations. The library also supports cache operations. WinInet functions can operate in synchronous (blocking) and asynchronous mode. Asynchronous mode requires the installation of a callback function that is used by the library to communicate status information pertaining to pending requests.

Telephony Applications with TAPI CHAPTER 40

711

40

Telephony Applications with TAPI


IN THIS CHAPTER
s TAPI Overview 712 s TAPI Software Architecture s TAPI Services 719 s A Data Communication Example 723 716

40
TELEPHONY APPLICATIONS WITH TAPI

712

Networks and Communications PART VI

The Microsoft Telephony API, or TAPI, provides telephony-related services for Win32 applications. TAPI is used extensively by those Windows 95/98 and Windows NT applications that use a modem; for example, the FAX driver and the CompuServe driver in Microsoft Exchange, the Microsoft Network software, or the Windows 95 Phone Dialer and Hyperterminal applications.

TAPI Overview
First, an important note. When most of us hear the term telephony in the context of computer communications, we think of data or FAX modems on voice grade telephone linesand very little else. TAPI goes far beyond these simple concepts and provides a consistent programming interface for a variety of devices operating on voice grade lines, ISDN lines, and private branch exchanges. The devices include modems, FAX modems, voice capable modems, computercontrolled telephone sets, and more. TAPI provides services for placing outgoing calls, accepting incoming calls, and managing calls and devices. What TAPI does not do is handle the media stream; that is, the data that is exchanged during a call. For example, when TAPI is used to place a voice call, it is not TAPI but you, the human operator, who talks. Similarly, when TAPI is used to place a data call, it is the communication application that takes over the device and performs I/O operations using standard Win32 file functions.

Assisted TAPI: The Simplest TAPI Application


Before we delve deeper into the TAPI architecture, take a look at the simple program in Listing 40.1. This is about as simple as a TAPI application can get. This program takes a single command-line argument, a telephone number, and dials that number for a voice call.

Listing 40.1. The simplest TAPI application.


#include <windows.h> #include <stdio.h> #include <tapi.h> void main(int argc, char *argv[]) { if (argc != 2) printf(Usage: %s telephone-number\n, argv[0]); else { printf(Dialing %s..., argv[1]); tapiRequestMakeCall(argv[1], NULL, NULL, NULL); } }

Telephony Applications with TAPI CHAPTER 40

713

The actual dialing is performed on behalf of the application by the default call control application. An example for a call control application is the Phone Dialer that is provided as part of Windows 95. The tapiRequestMakeCall function that is used in this program is ideally suited for use in scripts. For example, a call to this function can be included as an external DLL call in Visual Basic for Applications. To compile this application from the command line, type cl
dial.c tapi32.lib.

The tapiRequestMakeCall function is part of TAPIs Assisted Telephony features. In the current version of TAPI, there is only one other Assisted Telephony function, the tapiGetLocationInfo function, which can be used to obtain the country code and city code for the users current location.

TAPI Concepts
TAPI provides a series of personal telephony services. Telephony, in this context, refers to technology in general that connects computers with the telephone network. TAPI services provide for all aspects of usage of the telephone network. This includes connecting to the network, placing and accepting calls, call management features (such as transferring calls, setting up conference calls), use of calling number identification (Caller ID) for identifying incoming calls, and more. TAPI services are divided into basic, supplementary, and extended services. Basic services are generally supported by all devices; supplementary services may only be available on special devices. Extended services are provider-specific. For example, TAPI can place a call on all telephone lines; however, call management functions, such as transferring a call, may only be available on devices that specifically support such a feature, and thus is considered a supplementary service. TAPI is not restricted to what is whimsically referred to by the acronym POTSPlain Old Telephone Service. POTS is analog service on the local loop (the wire connecting the telephone set with the nearest switching office). POTS supports voice calls with a 3.1KHz bandwidth, or data calls at speeds up to 33.6Kbps using enhanced V.34 modems. (Note that the new 56Kbps modems do not operate on POTS; they require that no more than one segment of the completed connection use analog facilities. That is, you may place your 56Kbps call to your Internet service provider on an analog line, but the service provider must have a digital connection to the switching office.) In contrast, Integrated Services Digital Network (ISDN) supports up to 128Kbps with its Basic Rate Interface (BRI-ISDN); the speed on PRI-ISDN (Primary Rate Interface) is much higher. TAPI supports ISDN as well as other connection types, such as switched 56, or T1/E1. TAPI can also utilize CENTREX features and the features of Private Branch Exchanges (PBXs).

40
TELEPHONY APPLICATIONS WITH TAPI

714

Networks and Communications PART VI

TAPI Devices
TAPI makes a distinction between line devices and phone devices. A line device is the abstract representation of a physical device that connects your computer to the telephone network. Examples of line devices include modems, FAX modems, or ISDN cards. A phone device is the abstract representation of a device with the capabilities of a telephone set. A phone device may have a speaker, a microphone, lamps, a display, buttons, and so on. A phone device is not necessarily a physical device; a software emulation that uses the computers speaker, microphone, sound card hardware, and a voice-capable modem and displays a telephone-like interface on the screen can also act as a phone device. Figure 40.1 shows a basic configuration consisting of a line device (a data modem) and a phone device (a programmable telephone under TAPI control). Note that the presence of a telephone set does not imply the existence of a TAPI phone device. For example, if you have a plain telephone set that is used in conjunction with a dialer that responds to the Hayes AT command set (or a modem used as a dialer), this configuration is represented by a line device. A phone device is used when TAPI has control over some of the features of the telephone set such as its display, switchhook, ringer, or buttons.

FIGURE 40.1.
TAPI devices.
Modem Telephone network

Personal computer Programmable telephone

Line devices all provide a basic set of functions (Basic Telephony). In contrast, all phone device functions are part of Supplementary Telephony; this is because there is no minimum set of functions that a phone device is expected to provide.

TAPI Addresses
TAPI distinguishes between the concepts of a line and that of an address. The line is the physical entity; the address is, for example, a telephone number assigned to the line. Although most POTS lines are associated with a single telephone number, this is not always the case. For example, an ordinary telephone line may be configured with more than one telephone number using the telephone companys distinctive ringing service. On digital lines, the use of more than one address on a single line is more common.

Telephony Applications with TAPI CHAPTER 40

715

A unique feature of telephone numbers is that their actual format is relative to the originating location. Take, for example, a number here in Ottawa, such as 613-555-1234. When I dial this number locally, all I need to dial is the seven digits of the local number, 555-1234. If I call this number from New York City, I need to dial 1-613-555-1234. Calling the same number from Budapest, Hungary, requires dialing 00w1-613-555-1234, where the letter w represents waiting for a second dial tone. Calling the same number from a telephone set attached to the PBX of a local company may require dialing the digits 8-555-1234. These differences become especially relevant on portable computers. TAPI does an excellent job translating telephone numbers. At the heart of its capability is the canonical address format. The syntax of a canonical address is as follows:
+CountryCode Space [(AreaCode) Space] SubscriberNumber [| SubAddress] [^ Name] CRLF

For example, the canonical format of the Ottawa number 555-1234 is as follows:
+1 (613) 555-1234

The canonical address differs from a dialable address. Dialable addresses are those that do not begin with the character +; these numbers are presumed to be dialable on the given line without modification. A canonical address can be translated into a dialable address by the lineTranslateAddress TAPI function. The syntax for a dialable number is as follows:
DialableNumber [| SubAddress] [^ Name] CRLF

Dialable addresses can contain, in addition to digits, the DTMF symbols A-D, # (DTMF gate), and * (DTMF star), and any of a variety of dial modifier characters. Dial modifier characters are based on the Hayes AT command set and include the characters shown in Table 40.1.

Table 40.1. Dial modifier characters used in dialable addresses. Dial modifier character Description
(exclamation mark) P or p T or t , (comma) w or W
! @ $ ;

Flash switchhook Pulse dial for subsequent digits Tone dial for subsequent digits Pause Wait for dial tone Wait for quiet answer Wait for billing signal Indicates incomplete dialable number

40
TELEPHONY APPLICATIONS WITH TAPI

716

Networks and Communications PART VI

TAPI Software Architecture


At the heart of TAPI is the TAPI DLL, the dynamic link library that offers TAPI services to applications. This DLL serves as a layer between telephony applications and TAPI service providers. One such service provider is the UNIMODEM driver; this Universal Modem driver is supplied with Windows 95 and provides TAPI services for modems compatible with the Hayes AT command set. This basic TAPI architecture is shown in Figure 40.2. In addition to the TAPI DLL and the telephony service providers (drivers), another important, albeit invisible, component of TAPI is the executable program tapiexe.exe or tapisrv.exe on Windows NT. This program plays an important role when TAPI sends notifications to the calling application via callback functions.

FIGURE 40.2.
The TAPI software architecture.

APPLICATION

TAPI DLL

TAPIEXE.EXE

TAPI service provider

TAPI service provider

TAPI service provider

Telephony device

Telephony device

Telephony device

Synchronous and Asynchronous Operations


Many TAPI operations are synchronous; that is, when the TAPI function returns, the operation is either completed or failed, in which case an error code is returned. However, some TAPI operations are asynchronous; the TAPI function returns indicating whether the TAPI operation has been successfully initiated, but the operation is completed in another thread, and the application is notified via a callback function. The callback function is registered with TAPI when the TAPI library is initialized. The actual callback mechanism deserves a closer examination, especially because it has some consequences as to how TAPI functions operate. When a service provider wishes to place a notification, it calls the TAPI DLL. In effect, it requests that the DLL notify all concerned applications that a specific event has taken place. This first call to the TAPI DLL takes place in the execution context of the service provider.

Telephony Applications with TAPI CHAPTER 40

717

The TAPI DLL in turn sends a message to tapiexe.exe (tapisrv.exe). This executable program calls the TAPI DLL itself, this time in its own execution context. This call instructs the TAPI DLL to post a Windows message to the applications that need to be modified. When the application receives and processes the message in its message loop, the message is dispatched to the TAPI DLL again, this time in the applications execution context. The TAPI DLL may in turn call the applications registered TAPI callback function to notify the application of a TAPI event. The TAPI notification mechanism is shown in Figure 40.3.

FIGURE 40.3.
Processing of TAPI events.
Message TAPIEXE.EXE Message

Telephony application

Callback

TAPI DLL

TAPI DLL

TAPI DLL

Event notification Service provider

This mechanism has important implications for the architecture of TAPI applications. For one thing, the scenario described here makes it clear that TAPI applications must have a message loop in order to process notifications correctly. Although the use of a callback function may imply that a message loop is unnecessary, this is not the case; the callback function is only called after the application receives a Windows message that the TAPI DLL processes. Without a message loop in the application that can properly receive and dispatch messages, the TAPI call will never successfully complete. Another consequence concerns the use of multiple threads. It is important to realize that in order for TAPI to operate as expected, threads that call asynchronous TAPI functions must have a message loop. The callback function is called in the context of the thread making the asynchronous call; this cannot happen unless the thread processes Windows messages. Although the need for a message loop does not completely rule out the use of TAPI with console applications, it places certain restrictions on them. The console application must have a message loop that processes and dispatches Windows messages. The example presented later in this chapter (see Listing 40.2) demonstrates the use of this technique.

40
TELEPHONY APPLICATIONS WITH TAPI

718

Networks and Communications PART VI

Variable-Length Structures
Throughout TAPI, variable-length structures are frequently used. These structures represent data that is variable in length, such as optional fields or strings. All TAPI variable-length structures make use of structure members dwTotalSize, dwNeededSize, and dwUsedSize. When a TAPI function is called that is expected to return data in such a structure, your first task is to allocate the structure and fill its dwTotalSize member prior to making the call. The TAPI documentation refers to structures of this kind as flattened. Instead of referred to through pointers, supplementary data fields and variable-length fields are simply appended to the end of the structure. Variable-length fields are referred to in the structure by an offset and a length parameter; the offset specifies the starting position of the field, in bytes, relative to the start of the structure; the length represents the length of the field in bytes. Suppose a TAPI function called tapiStrangeFunc returns variable-length data in a VARSTRUCT structure. This structure is declared as follows:
typedef struct { DWORD dwTotalSize; DWORD dwNeededSize; DWORD dwUsedSize; // other fixed-length elements here DWORD dwVarItem1Size; DWORD dwVarItem1Offset; // more fixed-length elements here DWORD dwVarItem2Size; DWORD dwVarItem2Offset; } VARSTRUCT, FAR *LPVARSTRUCT;

Before tapiStrangeFunc is called, you must allocate a VARSTRUCT structure. Although you can allocate it as an automatic variable, doing so is not recommended. Instead, use the following mechanism:
LPVARSTRUCT pVarStruct; pVarStruct = (LPVARSTRUCT)malloc(sizeof(pVarStruct)); pVarStruct->dwTotalSize = sizeof(VARSTRUCT); tapiStrangeFunc(pVarStruct);

The description of tapiStrangeFunc may tell you that this function appends an extra DWORD member to this structure. Clearly, this extra member requires additional memoryand so do the two variable-length items identified by the members dwVarItem1Size/dwVarItem1Offset, and dwVarItem2Size/dwVarItem2Offset. When tapiStrangeFunc returns, this fact is indicated by the value of the dwNeededSize structure member. This member will indicate that additional memory is needed to return all values. A possible response to this would be a reallocation of the structure, and another call to tapiStrangeFunc:

Telephony Applications with TAPI CHAPTER 40


pVarStruct = (LPVARSTRUCT)malloc((LPVOID)pVarStruct, pVarStruct->dwNeededSize); pVarStruct->dwTotalSize = pVarStruct->dwNeededSize; tapiStrangeFunc(pVarStruct);

719

When this call to tapiStrangeFunc returns, pVarStruct points to a structure in memory, as shown in Figure 40.4.

FIGURE 40.4.
An example for a TAPI variable-length structure.

dwTotalSize dwNeededSize dwUsedSize Other fixed-length structure members dwVarItem1Size dwVarItem1Offset


dwVarItem1Offset

Original structure size

Other fixed-length structure members dwVarItem2Size dwVarItem2Offset Extended fixed-length items


dwVarItem1Size dwNeededSize

dwVarItem2Offset

Variable-length item 1

dwVarItem2Size

Variable-length item 2

The dwUsedSize field is somewhat of a lesser significance; it comes into play when TAPI could not fill in all the structure members (dwTotalSize was less than dwNeededSize). In this case, rather than truncating a variable-length field, TAPI simply leaves that field empty.

40
TELEPHONY APPLICATIONS WITH TAPI

TAPI Services
In addition to the Assisted TAPI services that we have seen already, TAPI provides services that fall into three categories: Basic Telephony, Supplementary Telephony, and Extended Telephony. Basic Telephony includes all functions that a POTS line can be expected to provide. This minimal set of functions must be supported by all service providers.

720

Networks and Communications PART VI

Supplementary Telephony includes all standard TAPI services that are not in the Basic Telephony set of functions. These include supplementary services found on most PBXs, such as hold, call transfer, conference calls, and so on. An application can query the set of supplementary services supported by a particular line device or phone device by calling lineGetDevCaps, lineGetAddressCaps, or phoneGetDevCaps.

NOTE
Because there is no minimum set of services that phone devices are expected to support, all phone device services are in the Supplementary Telephony category.

Extended TAPI services are provider-specific. These include all device-specific TAPI extensions. TAPI provides the necessary mechanisms for extending services through variable-length structures, and functions through which service providers can inform applications about the extended services they support.

The TAPI Programming Model


The basic TAPI programming model for line devices is illustrated in Figure 40.5.

FIGURE 40.5.
The TAPI programming model: Calls in a typical TAPI application.

lineInitialize();

// Initialize TAPI for use of line device

lineNegotiateAPIVersion();// Negotiate API version number lineOpen(); // Opens the line device lineMakeCall(); // // // // Talk Transfer data Transfer Fax etc. // Terminate the call // Free the line device // Shut down TAPI // Place an asynchronous call request

lineDrop(); lineClose(); lineShutdown();

All TAPI applications that utilize line devices begin by a call to lineInitialize. This call initializes TAPI for use with line devices. Note that this call should not be made unless the application actually intends to utilize TAPI services; making this call unnecessarily may use up valuable TAPI resources and cause other applications to fail. One parameter to lineInitialize is the address of a callback function. It is through this function that the application is informed of the completion of asynchronous function requests and other TAPI events.

Telephony Applications with TAPI CHAPTER 40

721

Before an application can open a specific line device, it must negotiate a TAPI version number by calling lineNegotiateAPIVersion. Through this call, the application and the service provider handling the specific device can agree on a version number they can both support. (The retail version of Windows 95 uses TAPI 1.4.) The line device is opened by calling lineOpen. Afterwards, applications can call a variety of functions that use the open device. One example is the lineMakeCall function that is used to place a call on the line device. This function is also an example for an asynchronous function; it returns immediately after the call request has been successfully placed. The application is notified of the completion of the call request through its callback function. After the call has been placed, an application can do a variety of things with the line depending on its intended function. A number of additional functions can be used to obtain information about the line and the call, configure the line, and manipulate addresses. Supplementary functions can be used to transfer the call, place it on hold, set up conference calls, and so on. A specific feature offered by TAPI assists data communication applications in particular. Such applications may use the TAPI lineGetID function to obtain a handle to the communication device. This handle is opened by TAPI for overlapped I/O and can be used by the application for exchanging data with a remote host. Using lineMakeCall is not the only way to establish a call. Applications may also obtain a call handle by accepting incoming calls. Applications that are set up to accept incoming calls are notified of such calls through their callback function. When the application wishes to terminate the call, it can use the lineDrop function. A call may also be terminated by the remote end; in this case, the application is notified through its callback function. When the application is finished using the line device, it should call the lineClose function to close the device. The lineShutdown function can be used to terminate the applications session with TAPI. The programming model used for phone devices is similar. The key steps of initializing TAPI, negotiating a version number, and opening the device are present. There is no equivalent to placing a call on a phone device; phone device functions exist to manipulate the various components of a telephone, such as its switchhook, display, or buttons. Applications that wish to use provider-specific Extended Telephony services must call the lineNegotiateExtVersion or phoneNegotiateExtVersion functions to negotiate the extended version number. Device-specific functions can be executed by calling the escape functions lineDevSpecific, lineDevSpecificFeature (for switch functions), and phoneDevSpecific.

40
TELEPHONY APPLICATIONS WITH TAPI

722

Networks and Communications PART VI

TAPI Media Modes


TAPI provides two concepts that specify the quality of service supported by a line and the type of a call. The bearer mode specifies the quality of service. For example, the voice bearer mode (LINEBEARERMODE_VOICE) indicates a POTS line with a 3.1KHz analog bandwidth and no provisions for data integrity. Other bearer modes describe ISDN or other data lines. The media mode determines the type of the call. For example, on a voice line it is possible to make voice or data calls; these correspond to the media modes LINEMEDIAMODE_INTERACTIVEVOICE or LINEMEDIAMODE_DATAMODEM.

Multiple Applications
The TAPI architecture enables multiple applications to coexist. This is a very important feature. This makes it possible, for example, for a TAPI FAX application to monitor a line for incoming FAX transmissions while at the same time enabling another TAPI application, such as a data communication application, to use the same line for outgoing calls. At the heart of this capability is the concept of call ownership. Initially, ownership of a call is assigned to one application; it is either the application that originated the call or the application that receives the incoming call. An application can pass ownership of a call to another application through the lineHandoff function. The original application also continues owning the call. It can then choose to remain a co-owner of the call (although doing so is not recommended), deallocate the call handle indicating that it is no longer interested in the call, or use lineSetCallPrivilege to become a call monitor. A call monitor is an application that cannot control the calls existence, but it can record facts about the call (logging).

NOTE
If a call is co-owned by multiple applications, TAPI offers no mechanism to prevent these applications from interfering with each other.

When handling incoming calls, applications may perform probing to determine the nature of the call. Probing can be used, for example, to determine whether an incoming call is a data, FAX, or voice call. Probing is usually done by applications, although some service providers can be configured to auto-answer a call and hand it off to the appropriate application. Note that TAPI does not launch applications to handle specific call types. Applications can learn about in-progress calls at startup by calling lineGetNewCalls. Through this function, an application can obtain handles with monitoring privilege for all calls that are currently in progress.

Telephony Applications with TAPI CHAPTER 40

723

Applications can communicate with each other by using the lineSetAppSpecific function. Through this function, they can set the dwAppSpecific field of the LINECALLINFO structure. Other applications that own or monitor the call are informed by receiving a LINE_CALLINFO message.

A Data Communication Example


I decided to put TAPI into practice by modifying a simple console application I wrote earlier. This simple communication application opens a communication port and uses overlapped I/O operations to perform input and output. In its original version, the application simply opened the port without making any attempt at placing a call. It was up to the user to use the appropriate AT commands to place a call. The communication port was hardcoded in the application. In the TAPI version presented in Listing 40.2, the call is placed through the TAPI function lineMakeCall. Before that happens, the user is given the opportunity to choose a TAPI device.

Listing 40.2. A simple TAPI data communication program.


#include <windows.h> #include <tapi.h> #include <stdio.h> volatile BOOL bConnected = FALSE; VOID FAR PASCAL lineCallback(DWORD hDevice, DWORD dwMsg, DWORD dwCallbackInstance, DWORD dwParam1, DWORD dwParam2, DWORD dwParam3) { if (dwMsg == LINE_CALLSTATE && dwParam1 == LINECALLSTATE_CONNECTED) bConnected = TRUE; } LINEDEVCAPS *GetDevCaps(HLINEAPP hLineApp, DWORD dwDeviceID, LPDWORD lpdwAPIVersion) { LINEDEVCAPS *pLineDevCaps; LINEEXTENSIONID extensionID; lineNegotiateAPIVersion(hLineApp, dwDeviceID, 0x10004, 0x10004, lpdwAPIVersion, &extensionID); pLineDevCaps = malloc(sizeof(LINEDEVCAPS)); pLineDevCaps->dwTotalSize = sizeof(LINEDEVCAPS); lineGetDevCaps(hLineApp, dwDeviceID, *lpdwAPIVersion, 0, pLineDevCaps); if (pLineDevCaps->dwNeededSize > pLineDevCaps->dwTotalSize) { pLineDevCaps = realloc(pLineDevCaps, pLineDevCaps->dwNeededSize); pLineDevCaps->dwTotalSize = pLineDevCaps->dwNeededSize; lineGetDevCaps(hLineApp, dwDeviceID, *lpdwAPIVersion, 0,

40
TELEPHONY APPLICATIONS WITH TAPI

continues

724

Networks and Communications PART VI

Listing 40.2. continued


pLineDevCaps); } return pLineDevCaps; } HANDLE SelectTAPIDevice(HLINEAPP hLineApp, DWORD dwNumDevs, LPHLINE lphLine, LPHCALL lphCall) { LINEDEVCAPS *pLineDevCaps; DWORD dwDeviceID; DWORD dwAPIVersion; DWORD i; LINECALLPARAMS lineCallParams; LPVARSTRING lpDeviceID; MSG msg; char szNumber[81]; for (i = 0; i < dwNumDevs; i++) { pLineDevCaps = GetDevCaps(hLineApp, i, &dwAPIVersion); if (pLineDevCaps->dwMediaModes & LINEMEDIAMODE_DATAMODEM) printf(%d: %s\n, i, (char*)pLineDevCaps + pLineDevCaps->dwLineNameOffset); free(pLineDevCaps); } dwDeviceID = -1; while (dwDeviceID < 0 || dwDeviceID >= dwNumDevs) { printf(Select device: ); scanf(%d, &dwDeviceID); if (dwDeviceID < 0 || dwDeviceID >= dwNumDevs) continue; pLineDevCaps = GetDevCaps(hLineApp, dwDeviceID, &dwAPIVersion); if(!(pLineDevCaps->dwMediaModes & LINEMEDIAMODE_DATAMODEM)) { dwDeviceID = -1; free(pLineDevCaps); } } printf(Enter telephone number: ); scanf(%s, szNumber); printf(Dialing %s on %s..., szNumber, (char *)pLineDevCaps + pLineDevCaps->dwLineNameOffset); free(pLineDevCaps); lineOpen(hLineApp, dwDeviceID, lphLine, dwAPIVersion, 0, 0, LINECALLPRIVILEGE_NONE, LINEMEDIAMODE_DATAMODEM, NULL); memset(&lineCallParams, 0, sizeof(LINECALLPARAMS)); lineCallParams.dwTotalSize = sizeof(LINECALLPARAMS); lineCallParams.dwMinRate = 2400; lineCallParams.dwMaxRate = 57600; lineCallParams.dwMediaMode = LINEMEDIAMODE_DATAMODEM; lineMakeCall(*lphLine, lphCall, szNumber, 0, &lineCallParams); while (!bConnected) if (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); putchar(\n); lpDeviceID = malloc(sizeof(VARSTRING)); lpDeviceID->dwTotalSize = sizeof(VARSTRING);

Telephony Applications with TAPI CHAPTER 40


lineGetID(0, 0, *lphCall, LINECALLSELECT_CALL, lpDeviceID, comm/datamodem); if (lpDeviceID->dwNeededSize > lpDeviceID->dwTotalSize) { lpDeviceID = realloc(lpDeviceID, lpDeviceID->dwNeededSize); lpDeviceID->dwTotalSize = lpDeviceID->dwNeededSize; lineGetID(0, 0, *lphCall, LINECALLSELECT_CALL, lpDeviceID, comm/datamodem); } return *((LPHANDLE)((char *)lpDeviceID + sizeof(VARSTRING))); } void main(void) { HLINEAPP hLineApp; HLINE hLine; HCALL hCall; DWORD dwNumDevs; HANDLE hConIn, hConOut, hCommPort; HANDLE hEvents[2]; DWORD dwCount; DWORD dwWait; COMMTIMEOUTS ctmoCommPort; DCB dcbCommPort; OVERLAPPED ov; INPUT_RECORD irBuffer; BOOL fInRead; char c; int i; lineInitialize(&hLineApp, GetModuleHandle(NULL), lineCallback, Test TAPI Application, &dwNumDevs); hCommPort = SelectTAPIDevice(hLineApp, dwNumDevs, &hLine, &hCall); hConIn = CreateFile(CONIN$, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); SetConsoleMode(hConIn, 0); hConOut = CreateFile(CONOUT$, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); ctmoCommPort.ReadIntervalTimeout = MAXDWORD; ctmoCommPort.ReadTotalTimeoutMultiplier = MAXDWORD; ctmoCommPort.ReadTotalTimeoutConstant = MAXDWORD; ctmoCommPort.WriteTotalTimeoutMultiplier = 0; ctmoCommPort.WriteTotalTimeoutConstant = 0; SetCommTimeouts(hCommPort, &ctmoCommPort); dcbCommPort.DCBlength = sizeof(DCB); GetCommState(hCommPort, &dcbCommPort); SetCommState(hCommPort, &dcbCommPort); SetCommMask(hCommPort, EV_RXCHAR); ov.Offset = 0; ov.OffsetHigh = 0; ov.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); hEvents[0] = ov.hEvent; hEvents[1] = hConIn; fInRead = FALSE; while (1) {

725

40
TELEPHONY APPLICATIONS WITH TAPI

continues

726

Networks and Communications PART VI

Listing 40.2. continued


if (!fInRead) while (ReadFile(hCommPort, &c, 1, &dwCount, &ov)) if (dwCount == 1) WriteFile(hConOut, &c, 1, &dwCount, NULL); fInRead = TRUE; dwWait = WaitForMultipleObjects(2, hEvents, FALSE, INFINITE); switch (dwWait) { case WAIT_OBJECT_0: if (GetOverlappedResult(hCommPort, &ov, &dwCount, FALSE)) if (dwCount == 1) WriteFile(hConOut, &c, 1, &dwCount, NULL); fInRead = FALSE; break; case WAIT_OBJECT_0 + 1: ReadConsoleInput(hConIn, &irBuffer, 1, &dwCount); if (dwCount == 1 && irBuffer.EventType == KEY_EVENT && irBuffer.Event.KeyEvent.bKeyDown) for (i = 0; i < irBuffer.Event.KeyEvent.wRepeatCount; i++) { if (irBuffer.Event.KeyEvent.uChar.AsciiChar) { WriteFile(hCommPort, &irBuffer.Event.KeyEvent.uChar.AsciiChar, 1, &dwCount, NULL); if (irBuffer.Event.KeyEvent.uChar.AsciiChar == 24) goto EndLoop; } } } } EndLoop: CloseHandle(ov.hEvent); CloseHandle(hConIn); CloseHandle(hConOut); CloseHandle(hCommPort); lineDrop(hCall, NULL, 0); lineClose(hLine); lineShutdown(hLineApp); }

The fact that this is a console application represented special challenges. In particular, it was necessary to use a message loop at one point to enable the TAPI callback mechanism to work. Although this approach may be somewhat unorthodox, it demonstrates the TAPI programming model and its traps and pitfalls surprisingly well. The first call in the applications main function is to the TAPI function lineInitialize. Next, main calls the function SelectTAPIDevice; this high-level function queries the user for a TAPI

Telephony Applications with TAPI CHAPTER 40

727

device and a telephone number, opens the device, places the call, and returns a Win32 handle that the application can use in subsequent I/O calls. The SelectTAPIDevice function first queries all TAPI line devices for the line name. These line names are presented to the user in the form of a numbered list, and the user is requested to choose one of them. When determining the line name, SelectTAPIDevice utilizes another function, GetDevCaps. Note the duplicate calls to the TAPI function lineGetDevCaps in GetDevCaps; the first call is used to determine the size of the structure lineGetDevCaps would return. The second call is made after a sufficiently large block of memory has been allocated. After a line device has been selected in SelectTAPIDevice, the user is asked to enter a telephone number. This number is then used, after the line has been opened and the appropriate structure initialized, in a call to lineMakeCall. Since lineMakeCall is an asynchronously executing TAPI function, the return of this function does not indicate completion of the request. In particular, the call handle pointed to by lphCall is not yet valid. The application must wait until its callback function is called indicating that the call has been set up; furthermore, it must ensure that the callback mechanism operates as expected by executing a message loop. The callback function, lineCallback, is extremely simplistic; it simply waits for a LINE_CALLSTATE message that indicates that the call has been connected. The rest of the application is notified of call completion when the global variable bConnected is set to TRUE by the callback function. In particular, this change causes the message loop in SelectTAPIDevice to terminate. When SelectTAPIDevice is notified of successful call completion through this mechanism, it uses the lineGetID function to retrieve a handle to the communication port. Note how lineGetID is called twice, first to determine the size of the data structure it is about to return. Note also how an extra structure member of type HANDLE is retrieved. When SelectTAPIDevice returns, it passes the communication device handle to main. In main, this handle is used to configure the communication device and the console for I/O and handle bidirectional data transfer. The application is terminated when the user hits the Control+X key combination. At this time, the application closes all handles, terminates the TAPI session, and exits. To compile this application from the command line, type cl tty.c tapi32.lib user32.lib. The USER library is required because of the references to the Windows functions GetMessage and DispatchMessage. I ran this application on my main desktop computer that has two modems attached to it. An internal FAX modem connects my desktop computer to my data line; an old external pocket modem connects it to my voice line. I mostly use this modem simply as a dialer; however, it comes in handy when I need to test communication applications like this one. Also connected to my data line through its own FAX modem is another computer running Linux (this is my server for Internet mail and TCP/IP connections). This server also accepts incoming data calls, so I can utilize the modem on my voice line to make calls to it.

40
TELEPHONY APPLICATIONS WITH TAPI

728

Networks and Communications PART VI

A sample session using this configuration looked like the following:


C:\TTY>tty 0: SupraFAXModem 144i 1: Practical Peripherals 2400 Select device: 1 Enter telephone number: 555-1234 Dialing 555-1234 on Practical Peripherals 2400... You have reached a private computer system. Calls to this system are logged using calling party identification (caller ID). Unauthorized calls violate my privacy, not to mention the law! If you have not been specifically authorized by me to access this system, now would be a great time to terminate your connection. Viktor Welcome to Linux 1.1.37.

vtt1!login: vttoth Password: Last login: Tue Oct 17 01:57:20 on ttyS0 Linux 1.1.37. (Posix). vtt1:~$ ^X C:\TTY>

NOTE
This simple application provides absolutely no error handling. In particular, it is written with the assumption that the call always succeeds; no provisions are made for unsuccessful call attempts, and the application will likely malfunction in such a case. Furthermore, the application does not handle the loss of carrier; because it does not operate a message loop while it is connected, its TAPI callback function is never notified when the call is terminated by the remote end.

Summary
TAPI, the Microsoft Telephony API, provides personal telephony services for Windows applications. TAPI provides abstractions for line devices that connect a computer to a telephone line, and phone devices, which are telephone sets with a variety of components such as a switchhook, display, buttons, or ring, that can be manipulated programmatically. (Note that most commonly used telephone sets cannot be manipulated this way.) A TAPI line device always represents a physical device; in contrast, a phone device can be a software representation of a telephone that uses the computers display, keyboard, and sound hardware to provide the services of a telephone set.

Telephony Applications with TAPI CHAPTER 40

729

TAPI line devices are not restricted to represent only devices attached to plain old telephone service (POTS) lines. Line devices can represent hardware connected to ISDN lines, T1/E1 data lines, switched 56 data lines, and other lines. TAPI provides services to place outgoing calls and accept incoming calls. It is the responsibility of applications to manage the media stream, the actual flow of data during a call. The media stream is a generic term that represents voice, data, FAX images, or other information that flows through a telephone line. The TAPI DLL represents a layer between applications and device-specific service providers (drivers). Another TAPI component is the tapiexe.exe or tapisrv.exe system application that is used in TAPI messaging. TAPI functions can execute synchronously and asynchronously. Synchronous functions return immediately with a success or failure result. Asynchronous functions, on the other hand, return only to indicate whether a request has been placed successfully; applications are notified of request completion through a callback function. In order for the callback function mechanism to operate, applications must maintain a Windows message loop. The TAPI interface is broken down into Assisted Telephony, Basic Telephony, Supplementary Telephony, and Extended Telephony. Assisted Telephony provides a set of simple functions for placing calls. These functions are ideally suited for use in script languages that can call external DLLs. Basic Telephony consists of those line device functions that all service providers must implement. These include functions to place and accept calls and monitor calls in progress. Supplementary Telephony consists of functions that require special hardware. For example, special hardware is required for call transfer, conference call, and other call management functions to work. Applications cannot expect that a supplementary function is available; they must query the service provider about the availability of a supplementary function. Extended Telephony represents device- and provider-specific functionality. Extended functions are accessed through special escape functions that TAPI provides. Before using extended functions, applications must determine their availability by querying the service provider.

40
TELEPHONY APPLICATIONS WITH TAPI

730

Networks and Communications PART VI

Named Pipes and Remote Procedure Calls CHAPTER 41

731

41

Named Pipes and Remote Procedure Calls


IN THIS CHAPTER
s Communicating with Pipes 732 s Microsoft Remote Procedure Calls 736

41
NAMED PIPES AND REMOTE PROCEDURE CALLS

732

Networks and Communications PART VI

Pipes offer an efficient, easy way for two cooperating Windows applications to communicate across a network. Pipes are easy to set up and use; they are one of the favored mechanisms for implementing client-server communications with server software running on Windows NT. Unfortunately, pipe support in Windows 95 is limited to client-side support. The most often used pipes are named pipes. In addition to their other uses, named pipes also represent one of several mechanisms used by Microsoft Remote Procedure Calls, or Microsoft RPCs. An RPC is a mechanism that enables applications to call procedures (functions) that are part of another application running (possibly) on a different computer on the network. Apart from some initialization and housekeeping functions, using RPC is almost as simple as calling a local function. This chapter examines these two communication mechanisms.

Communicating with Pipes


Pipes offer a one- or two-way conduit of communication between cooperating applications. Programming with pipes is based on the client-server model; a typical server application creates a pipe and waits for client applications to request access to the pipe.

Creating Pipes
The Win32 API distinguishes between named pipes and anonymous pipes. The use of anonymous pipes is somewhat limited. When an application creates an anonymous pipe, it receives two handles; one of these handles must be passed on to another application in order for the two applications to be able to communicate. A possible mechanism for passing a handle is inheritance; therefore, anonymous pipes are often used for communication between a parent and a child process, or between child processes of the same parent process. Because anonymous pipes are identified by handles, they are inherently local (a handle has no validity on another machine on the network). In contrast, named pipes are identified by a UNC name that is valid across the entire network. Anonymous pipes are created using the CreatePipe function. This function returns two handles: one to the read end of the pipe, and another to its write end. Named pipes are created by calling the CreateNamedPipe function. Named pipes have names in the following form:
\\hostname\\pipe\\pipename

Servers cannot create a pipe on another computer. For this reason, the hostname component in the pipe name, when the name is used in CreateNamedPipe, must be set to a single period, indicating the local host:
\\.\pipe\pipename

Named Pipes and Remote Procedure Calls CHAPTER 41

733

A named pipe can be one-way or two-way. When the server creates the pipe, it specifies PIPE_ACCESS_DUPLEX, PIPE_ACCESS_INBOUND, or PIPE_ACCESS_OUTBOUND to indicate the directionality of the pipe. Named pipes support asynchronous (overlapped) I/O operations. However, overlapped I/O is not supported by anonymous pipes. When a named pipe is created, a server specifies the pipes typeeither byte mode or message mode. Byte mode pipes treat data as a stream of bytes; message mode pipes treat data as a stream of messages. The pipes read mode can be byte read mode or message read mode; message mode pipes support both read modes, but byte mode pipes support only the byte read mode. The pipes read mode and its wait mode (whether the pipe operates in a blocking or nonblocking mode) can be specified when the pipe is created and later modified using the SetNamedPipeHandleState function. The current state of the pipe can be obtained by calling GetNamedPipeHandleState. Further information about a named pipe can be obtained by calling GetNamedPipeInfo.

41
NAMED PIPES AND REMOTE PROCEDURE CALLS

Connecting to Named Pipes


When an anonymous pipe is created, the process creating it receives handles for both ends of the pipe. In contrast, when a server creates a named pipe, only one end of the pipe is opened; the server must wait for a client to make an attempt to connect to the pipe before the pipe can be used. An overview of setting up, using, and shutting down named pipes is presented in Figure 41.1.

FIGURE 41.1.
Communicating with named pipes.
Creates a pipe

Server
CreateNamedPipe() WaitNamedPipe()

Client
Waits for the pipe to become available

Waits for incoming connections

ConnectNamedPipe()

CreateFile()

Connects to pipe

Receives data

ReadFile()

WriteFile()

Sends data

Sends data

WriteFile()

ReadFile()

Receives data

Disconnects the pipe Closes the handle

DisconnectNamedPipe()

CloseHandle()

CloseHandle()

Closes the handle

734

Networks and Communications PART VI

To wait for a connection, servers issue a call to the ConnectNamedPipe function. This function waits until a client connects to the pipe, identifying the pipe by its name (note that ConnectNamedPipe can also be used asynchronously, in which case the server is free to attend to other tasks while it is waiting for an incoming connection on the pipe). Clients can determine whether a named pipe is available to be connected to using the function. Once a pipe is available, applications can connect to it using the CreateFile function and giving the name of the pipe as the filename.
WaitNamedPipe

An established connection can be broken by the server by calling the DisconnectNamedPipe function. When a server calls this function, the client-side handle of the pipe becomes invalid and all data that was not yet read by the client is discarded. To ensure that the client read all data before DisconnectNamedPipe is called, servers can call the FlushFileBuffers function. After DisconnectNamedPipe returns, servers can either close the pipe handle or reuse it in another call to ConnectNamedPipe. Clients can break the connection by simply calling CloseHandle on the pipe handle they obtained through calling CreateFile. Servers can utilize several strategies for handling multiple connections, such as spawning separate threads and using overlapped I/O. A server can create multiple instances of the same pipe by repeatedly calling CreateNamedPipe with the same pipe name.

Transferring Data Through Pipes


Pipes can be written to or read from using WriteFile and ReadFile. Named pipes can also be used for overlapped I/O; in this case, the WriteFileEx and ReadFileEx functions may need to be used. For message mode pipes, an alternative mechanism exists for quick and efficient message transfer. Clients can call the TransactNamedPipe function to write a message to and read a message from the specified pipe in a single network operation. The CallNamedPipe function combines into a single operation the calls to WaitNamedPipe, CreateFile, TransactNamedPipe, and CloseHandle. In other words, CallNamedPipe waits for the specified pipe to become available, opens the pipe, exchanges messages, and closes the pipe in a single function call.

A Working Example
This simple example implements a named pipe client and server pair. Listing 41.1 shows the server application.

Listing 41.1. A simple named pipe server.


#include <windows.h> void main(void) {

Named Pipes and Remote Procedure Calls CHAPTER 41


HANDLE hPipe; DWORD dwRead; char c = -1; hPipe = CreateNamedPipe(\\\\.\\pipe\\hello, PIPE_ACCESS_INBOUND, PIPE_WAIT, PIPE_UNLIMITED_INSTANCES, 256, 256, 1000, NULL); ConnectNamedPipe(hPipe, NULL); while (c != \0) { ReadFile(hPipe, &c, 1, &dwRead, NULL); if (dwRead > 0 && c != 0) putchar(c); } DisconnectNamedPipe(hPipe); CloseHandle(hPipe); }

735

41
NAMED PIPES AND REMOTE PROCEDURE CALLS

This application creates the pipe hello on the local server. The pipe is created in blocking mode (PIPE_WAIT); operations on this pipe will not return until they are completed. Once the pipe has been created, the server calls ConnectNamedPipe and waits for incoming client connections. When a client is connected, the server attempts to read from the pipe. It reads single bytes from the pipe and writes them to standard output until a null character is encountered; at that time, it disconnects the pipe, closes the handle, and terminates. The client, shown in Listing 41.2, takes two command-line parameters; the first identifies the host that it should connect to, the second represents the string that it sends to the server on that host.

Listing 41.2. A simple named pipe client.


#include <windows.h> #include <string.h> #include <stdio.h> void main(int argc, char *argv[]) { HANDLE hPipe; DWORD dwWritten; char *pszPipe; if (argc != 3) { printf(Usage: %s hostname string-to-print\n, argv[0]); exit(1); } pszPipe = malloc(strlen(argv[1]) + 14); sprintf(pszPipe, \\\\%s\\pipe\\hello, argv[1]); WaitNamedPipe(pszPipe, NMPWAIT_WAIT_FOREVER); hPipe = CreateFile(pszPipe, GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); free(pszPipe); WriteFile(hPipe, argv[2], strlen(argv[2])+1, &dwWritten, NULL); CloseHandle(hPipe); }

736

Networks and Communications PART VI

After processing its command-line parameters, the client begins to wait for the server to become available by calling WaitNamedPipe. When WaitNamedPipe returns, the client calls CreateFile to actually connect to the server; it then uses WriteFile to send the string to the server before shutting down the connection by calling CloseHandle. Both the client and the server can easily be compiled from the command line. Type cl to compile the server and cl pipec.c to compile the client.
pipes.c

You can test this application on a single machine running Windows NT (remember that Windows 95 does not support the server end of pipes) or on two machines across a network. If you are using a single machine, start pipes.exe in a DOS window and use a separate DOS window for running the client. When using the client, run it with a command line similar to the following:
pipec MYHOST Hello, World!

Make sure that in place of MYHOST you substitute the actual Windows name of the Windows NT host computer on which pipes.exe is run. (Remember that the server-side program can only be run under Windows NT.)

Microsoft Remote Procedure Calls


Microsofts RPC extend the conceptual elegance and simplicity of a function or subroutine call by providing a similar mechanism for calls across the network. RPC servers can offer a set of functions that can be called by RPC clients. The RPC mechanism is used widely in Windows. In particular, the RPC mechanism is the basis of the COM and OLE technologies.

RPC Fundamentals
The key to the RPC mechanism is stub functions. When an RPC client calls an RPC function, it issues a regular function call. However, the function that receives the call is an RPC stub; this stub function converts function arguments for transmission across the network (a procedure referred to as marshaling) and transmits the call request and the arguments to the server. Stub functions on the server side unmarshal function arguments and call the servers implementation of the function. When the function returns, its return value is passed back to the client using a reverse mechanism. This entire procedure is illustrated in Figure 41.2.

Named Pipes and Remote Procedure Calls CHAPTER 41

737

FIGURE 41.2.
RPC overview.

Client system

Server system

Client process Remote procedure call

Server process Remote procedure

41
NAMED PIPES AND REMOTE PROCEDURE CALLS

Stub function

Stub function

RPC runtime library

RPC runtime library

Network

When developing RPC applications, an element of central importance is the interface. Clearly, the stub functions on the client and the server sides must be based on identical function definitions; otherwise, the RPC process will surely fail. The tool that ensures that the stub functions on the two sides are compatible is the Microsoft Interface Development Language (MIDL) compiler.

A Simple Example
The next example clarifies the basic concepts of Microsoft RPC. This example, the RPC equivalent of the Hello, World application, implements a simple server function that prints a string received from client applications. The first step in developing this example is to specify the interface. This is done in the form of two files that represent the input files for the MIDL compiler. The MIDL compiler will produce three files: a header file that must be included in both the client and the server application, and two C source files that implement the client and server stub functions. These files must be linked with our implementation of both the client and the server application to produce the final executables. This process is illustrated in Figure 41.3.

738

Networks and Communications PART VI

FIGURE 41.3.
RPC development.

HELLOS.C

HELLO.IDL

HELLO.ACF

HELLOC.C

MIDL compiler

HELLO_S.C

HELLO.H

HELLO_C.C

C/C++ compiler and linker

C/C++ compiler and linker

HELLOS.EXE

HELLOS.EXE

Specifying the Interface


The interface for an RPC implementation is specified in the form of two files: the Interface Definition Language File and the Application Configuration File. These two files together serve as input for the MIDL compiler. The Interface Definition Language File for our example application is shown in Listing 41.3.

Listing 41.3. The Interface Definition Language File.


[ uuid (6fdd2ce0-0985-11cf-87c3-00403321bfac), version(1.0) ] interface hello { void HelloProc([in, string] const unsigned char *pszString); void Shutdown(void); }

This file consists of two parts: the interface header and the interface body. The interface header has the following syntax:
[ interface-attributes ] interface interface-name

Named Pipes and Remote Procedure Calls CHAPTER 41

739

Perhaps the most important of the interface attributes is the interfaces GUID, or globally unique identifier. The GUID is a 128-bit identifier that is supposed to be world-unique; in other words, no two applications in the world are supposed to have identical identifiers. The Visual C++ distribution provides a tool, the executable program guidgen.exe, that is supposed to create such unique identifiers using, in part, information obtained from your computers hardware, and in part a randomization algorithm. The GUID is expressed in the form of a string of 32 hexadecimal digits. The specific form is 8, 4, 4, 4, and 12 digits separated by hyphens. The guidgen.exe program generates GUID strings in this form. In addition to the GUID, another interface attribute specifies the interfaces version number. The function of the version number is to identify potentially incompatible versions of the interface. In the second part of the Interface Definition Language File, function prototypes are defined. The prototype syntax is similar to the syntax of the C language, but it also contains extra elements. The in keyword indicates to the MIDL compiler that the following parameter is an input-only parameter; that is, it is sent to the server by the client. The string keyword indicates that the data sent is a character array. The Application Configuration File is similar in syntax and appearance to the Interface Definition Language File. However, this file contains information on data and attributes not related to the actual transmission of RPC data. The Application Configuration File for our project is shown in Listing 41.4.

41
NAMED PIPES AND REMOTE PROCEDURE CALLS

Listing 41.4. The Application Configuration File.


[ implicit_handle(handle_t hello_IfHandle) ] interface hello { }

In this file, we specify a binding handle for the interface. This handle is a data object that represents the connection between the client and the server. Because the handle is not transmitted over the network, it is specified in the Application Configuration File. The use of the keyword implicit_handle prescribes a handle that is maintained as a global variable. This handle will be used by the client in calls to RPC runtime functions. The two files, hello.idl and hello.acf, can be compiled by the MIDL compiler using a single command:
midl hello.idl

740

Networks and Communications PART VI

The result of the compilation is three files produced by the MIDL compiler: hello_c.c, hello_s.c, and hello.h. The files hello_c.c and hello_s.c provide the client- and server-side implementation of stub functions; the hello.h header provides necessary declarations. By looking at these generated files, one can easily see just how much of the network programmers work is automated by the use of the MIDL compiler. The generated stub functions perform all the required runtime library calls to marshal and unmarshal arguments and to communicate across the network. However, the real beauty and elegant simplicity of the RPC mechanism are yet to become evident when we implement our client and server.

Implementing the Server


The server applications implementation is shown in Listing 41.5.

Listing 41.5. A simple RPC server.


#include <stdlib.h> #include <stdio.h> #include hello.h void HelloProc(const unsigned char *pszString) { printf(%s\n, pszString); } void Shutdown(void) { RpcMgmtStopServerListening(NULL); RpcServerUnregisterIf(NULL, NULL, FALSE); } void main(int argc, char * argv[]) { RpcServerUseProtseqEp(ncacn_ip_tcp, 20, 8000, NULL); RpcServerRegisterIf(hello_v1_0_s_ifspec, NULL, NULL); RpcServerListen(1, 20, FALSE); } void __RPC_FAR * __RPC_USER midl_user_allocate(size_t len) { return(malloc(len)); } void __RPC_USER midl_user_free(void __RPC_FAR * ptr) { free(ptr); }

The RPC runtime library is initialized and the server is set up to wait for incoming connections through the three RPC runtime library calls in the programs main function. First of these calls is the RpcServerUseProtseqEp call; this call defines the network protocol and endpoint

Named Pipes and Remote Procedure Calls CHAPTER 41

741

that is to be used by the application. In our example, I use the TCP over IP protocol (ncacn_ip_tcp) as a protocol that is supported both under Windows 95 and Windows NT. The RPC mechanism can utilize many other protocols, as shown in Table 41.1.

41
NAMED PIPES AND REMOTE PROCEDURE CALLS Table 41.1. RPC protocols. Protocol name
ncacn_ip_tcp ncacn_nb_tcp ncacn_nb_ipx ncacn_nb_nb ncacn_np ncacn_spx ncacn_dnet_nsp ncadg_ip_udp ncadg_ipx ncalrpc

Description TCP over IP TCP over NetBIOS IPX over NetBEUI NetBIOS over NetBEUI Named pipes SPX DECnet transport UDP over IP IPX local procedure call

Protocols with a name that begins with ncacn are connection-oriented protocols; those with a name that starts with ncadg are datagram (connectionless) protocols. Because Windows 95 supports named pipes for the client side only, the ncacn_np protocol is only supported for RPC client applications. Windows 95 does not support ncacn_nb_ipx and ncacn_nb_tcp. The ncacn_dnet_nsp protocol is supported only for 16-bit Windows and MSDOS clients.
ncacn_ip_tcp

The meaning of the endpoint parameter is dependent on the protocol. For example, when the protocol is used, the endpoint parameter represents a TPC port number.

Starting up the server consists of two steps. First, the server interface is registered; second, it enters a state where it listens for incoming connections. Registering the server makes it available for incoming client connections. The actual remote procedure, HelloProc, is implemented the same way as a local function. In fact, it is possible to place the implementation of this function in a separate file; this way, applications that locally call the function could be linked with it, while applications that call this function through RPC link with the client-side stub instead. Another function, Shutdown, has been provided to facilitate remote shutdown of the server. Server shutdown is accomplished by exiting the listening state and unregistering the server.

742

Networks and Communications PART VI

In addition to these two functions, the Microsoft RPC specifications require that we implement two additional memory management functions. The midl_user_allocate and midl_user_free functions are used to allocated a block of memory and to free up allocated memory. In simple cases, these can be mapped to the C runtime functions malloc and free; however, in large, complex applications these functions enable finer control over memory use. Both of these functions are used when arguments are marshaled or unmarshaled. To compile the server application, use the following command line:
cl hellos.c hello_s.c rpcrt4.lib

Implementing the Client


Implementing the RPC client is only slightly more complicated than implementing an application containing a local function call. The client implementation for our example is shown in Listing 41.6.

Listing 41.6. A simple RPC client.


#include #include #include #include <stdlib.h> <stdio.h> <string.h> hello.h

void main(int argc, char *argv[]) { unsigned char *pszStringBinding; if (argc != 3) { printf(Usage: %s hostname string-to-print\n, argv[0]); exit(1); } RpcStringBindingCompose(NULL, ncacn_ip_tcp, argv[1], 8000, NULL, &pszStringBinding); RpcBindingFromStringBinding(pszStringBinding, &hello_IfHandle); if (strcmp(argv[2], SHUTDOWN)) HelloProc(argv[2]); else Shutdown(); RpcStringFree(&pszStringBinding); RpcBindingFree(&hello_IfHandle); exit(0); } void { } void __RPC_USER midl_user_free(void __RPC_FAR * ptr) { free(ptr); } __RPC_FAR * __RPC_USER midl_user_allocate(size_t len) return(malloc(len));

Named Pipes and Remote Procedure Calls CHAPTER 41

743

This client implementation takes two command-line parameters: the name of the server to connect to and the string to be sent to the server. Because of the protocol being used (TCP over IP), the server name must be the hosts internet name or IP address. If you are testing the client and the server on the same host, you can use the default name for the local host, localhost, or its default IP address, 127.0.0.1. The protocol name, hostname, and endpoint are combined into a string binding using the function. For example, the string binding representing the ncacn_ip_tcp protocol, the local host, and TCP port 8000 would appear as follows:
RpcStringBindingCompose ncacn_ip_tcp:localhost[8000]

41
NAMED PIPES AND REMOTE PROCEDURE CALLS

The RpcStringBindingCompose function is merely a convenience function that frees the programmer from the task of having to assemble the string binding from its components by hand. This string binding is used to obtain the binding handle for the interface in the call to RpcBindingFromStringBinding. The receipt of the binding handle indicates that the connection is ready to be used. Once the connection is established, using it is simplicity itself. Calling a remote procedure becomes identical to calling a local function. In our client implementation, we call the function HelloProc with the second command-line argument; that is, unless that argument is the string SHUTDOWN, in which case the Shutdown function is called instead to shut down the remote server. Like the client, the server must also provide its implementations for midl_user_allocate and
midl_user_free.

The client application can be compiled using the following command line:
cl helloc.c hello_c.c rpcrt4.lib

After you have compiled both the server and the client executables, you can test the two programs from two DOS windows. After you start the server in one of the windows, run the client in the other window as follows:
helloc localhost Hello, World!

To shut down the server from the client side, type


helloc localhost SHUTDOWN

RPC Exception Handling


If you attempt to run the client application developed in the previous section alone, without starting up a server, a serious deficiency of our implementation becomes evident. Neither the client nor the server in this example performs any error handling; in particular, this means that the client does not respond well to situations when no server is available.

744

Networks and Communications PART VI

Unlike other errors, the unavailability of a server should be considered a likely occurrence that must be handled. (Not that I am suggesting that handling of other errors can or should be neglected in production applications!) The Microsoft RPC implementation provides a special mechanism for this purpose that is very similar in its appearance to Win32 structured exceptions or C++ exception handling. By protecting the remote procedure calls using the RPC exception handling macros, one can ensure graceful handling of network error conditions by a client application. For example, in our Hello, World application, the calls to HelloProc and Shutdown on the client side could be protected as follows:
RpcTryExcept { if (strcmp(argv[2], SHUTDOWN)) HelloProc(argv[2]); else Shutdown(); } RpcExcept(1) { printf(RPC run-time exception %08.8X\n, RpcExceptionCode()); } RpcEndExcept

If you recompile the client application with this change, it will no longer crash when the server is not available; it will gracefully terminate, printing the exception number.

Advanced RPC Features


Although this example application demonstrates the simplicity of using Microsoft RPC adequately, it only hints at the power and rich features of the RPC mechanism. This section mentions a few of the most notable features of Microsoft RPC. The MIDL compiler can be used for specifying remote procedures that accept all kinds of arguments. This includes pointers and arrays; however, pointers and arrays require special consideration. Because the RPC stub functions must not only marshal the pointer arguments themselves but also the data they point to, it is necessary to define the size of the memory block a pointer points to in the interface specification. A series of attributes is available for specifying an arrays size. For example, consider a remote procedure that takes the size of an array and a pointer to it as its parameters:
void myproc(short s, double *d);

You can identify the interface for such a procedure in your IDL file as follows:
void myproc([in] short s, [in, out, size_is(d)] double d[]);

This informs the MIDL compiler to generate stub code that marshals s number of array elements. The size_is attribute is not the only attribute that assists in specifying array arguments. Others include length_is, first_is, last_is, and max_is.

Named Pipes and Remote Procedure Calls CHAPTER 41

745

The RPC mechanism can utilize the Microsoft RPC Name Service Provider on Windows NT. Through this mechanism, it is possible for clients to locate an RPC server by name. In particular, the use of RPC Name Service enables you to develop clients that do not use an explicit binding handle (like our Hello, World example did). Such clients would contain no RPC runtime library calls whatsoever and apart from being linked with the client-side stub, they look no different from programs that use local functions. The MIDL syntax enables you to define interfaces that are derived from other interfaces. The syntax is similar to that used for deriving classes in C++:
[attributes] interface interface-name : base-interface

41
NAMED PIPES AND REMOTE PROCEDURE CALLS

The derived interface inherits member functions, status codes, and interface attributes from the base interface.

Summary
Pipes represent a simple, efficient interapplication communication mechanism supported by Windows 95 (client side only) and Windows NT. Pipes can be named and unnamed. Anonymous pipes are typically used between a parent and a child process or two sibling processes. The use of anonymous pipes requires communicating a pipe handle from one process to another (for example, by inheriting the handle). Because anonymous pipes are identified by handle alone, they cannot be used for communication across a network. Unlike anonymous pipes, named pipes support overlapped I/O operations. A server can use a combination of techniques, including overlapped I/O and using separate threads to serve several clients simultaneously. Both servers and clients communicate on a named pipe using the pipes handle and standard Win32 input and output functions. Microsoft RPC is a mechanism for applications to call functions remotely. It provides a transparent interface where client applications can call remote functions in a fashion that is very similar to the calling of local functions. The key to Microsoft RPC, in addition to the RPC runtime library, is the Microsoft Interface Definition Language (MIDL) compiler. The interface between a client and a server can be specified using a simple C-like syntax, from which the MIDL compiler generates server and client-side stub functions. This frees the programmer from the burden of complex network programming tasks and from having to maintain compatible versions of these functions on the server and client sides.

746

Networks and Communications PART VI

The actual implementation is through these stub functions. When a client calls a remote procedure, the call is handled by the corresponding client-side stub function. The stub function, in turn, calls the RPC runtime library that uses the underlying network transport to invoke stub functions on the server side. The client-side stub function also marshals function arguments for transmission over the network. The server-side stub function unmarshals these arguments and calls the actual function implementation. When the function returns, this process is played out in reverse, as return values are transported back to the client application. Advanced RPC features include RPC Name Service, pointer and array function arguments, derived interfaces, and much more.

IN THIS PART
s Multimedia Applications 749 s The OpenGL Graphics Library 767 s High-Performance Graphics and Sound: DirectX 783

VII
PART

Graphics and Multimedia

Multimedia Applications CHAPTER 42

749

Multimedia Applications

42

IN THIS CHAPTER
s Video Playback with One Function Call 750 s Fundamentals of Multimedia Programming 752 s Programming with MCIWnd 754 s The Media Control Interface s Advanced Interfaces 763 759

42
MULTIMEDIA APPLICATIONS

750

Graphics and Multimedia PART VII

It has been several years since Microsoft Windows became a platform of choice for multimedia applications. The multimedia programmer today has a dizzying array of technologies when it comes to developing multimedia solutions. In this chapter we review the traditional software technologies for Windows multimedia: MCI (the Media Control Interface) and related software libraries. The advanced multimedia technologies of DirectX 5 are discussed in another chapter. Despite the availability of DirectX 5, there are still advantages to using old style media services in Windows. They are very simple to use, robust, and guaranteed to be available as part of a standard operating system installation.

Video Playback with One Function Call


Although multimedia programming under Windows can be quite complex, applications can accomplish many things by a few simple function calls. Nothing demonstrates this better than the MCIWndCreate function, which can be used to replay videos in a single function call. Our review of Windows multimedia programming begins with a closer look at this function; for many simple applications, you may not need anything more sophisticated. The program shown in Listing 42.1 can perhaps be viewed as the Windows multimedia equivalent of a Hello, World application. This program takes a single command-line parameter, the name of a multimedia file such as an AVI video file, and plays it back in its window. It can also be launched without a parameter; in that case, use the button controls that appear in its window (see Figure 42.1) to open a file for playback.

FIGURE 42.1.
AVI playback using
MCIWndCreate.

Listing 42.1. A simple multimedia playback application.


#include <windows.h> #include <vfw.h> void SetClientRect(HWND hwnd, HWND hwndMCI)

Multimedia Applications CHAPTER 42


{ RECT rect; GetWindowRect(hwndMCI, &rect); AdjustWindowRectEx(&rect, GetWindowLong(hwnd, GWL_STYLE), FALSE, GetWindowLong(hwnd, GWL_EXSTYLE)); MoveWindow(hwnd, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, TRUE); } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case MCIWNDM_NOTIFYPOS: case MCIWNDM_NOTIFYSIZE: SetClientRect(hwnd, (HWND)wParam); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); SetClientRect(hwnd, MCIWndCreate(hwnd, hInstance, WS_VISIBLE | WS_CHILD | MCIWNDF_SHOWALL | MCIWNDF_NOTIFYSIZE | MCIWNDF_NOTIFYPOS, lpCmdLine)); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); return msg.wParam; }

751

42
MULTIMEDIA APPLICATIONS

752

Graphics and Multimedia PART VII

All right, I cheated. In addition to the single function call, we also have a few lines of code handling notification messages. By responding to MCIWNDM_NOTIFYPOS and MCIWNDM_NOTIFYSIZE messages, we can ensure that the applications main window is automatically resized. That way, we end up with a window that is properly sized for video playback. Nevertheless, the application demonstrates the power of MCIWndCreate well; with that one function call, we have been able to create a full-featured video playback application. This simple program can be compiled from the command line by typing cl vfw32.lib.
hello.c user32.lib

MCIWndCreate is not the only simple function that can be used for media playback. Another such function is the Playback function; this function can be used to play waveform audio files.

Fundamentals of Multimedia Programming


Multimedia represents the operating systems capability to record and play video and sound and to control multimedia equipment. Multimedia programming is accomplished through a series of interfaces, or APIs, that applications can utilize for this purpose.

Multimedia Data Formats


Windows recognizes three fundamental multimedia formats: waveform audio, MIDI sequences, and video (see Figure 42.2).

FIGURE 42.2.
Multimedia in Windows.

2 4

AVI Playback

Waveform audio is sampled, digitized audio data. Waveform audio is typically stored in files with the .wav extension. Windows recognizes waveform audio files with mono and stereo data, a variety of sampling rates, and sampling depths. There are also several different compression methods used for the efficient storage of waveform data.

Multimedia Applications CHAPTER 42

753

MIDI is the acronym for Musical Instrument Digital Interface. This international standard specifies a protocol for interfacing computers and electronic musical instruments. MIDI sequences are data that can be played back on MIDI-compatible instruments. MIDI data is stored under windows in files with the .mid extension. MIDI files can be played back on external devices connected to the computer or on built-in synthesizers that support MIDI capabilities. The most widely used Windows video format is the Audio Video Interleaved (AVI) file format. AVI files can be used to store a motion video stream and one or more audio channels. Windows recognizes video data at varying resolutions, color depths, and refresh rates. There are also several compression formats in wide use. Other video formats, such as MPEG and QuickTime, are also supported through drivers supplied by Microsoft or third parties. Windows multimedia functions can also be used for audio-CD playback. Using third-party drivers, recording and playback of other multimedia formats are also possible.

42
MULTIMEDIA APPLICATIONS

Multimedia Interfaces
Depending on your programming needs, you can choose one of three interface levels to interact with the multimedia subsystem in Windows. The high-level interface is based on the MCIWnd window class. In the sample program shown in Listing 42.1, the MCIWnd function enables video playback with only a single function call. The mid-level interface is the Media Control Interface, or MCI. MCI provides a deviceindependent command-message and command-string interface for the playback and recording of audio and visual data. At the lowest level, there are several interfaces for waveform audio, video, and MIDI recording and playback. Additional interfaces provide audio mixer capabilities, buffered file I/O, and joystick and timer control. All these interfaces serve but one purpose: to provide a programming interface between enduser applications on the one hand, and drivers for multimedia hardware on the other. Through these drivers, device-independence in Windows is achieved. Which interface should you choose for your application? That depends on the needs and requirements of your project. Applications that require only simple playback capabilities are good candidates for using the services. For example, an encyclopedia application that offers video clips accompanying some articles may use an MCIWnd window for video playback.
MCIWnd

An example of a more demanding multimedia program is an audio recording and playback application. Such a program should probably rely on the services of the Media Control Interface to implement its functionality. A sophisticated multimedia application, such as a video capture and mixer application, would probably require using low-level video and file services.

754

Graphics and Multimedia PART VII

Programming with MCIWnd


The MCIWnd window class represents the simplest, highest-level multimedia programming interface in Windows. Applications that require simple playback capabilities can utilize MCIWnd windows for this purpose; such a window can be created with a single function call, as demonstrated by the program in Listing 42.1.

The MCIWnd Window Class


Windows of class MCIWnd provide a user interface that consists of up to four buttons, a trackbar, and an optional playback area (see Figure 42.3).

FIGURE 42.3.
MCIWnd window controls.

The Play and Stop buttons can be used to start and stop playback. Playback starts at the current position indicated by the trackbar. Special playback effects can be utilized by holding down the Shift or Control keys while clicking on the Play button; holding down the Control key results in full-screen video playback, whereas holding down the Shift key results in backward play. The Menu button can be used to invoke a pop-up menu. The options in this menu are specific to the type of media file currently selected. For example, if an AVI file is loaded, the pop-up menu includes options to set the video playback speed, sound volume, zoom, and video configuration. Other commands enable you to copy the current data to the Windows clipboard and to open another file. Yet another menu option enables you to send an MCI command string directly to the currently active multimedia device. The Record button can be made available for devices that can record. The trackbar is used to display the current playback or recording position relative to the size of the file. The trackbar can also be used to move to different locations in the file during playback. Optimal video playback performance requires that the playback window be aligned on a fourpixel boundary. Normally, Windows aligns the playback window automatically.

Multimedia Applications CHAPTER 42

755

MCIWnd Functions
An
MCIWnd MCIWnd

window is created by a call to MCIWndCreate. A call to this function registers the class and creates an MCIWnd window.

In addition to specifying the handle of the parent window and an instance handle, parameters to this function also specify a set of window styles and an optional filename. The window style settings control which elements of the MCIWnd window are visible and how the window interacts with the user on the one hand and the application code on the other. For example, by specifying the MCIWNDF_RECORD window style, you can create an MCIWnd window with a visible Record button. Specifying the MCIWNDF_NOTIFYSIZE causes the MCIWnd window to send notification messages to its parent whenever the windows size changes.
MCIWnd windows can be created as child windows or overlapped windows. If created as an over-

42
MULTIMEDIA APPLICATIONS

lapped window, an MCIWnd window will have a title bar with contents specified with the appropriate style settings (MCIWNDF_SHOWMODE, MCIWNDF_SHOWNAME, or MCIWNDF_SHOWPOS). windows can also be created via calls to CreateWindow or CreateWindowEx. Before you can do so, however, you must call MCIWndRegisterClass. This function registers the window class specified by the constant MCIWND_WINDOW_CLASS.
MCIWnd

There are two additional functions that use windows of class

MCIWnd .

The functions

GetOpenFileNamePreview and GetSaveFileNamePreview enhance the standard GetOpenFileName

and GetSaveFileName functions by adding a multimedia preview window to the standard file open dialog (see Figure 42.4).

FIGURE 42.4.
A file open dialog with a preview window, created using GetOpen
FileNamePreview.

The program in Listing 42.2 demonstrates the use of this program, type cl ofnp.c vfw32.lib.

GetOpenFileNamePreview.

To compile

756

Graphics and Multimedia PART VII

Listing 42.2. Using GetOpenFileNamePreview.


#include <windows.h> #include <vfw.h> int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { OPENFILENAME ofn; memset(&ofn, 0, sizeof(ofn)); ofn.lStructSize = sizeof(ofn); GetOpenFileNamePreview(&ofn); }

MCIWnd Macros
Applications can communicate with an MCIWnd window by sending messages to it using the Windows SendMessage function. A large number of helper macros exist that simplify sending most of these messages. Through these messages, applications can control the appearance and behavior of the MCIWnd window, start and stop playback and recording, close the MCI device or file and open a new file or device for recording or playback, seek specific playback and recording positions, retrieve information on device capabilities and current settings, specify MCI device attributes, and control the MCI device. Table 42.1 summarizes MCIWnd messages and helper macros.

Table 42.1. MCIWnd messages and macros. Message Macro


MCI_CLOSE MCI_OPEN MCI_PAUSE MCI_PLAY MCI_RECORD MCI_RESUME MCI_SAVE MCI_SAVE MCI_SEEK MCI_SEEK MCI_SEEK MCI_STEP MCIWndClose MCIWndOpenDialog MCIWndPause MCIWndPlay MCIWndRecord MCIWndResume MCIWndSave MCIWndSaveDialog MCIWndEnd MCIWndHome MCIWndSeek MCIWndStep

Description Close MCI device or file Open data file Pause playback or record Start playback Start recording Resume playback or record Save content Save content Move to end of content Move to start of content Move to position Move position

Multimedia Applications CHAPTER 42

757

Message
MCI_STOP MCIWNDM_CAN_CONFIG MCIWNDM_CAN_EJECT MCIWNDM_CAN_PLAY MCIWNDM_CAN_RECORD MCIWNDM_CAN_SAVE MCIWNDM_CAN_WINDOW

Macro
MCIWndStop MCIWndCanConfig MCIWndCanEject MCIWndCanPlay MCIWndCanRecord MCIWndCanSave MCIWndCanWindow

Description Stop playback or record Can configure device? Can eject media? Can play device? Can record on device? Can save to file? Window commands supported? Change window style Eject media Return update period Return device alias Return playback rectangle Return device name Return device identifier Return end location Return last MCI error Return current filename Return update period Return content length Return current mode Return MCI palette handle Return position Return position Continuous playback? Return cropping rectangle Return playback speed Return start location Return window style Return time format Return volume setting Return zoom setting
continues

MCIWNDM_CHANGESTYLES MCIWNDM_EJECT MCIWNDM_GETACTIVETIMER MCIWNDM_GETALIAS MCIWNDM_GET_DEST MCIWNDM_GETDEVICE MCIWNDM_GETDEVICEID MCIWNDM_GETEND MCIWNDM_GETERROR MCIWNDM_GETFILENAME MCIWNDM_GETINACTIVETIMER MCIWNDM_GETLENGTH MCIWNDM_GETMODE MCIWNDM_GETPALETTE MCIWNDM_GETPOSITION MCIWNDM_GETPOSITION MCIWNDM_GETREPEAT MCIWNDM_GETSOURCE MCIWNDM_GETSPEED MCIWNDM_GETSTART MCIWNDM_GETSTYLES MCIWNDM_GETTIMEFORMAT MCIWNDM_GETVOLUME MCIWNDM_GETZOOM

MCIWndChangeStyles MCIWndEject MCIWndGetActiveTimer MCIWndGetAlias MCIWndGetDest MCIWndGetDevice MCIWndGetDeviceID MCIWndGetEnd MCIWndGetError MCIWndGetFileName MCIWndGetInactiveTimer MCIWndGetLength MCIWndGetMode MCIWndGetPalette MCIWndGetPosition MCIWndGetPositionString MCIWndGetRepeat MCIWndGetSource MCIWndGetSpeed MCIWndGetStart MCIWndGetStyles MCIWndGetTimeFormat MCIWndGetVolume MCIWndGetZoom

42
MULTIMEDIA APPLICATIONS

758

Graphics and Multimedia PART VII

Table 42.1. continued Message


MCIWNDM_NEW MCIWNDM_OPEN MCIWNDM_OPENINTERFACE MCIWNDM_PLAYFROM MCIWNDM_PLAYTO MCIWNDM_PLAYREVERSE MCIWNDM_PLAYTO MCIWNDM_PUT_DEST MCIWNDM_PUT_SOURCE MCIWNDM_REALIZE MCIWNDM_RETURNSTRING MCIWNDM_SENDSTRING MCIWNDM_SETACTIVETIMER MCIWNDM_SETINACTIVETIMER MCIWNDM_SETOWNER MCIWNDM_SETPALETTE MCIWNDM_SETREPEAT MCIWNDM_SETSPEED MCIWNDM_SETTIMEFORMAT MCIWNDM_SETTIMERS MCIWNDM_SETVOLUME MCIWNDM_SETZOOM MCIWNDM_SETTIMEFORMAT MCIWNDM_SETTIMEFORMAT MCIWNDM_VALIDATEMEDIA WM_CLOSE

Macro
MCIWndNew MCIWndOpen MCIWndOpenInterface MCIWndPlayFrom MCIWndPlayFromTo MCIWndPlayReverse MCIWndPlayTo MCIWndPutDest MCIWndPutSource MCIWndRealize MCIWndReturnString MCIWndSendString MCIWndSetActiveTimer MCIWndSetInactiveTimer MCIWndSetOwner MCIWndSetPalette MCIWndSetRepeat MCIWndSetSpeed MCIWndSetTimeFormat MCIWndSetTimers MCIWndSetVolume MCIWndSetZoom MCIWndUseFrames MCIWndUseTime MCIWndValidateMedia MCIWndDestroy

Description Create new file Open MCI device and file Open IAVI interface Playback at position Playback range Playback in reverse Playback to position Change playback rectangle Change cropping rectangle Realize MCI palette Return MCI reply Send MCI command Set update period Set update period Set owner window Set MCI palette Set repeat mode Set playback speed Set time format Set update period Set volume Sets video zoom Set time format Set time format Updates positions Close MCIWnd window

MCIWnd Notifications
If enabled, MCIWnd windows can send notification messages to their parent windows. Specifically, five types of notification messages can be sent. All five can be enabled by specifying the MCIWNDF_NOTIFYALL window style when creating the window. Alternatively, notification messages can be enabled individually.

Multimedia Applications CHAPTER 42

759

The MCIWNDM_NOTIFYERROR message is sent to the parent window to notify it of MCI errors. This notification can be enabled by specifying the MCIWNDF_NOTIFYERROR window style. The MCIWNDM_NOTIFYMEDIA message notifies the parent window of any media changes that may have occurred. These messages are enabled by specifying the MCIWNDF_NOTIFYMEDIA window style. The parent window is notified of changes in the MCIWnd windows position and size through MCIWNDM_NOTIFYPOS and MCIWNDM_NOTIFYSIZE messages. These messages are enabled by the MCIWNDF_NOTIFYPOS and MCIWNDF_NOTIFYSIZE window styles, respectively. Finally, the parent window is notified of any operating mode changes (for example, changes from play to stop mode) by MCIWNDM_NOTIFYMODE messages. These messages are enabled by the MCIWNDF_NOTIFYMODE window style.

The Media Control Interface


The Media Control Interface provides a set of device-independent command messages and command strings for controlling multimedia devices. Command messages and command strings can be used interchangeably. The MCI recognizes a variety of different multimedia devices. These devices are listed in Table 42.2.

42
MULTIMEDIA APPLICATIONS

Table 42.2. Multimedia devices. Device name Description


animation cdaudio dat digitalvideo other overlay scanner sequencer vcr videodisc waveaudio

Animation device Audio CD player Digital-audio tape player Non GDI-based digital video in a window Undefined device Analog video in a window Image scanner MIDI sequencer Video cassette recorder or player Video disc player Waveform audio device

Both command messages and command strings can be used to control devices and to retrieve information from devices. Command messages retrieve information in the form of structures, which are easy to interpret in C programs. Command strings retrieve information in the form of strings that must be parsed and interpreted by the application.

760

Graphics and Multimedia PART VII

All MCI devices support a core set of MCI commands and messages. Many devices support additional, device-specific commands. Command strings are sent to devices using the mciSendString function. A simple use of this function is demonstrated in Listing 42.3. This application, which takes a single filename as its command-line argument, plays back a multimedia file. For example, if you specify the pathname for an AVI video file, this application will play back the video in full-screen mode. To compile this application, type cl mcistr.c user32.lib winmm.lib.

Listing 42.3. MCI playback using command strings.


#include <windows.h> #include <stdlib.h> #define CMDSTR OPEN %s ALIAS MOVIE int WINAPI WinMain(HINSTANCE d1, HINSTANCE d2, LPSTR lpCmdLine, int d4) { char *pBuf; pBuf = malloc(sizeof(CMDSTR) + strlen(lpCmdLine) - 2); if (!pBuf) return -1; wsprintf(pBuf, CMDSTR, lpCmdLine); mciSendString(pBuf, NULL, 0, NULL); free(pBuf); mciSendString(PLAY MOVIE WAIT, NULL, 0, NULL); mciSendString(CLOSE MOVIE, NULL, 0, NULL); return 0; }

Command messages are sent to devices using the mciSendCommand function. This function takes several parameters, one of which is a command-specific structure. The structure may either be a general purpose one, such as MCI_OPEN_PARMS, or a device-specific extension to the general purpose version. Applications fill in this structure as appropriate prior to executing the call to mciSendCommand; commands that return information do so by modifying elements in this structure. The program in Listing 42.4 demonstrates simple playback using the MCI command message interface. This program can be compiled from the command line by typing cl mcimsg.c winmm.lib. Like its command string counterpart, this program also takes the name of a multimedia file on the command line and plays back that file.

Listing 42.4. MCI playback using command messages.


#include <windows.h> #include <stdlib.h> int WINAPI WinMain(HINSTANCE d1, HINSTANCE d2, LPSTR lpCmdLine, int d4) {

Multimedia Applications CHAPTER 42


MCI_OPEN_PARMS mciOpen; MCI_PLAY_PARMS mciPlay; MCI_GENERIC_PARMS mciClose; mciOpen.dwCallback = 0; mciOpen.lpstrElementName = lpCmdLine; mciOpen.lpstrAlias = MOVIE; mciSendCommand(0, MCI_OPEN, MCI_OPEN_ALIAS | MCI_OPEN_ELEMENT, (DWORD)&mciOpen); mciPlay.dwCallback = 0; mciSendCommand(mciOpen.wDeviceID, MCI_PLAY, MCI_WAIT, (DWORD)&mciPlay); mciClose.dwCallback = 0; mciSendCommand(mciOpen.wDeviceID, MCI_CLOSE, 0, (DWORD)&mciClose); return 0; }

761

42
MULTIMEDIA APPLICATIONS

MCI Command String Syntax


The generic syntax for MCI command strings is as follows:
command identifier [argument [, argument]]

The command portion specifies an MCI command such as PLAY, OPEN, or CLOSE. The identifier identifies an MCI device. This may be an MCI device name or an alias name. This identifier represents an instance of the appropriate MCI driver that was created when the device was opened. Command arguments are used to specify flags and parameters specific to each MCI command. For example, consider the following MCI command string:
play music from 0 to 100 wait

In this string, play is an MCI command; music is (presumably) an alias that was created when the device was opened; and from 0 to 100 wait is a series of arguments and flags applicable to the play command. The wait flag in this command specifies that the call to mciSendString should not return before the command is finished. Normally, calls return immediately and the commands are processed in the background. Another commonly used flag is the notify flag. Specifying this flag causes the MCI to post a multimedia notification (MM_MCINOTIFY) message to the application whenever a command is completed. Some devices (for example, digital video devices) support the test flag. A command submitted with this flag is not executed; however, the MCI tests whether the command can be executed by the specified device and returns an error if that is not the case.

762

Graphics and Multimedia PART VII

These flags can also be specified for commands submitted in the form of MCI command messages.

MCI Command Sets


MCI commands fall into different categories. These include system commands, required commands, and optional commands. The two system commands, break and sysinfo, are recognized and processed by the MCI itself. The break command is used to set a virtual key code that aborts other MCI commands; the sysinfo command returns information on MCI services and devices. Required commands are those that every MCI device must implement. This set includes the following five commands: capability, close, info, open, and status. The capability, info, and status commands obtain information on the status and capabilities of the device. The open and close commands are used to open or close the device. Optional commands can be further broken down into two categories: basic commands and extended commands. Basic commands include load and save, play, record, and stop, pause and resume, seek and set, and certain forms of the status command. For most devices, it is reasonable to assume that a subset of these commands applicable to the device is supported. For example, a playback device can reasonably be expected to support at least the play and stop commands; a recording device can be expected to support record and stop. Extended commands include several additional configuration and editing commands. Your application should not expect that any of these commands are supported; instead, it should use the capability command to find out about the available set of commands.

MCI Functions and Macros


You already have encountered two MCI functions: mciSendCommand is used to send an MCI command message, while mciSendString is used to send an MCI command string. Another set of three functions can be used to retrieve information about an MCI device. The mciGetCreatorTask function returns the handle of the task that created a specific MCI device. The mciGetDeviceID function returns the MCI device identifier for a named device. The mciGetErrorString returns the error message string corresponding to a specific error code. Two additional functions, mciGetYieldProc and mciSetYieldProc, can be used for yield procedures. The yield procedure is a function that the MCI calls regularly while waiting for the completion of a command that was issued with the wait flag. The MCI also offers a series of macros that deal with time formats. Various time formats are used in MCI commands to set the recording or playback position, or to read back the current position. Positions can be expressed as time values, track positions, and so on. Time format macros are used to create position values and to retrieve individual elements of a position value.

Multimedia Applications CHAPTER 42

763

For example, MCI_HMS_HOUR retrieves the hour component of a position expressed in the form of hours, minutes, and seconds (HMS); MCI_MAKE_HMS creates a time value from parameters specifying hours, minutes, and seconds.

MCI Notifications
MCI devices can send two types of messages to the application. The MM_MCINOTIFY message is used to notify the application of the completion of a command. Parameters to this message identify the command and specify whether the command was successfully completed, or whether its execution was interrupted due to an error or some other condition. The MM_MCISIGNAL message is used specifically in response to the extended MCI command signal. Through this command, applications can request that the MCI send an MM_MCISIGNAL message when a specific spot in the content is reached.

42
MULTIMEDIA APPLICATIONS

Advanced Interfaces
Windows offers several low-level interfaces for manipulating multimedia.

AVIFile and AVIStream Functions


AVIFile

functions and macros offer low-level access to files containing Resource Information File Format (RIFF) data; examples of such files include digital video and waveform audio files. used to open and close files, place files to the Windows clipboard, and obtain and manipulate file properties. AVIStream functions can be used to read or write the actual audio or video data.

AVIFile functions are based on the OLE Component Object Model. AVIFile functions can be

Custom File and Stream Handlers


For video and audio data sources other than AVI video and waveform audio files, you can write custom file and stream handlers. Custom file and stream handlers are installable drivers that provide access to data in different sources using the OLE Component Object Model. Custom handlers are dynamic link libraries (inproc OLE servers) that implement the IAVIFile and IAVIStream interfaces. These interfaces are used by the AVIFile and AVIStream family of functions.

DrawDib Functions
DrawDib

functions provide high-performance capabilities for drawing device-independent bitmaps (DIBs) in 8-bit, 16-bit, 24-bit, and 32-bit graphics modes.

764

Graphics and Multimedia PART VII


DrawDib functions do not rely on the GDI, but write directly to video memory. They provide a variety of services ranging from image stretching and dithering, to compression and decompression of many known formats.

The Video Compression Manager


The video compression manager, or VCM, provides access to installable compressors that handle real-time video data. Applications can use the VCM to compress and decompress video data and handle the interaction between compressed video data, custom data, and renderers.

Video Capture
Video capture can be accomplished using the AVICap window class and a series of related functions and macros. In the simplest scenario, applications create an A V I C a p window through the capCreateCaptureWindow function and send messages to this window to control capturing. A large number of macros exist that simplify the sending of messages to AVICap windows and the processing of message results.

Waveform Audio Recording and Playback


Windows offers a series of functions dealing with recording and playback of waveform audio. The simplest is the PlaySound function that enables you to play audio files that fit into available memory. Other waveform audio functions can be used to control individual waveform input and output devices.

The Audio Compression Manager


Windows also provides a programming interface to the audio compression manager, or ACM. The ACM provides for the transparent compression and decompression of waveform audio data during recording and playback. The ACM is installed as a mapper. This means that the ACM can intercept audio recording and playback requests and decode or encode data as necessary. The ACM can also search for a waveform device or an ACM compressor or decompressor that can handle a specific format.

MIDI Recording and Playback


MIDI file playback is performed by the MCI MIDI sequencer. Low-level MIDI functions enable applications to control MIDI playback and recording and to process MIDI data.

Multimedia Applications CHAPTER 42

765

Audio Mixers
Audio lines can be controlled through mixer devices. Windows offers a series of functions for opening and using mixer devices.

Miscellaneous Multimedia Services


Other multimedia services include a series of functions for low-level buffered file I/O optimized for media recording and playback; file services specific to Resource Interchange File Format (RIFF); functions and notification messages for handling joystick devices; and functions to create and manage high-resolution multimedia timers.

Summary
Todays Windows programmer can choose from several technologies when dealing with multimedia programming. In this chapter you learned how to program through the Media Control Interface and some low-level services. Multimedia in windows consists of the capability to record and play video and audio files. Windows recognizes one video format (AVI files) and two audio formats (MIDI and waveform audio). Windows offers multimedia interfaces on three distinct levels. At the highest level is the MCIWnd window class. Through this window class, it is possible to perform video playback with a single function call. The MCIWnd API also offers a large number of macros that utilize the SendMessage function to communicate with an MCIWnd window. These windows can also send notification messages to their parent windows. The mid-level multimedia programming interface is the Media Control Interface, or MCI. MCI offers command strings and command messages for controlling multimedia devices; command strings and command messages can be used interchangeably. All MCI devices recognize a set of core commands; many devices recognize a set of additional, device-specific commands. Although using the command string interface is generally simpler, for commands that retrieve information from the device it is often advantageous to use command messages instead. Unlike command strings, which return responses in the form of strings that need to be parsed and interpreted by the application, command messages return information in the form of structures. Low-level video services include the AVIFile family of functions, interfaces to video compression and video capture, and high-performance functions for drawing device-independent bitmaps. For data in nonstandard sources, custom file and stream handler drivers can be developed.

42
MULTIMEDIA APPLICATIONS

766

Graphics and Multimedia PART VII

The AVIFile function family as well as custom file and stream handlers are applicable to waveform audio. Other low-level audio services include functions for recording and playback of waveform audio and MIDI data. Additional interfaces are provided to access the audio compression manager and audio mixer devices. Other low-level multimedia interfaces include high-performance buffered file I/O, joystick control, and multimedia timers.

The OpenGL Graphics Library CHAPTER 43

767

The OpenGL Graphics Library

43

IN THIS CHAPTER
s OpenGL Overview 768 s Writing OpenGL Windows Applications in C 772 s OpenGL in MFC Applications 776

43
THE OPENGL GRAPHICS LIBRARY

768

Graphics and Multimedia PART VII

OpenGL is a device- and operating system-independent library for three-dimensional graphics and graphics rendering. OpenGL was originally developed by Silicon Graphics Inc. (SGI) for use on their high-end graphics workstations. Since then, OpenGL has become a widely accepted standard with implementations on many operating system and hardware platforms, including the Windows NT and Windows 95/98 operating systems.

NOTE
Windows 98, as well as Windows 95 Service Release 2, now come with OpenGL support; however, OpenGL support was not available in the initial release of Windows 95. Therefore, it may be necessary to redistribute the OpenGL runtime files with your OpenGL application. These DLLs can be found on the October 1995 or later release of the Microsoft Developer Network professional subscription CDs. They can also be downloaded from the Microsoft Web site.

In addition to the standard OpenGL Library implementation, Windows also provides a series of functions that integrate OpenGL with the operating system. In particular, functions are provided that associate OpenGL rendering contexts with GDI device contexts. These Windows extensions to the OpenGL Library are identified by names that begin with wgl. In addition to these OpenGL extensions, a series of new Win32 API functions has also been defined to facilitate certain aspects of OpenGL programming. The OpenGL Library is large and complex. If you want access to a comprehensive set of manuals, you should consider purchasing The OpenGL Reference Manual from the OpenGL Architecture Review Board, or The OpenGL Programming Guide by Jackie Neider, Tom Davis, and Mason Woo. Both books are published by Addison-Wesley. In this chapter, in addition to presenting a brief (and far from comprehensive) overview of the OpenGL Library, I focus on using OpenGL from Windows and MFC applications.

OpenGL Overview
The purpose of the OpenGL Library is to render two- and three-dimensional objects into a frame buffer, such as the pixel memory of your computers graphics hardware. The OpenGL Library is fundamentally procedural. What this means is that in your application, you dont describe what an object looks like; instead, you specify how an object is to be drawn. Complex geometric objects are described in terms of simple elements that your application defines. The OpenGL Library implementation follows the client-server model. OpenGL clients and servers need not even reside on the same machine.

The OpenGL Graphics Library CHAPTER 43

769

Basic OpenGL Concepts


At the basic level, the OpenGL Library deals with vertices. A vertex is a point, for example the end point of a line, or a corner of a polygon. Vertices can be two- or three-dimensional. At the next level are primitives. Primitives consist of a group of one or more vertices. For example, a rectangle described as a set of four vertices is a primitive. A variety of settings control how vertices are assembled into primitives and how primitives are drawn into a frame buffer. For example, applications can specify a three-dimensional transformation matrix that defines how the coordinates of an object are translated into coordinates on the drawing surface. In addition to its capability to draw points and lines, OpenGL can also draw surfaces, apply lighting specifications, and use texture bitmaps. Another set of features enables applications to selectively use or discard pixels. For example, drawing a pixel can be made conditional upon properties such as the pixels depth or its opacity. A greatly simplified view of how OpenGL works is presented in Figure 43.1.

43
THE OPENGL GRAPHICS LIBRARY

FIGURE 43.1.
Simplified overview of OpenGL operations.

Verticles

Lighting, color, and other parameters

Texture settings

Primtive Assembly

Primitives

Clipping and transformation settings

Fragment and pixel operations

Rasterization

Frame buffer

770

Graphics and Multimedia PART VII

Initialization
Before the OpenGL Library can be used, a number of initialization steps must be executed. Every Windows OpenGL application must associate a rendering context with a device context. The device context must be a display device context or a memory device context that is compatible with the display device context. To set up a rendering context, applications must first use the SetPixelFormat Win32 function to set up a pixel format for the device; next, they must call wglCreateContext with the device context handle as its parameter. If successful, wglCreateContext returns a rendering context handle of type HGLRC.

NOTE
Windows does not support drawing into a printer device context using the OpenGL Library. If you wish to print an image created with OpenGL, one possible workaround is to draw into a memory device context that is compatible with the display device and then transfer the resulting bitmap to the printer device.

OpenGL under Windows recognizes two types of pixel data modes: RGBA formats and color index-based modes. When the RGBA mode is selected, pixel colors are specified in the form of RGB color values. When color index mode is selected, pixel colors are selected from the system palette using an index value. These two modes become relevant on palette-based 256-color devices (many VGA-compatible display cards). When your application uses the RGBA mode on such a device, it must manage its own palette and respond to Windows palette notification messages. There are specific requirements that must be met by a window that is to be used for OpenGL operations. Specifically, such windows cannot be created using a window class that has the CS_PARENTDC style set. The window itself must have the WS_CLIPCHILDREN and the WS_CLIPSIBLINGS styles in order to be compatible with OpenGL. Note that to increase your applications performance, you may wish to use a window class that has a null background brush; the window background will be erased through the OpenGL Library anyway. Before a rendering context can be used, it must be set up as the current context using the wglMakeCurrent function. This function takes two parameters, one of which is a devicecontext handle. Interestingly, this handle does not need to be identical to the handle used in wglCreateContextbut it must refer to the same device. Thus it is possible, for example, to set up an OpenGL rendering context using a device-context handle returned by GetDC, but use wglMakeCurrent with a device-context handle returned by BeginPaint. Once a rendering context is ready to accept commands, you may wish to send additional initialization commands; for example, you may wish to erase the frame buffer before drawing, set up coordinate transformations, configure light sources, or enable and disable other options.

The OpenGL Graphics Library CHAPTER 43

771

One initialization step that cannot be omitted is the call to the glViewport function. Through this function, you can set up or modify the size of the rendering viewport. Typically, you should call this function once when the rendering context is initialized, and subsequently every time your application receives a WM_SIZE message indicating that its window size has changed.

Drawing with OpenGL


Most OpenGL drawing consists of a series of vertex operations enclosed between a pair of glBegin and glEnd calls. The glBegin call identifies the type of primitive that subsequent vertex operations define; glEnd marks the end of constructing the primitive. For example, the following series of calls constructs a pentagon:
glBegin(GL_POLYGON); glVertex2d(0.0, 1.0); glVertex2d(-0.951057, 0.309017); glVertex2d(-0.587785, -0.809017); glVertex2d(0.587785, -0.809017); glVertex2d(0.951057, 0.309017); glEnd();

The glBegin function can be used to define a variety of primitives. Table 43.1 lists the allowable parameters for this function.

43
THE OPENGL GRAPHICS LIBRARY

Table 43.1. Primitives constructed through glBegin. glBegin parameter Description


GL_POINTS GL_LINES GL_LINE_STRIP GL_LINE_LOOP GL_TRIANGLES GL_TRIANGLE_STRIP GL_TRIANGLE_FAN GL_QUADS GL_QUAD_STRIP GL_POLYGON

A series of points A series of lines A connected group of line segments A connected, closed group of line segments A set of triangles A set of connected triangles A set of connected triangles A set of quadrilaterals A set of connected quadrilaterals A polygon

In the case when glBegin defines a set of connected primitives, specific rules govern how vertices of a primitive are reused as vertices of the subsequent primitive. For example, if GL_LINE_STRIP is specified, the vertex representing the end point of a line segment also becomes the starting point of the next line segment.

772

Graphics and Multimedia PART VII

Additional Libraries
In addition to basic OpenGL functions, Microsofts OpenGL implementation provides two additional OpenGL libraries. The OpenGL Utility Library (GLU) contains a series of functions that deal with texture support; coordinate transformation; rendering of spheres, disks, and cylinders; B-spline curves and surfaces; and error handling. Additionally, the GLU Library provides polygon tessellation functions; these functions can be used to break down complex or concave polygons into simple convex polygons (the only kind that OpenGL can handle). The OpenGL Programming Guide Auxiliary Library (GLAUX), in addition to providing functions for handling several three-dimensional objects, also provides functions to manage and run an OpenGL application. These functions are most useful for quick porting OpenGL applications from other environments. In particular, these functions provide basic window management, implement a simple message loop, and provide a window procedure for basic message handling. However, these library functions are not intended for use in production applications.

Writing OpenGL Windows Applications in C


Now for a look at a very simple OpenGL application. This application, shown in Listing 43.1, displays a cube. The cube is slightly rotated to show a three-dimensional appearance and is lit from the side. In its simplicity, this application is the OpenGL version of a Windows Hello, World application.

Listing 43.1. A simple OpenGL application.


#include <windows.h> #include <gl/glu.h> HGLRC hglrc; void DrawHello(HWND hwnd) { HDC hDC; PAINTSTRUCT paintStruct; RECT clientRect; GLfloat lightPos[4] = {-1.0F, 2.0F, 0.6F, 0.0F}; hDC = BeginPaint(hwnd, &paintStruct); if (hDC != NULL) { GetClientRect(hwnd, &clientRect); wglMakeCurrent(hDC, hglrc); glViewport(0, 0, clientRect.right, clientRect.bottom); glLoadIdentity(); glClear(GL_COLOR_BUFFER_BIT); glColor4d(1.0, 1.0, 1.0, 1.0);

The OpenGL Graphics Library CHAPTER 43


glRotated(30.0, 0.0, 1.0, 0.0); glRotated(15.0, 1.0, 0.0, 0.0); glEnable(GL_LIGHTING); glEnable(GL_LIGHT0); glLightfv(GL_LIGHT0, GL_POSITION, lightPos); glBegin(GL_QUADS); glNormal3d(0.0, -1.0, 0.0); glVertex3d(0.5, -0.5, 0.5); glVertex3d(-0.5, -0.5, 0.5); glVertex3d(-0.5, -0.5, -0.5); glVertex3d(0.5, -0.5, -0.5); glNormal3d(0.0, 0.0, -1.0); glVertex3d(-0.5, -0.5, -0.5); glVertex3d(-0.5, 0.5, -0.5); glVertex3d(0.5, 0.5, -0.5); glVertex3d(0.5, -0.5, -0.5); glNormal3d(1.0, glVertex3d(0.5, glVertex3d(0.5, glVertex3d(0.5, glVertex3d(0.5, 0.0, 0.0); -0.5, -0.5); 0.5, -0.5); 0.5, 0.5); -0.5, 0.5);

773

43
THE OPENGL GRAPHICS LIBRARY

glNormal3d(0.0, 0.0, 1.0); glVertex3d(-0.5, -0.5, 0.5); glVertex3d(-0.5, 0.5, 0.5); glVertex3d(0.5, 0.5, 0.5); glVertex3d(0.5, -0.5, 0.5); glNormal3d(-1.0, glVertex3d(-0.5, glVertex3d(-0.5, glVertex3d(-0.5, glVertex3d(-0.5, 0.0, 0.0); -0.5, 0.5); 0.5, 0.5); 0.5, -0.5); -0.5, -0.5);

glNormal3d(0.0, 1.0, 0.0); glVertex3d(-0.5, 0.5, 0.5); glVertex3d(0.5, 0.5, 0.5); glVertex3d(0.5, 0.5, -0.5); glVertex3d(-0.5, 0.5, -0.5); glEnd(); glFlush(); wglMakeCurrent(NULL, NULL); EndPaint(hwnd, &paintStruct); } } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_PAINT:

continues

774

Graphics and Multimedia PART VII

Listing 43.1. continued


DrawHello(hwnd); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; HDC hDC; PIXELFORMATDESCRIPTOR pfd; int iPixelFormat; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN | WS_CLIPSIBLINGS, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); hDC = GetDC(hwnd); memset(&pfd, 0, sizeof(pfd)); pfd.nSize = sizeof(pfd); pfd.nVersion = 1; pfd.dwFlags = PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL; pfd.iPixelType = PFD_TYPE_RGBA; pfd.iLayerType = PFD_MAIN_PLANE; pfd.cDepthBits = 16; iPixelFormat = ChoosePixelFormat(hDC, &pfd); SetPixelFormat(hDC, iPixelFormat, &pfd); hglrc = wglCreateContext(hDC); ReleaseDC(hwnd, hDC); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); wglMakeCurrent(NULL, NULL); wglDeleteContext(hglrc); return msg.wParam; }

The OpenGL Graphics Library CHAPTER 43

775

The following sections explain this applications method of operation.

OpenGL Initialization
The first series of OpenGL calls in this application begins in WinMain, immediately after the applications window has been created. After obtaining a device-context handle for the client area of this window, the device contexts pixel format is set to a pixel format obtained through ChoosePixelFormat. The ChoosePixelFormat function can be used to identify pixel formats for a specific device that best matches a set of required characteristics. Note that although we are using the RGBA data mode, this application does not handle palette notification messages. As a consequence, it may not display the cube properly in 256-color palettized display modes. This is done in order to keep the application as simple as possible; in a production application, you would certainly not want to omit creating and managing a palette that is appropriate for your application. After the pixel format has been specified, a rendering context is created by a call to wglCreateContext. The rendering context handle is saved in a global variable that will be accessed from within other functions. When all initializations have been completed, the application enters its message loop. After the message loop terminates, cleanup is performed by calling wglMakeCurrent and wglDeleteContext before the application terminates.

43
THE OPENGL GRAPHICS LIBRARY

The Window Procedure


The applications simple window procedure processes only two messages: WM_PAINT and WM_DESTROY. When a WM_PAINT message is received, the window procedure calls the DrawHello function; it is in this function where OpenGL drawing operations take place. The first step in DrawHello is to select the rendering context as the current context and set the viewport size by calling glViewport. The viewport size was obtained by a call to the Win32 GetClientRect function. Next, the frame buffer is erased, and an identity transformation matrix is loaded. The transformation matrix is changed by two subsequent rotations, specified by calls to
glRotated. The first call rotates the view around the vertical axis. The second call tips the view

forward by rotating it around the horizontal axis. As a result, we will see the cube from a viewpoint somewhat above and to the left of the cube. The rotations are followed by calls that enable lighting mode and specify a light source. The code specifies a single light source that illuminates the cube from the left and above. With all this initialization work complete, actual drawing can begin. A series of six quadrilaterals is drawn, representing the six sides of the cube. For each of the quadrilaterals, the normal vector is defined by a separate call to glNormal3d. When the construction of the six primitives

776

Graphics and Multimedia PART VII

is complete, a call to glFlush is used to ensure that all OpenGL operations are complete, and then the device context is released and the function returns.

Compiling and Running the Application


This application can be compiled simply from the command line. I called the source file cube.c; to compile this file, type the following:
cl cube.c user32.lib gdi32.lib opengl32.lib

Note that applications that use the GLU Library or the GLAUX Library must also specify glaux.lib or glu32.lib on the command line. And because OpenGL is computation-intensive, it might be a useful idea to compile with the appropriate optimization flags set. The application should display a window with a three-dimensional image of a cube rendered in it, similar to that shown in Figure 43.2.

FIGURE 43.2.
Running the cube.exe Windows application.

OpenGL in MFC Applications


The OpenGL Library can easily be utilized from MFC applications as well. To enable the OpenGL libraries, add the appropriate library names to your project settings (Figure 43.3). When initializing the OpenGL Library in an MFC application, it is important to remember which window you wish to use for a rendering context. For example, if it is a view window that will serve as the rendering context, it is this window that should be used when the OpenGL rendering context is created.

The OpenGL Graphics Library CHAPTER 43

777

FIGURE 43.3.
Adding the OpenGL libraries to MFC project settings.

OpenGL Initialization
The MFC OpenGL application presented in this section is based on an AppWizard-generated single document interface application skeleton. In this application, we draw a cube identical to the cube drawn in the C application discussed earlier. The cube is drawn into the applications view window. Accordingly, the first task after creating the applications skeleton is to modify the view classs PreCreateWindow member function, to ensure that the view window is created with the appropriate flags. The modified version of this function is shown in Listing 43.2.

43
THE OPENGL GRAPHICS LIBRARY

Listing 43.2. Modified version of CCubeView::PreCreateWindow.


BOOL CCUBEView::PreCreateWindow(CREATESTRUCT& cs) { // TODO: Modify the Window class or styles here by modifying // the CREATESTRUCT cs cs.style |= WS_CLIPSIBLINGS | WS_CLIPCHILDREN; return CView::PreCreateWindow(cs); }

As you can see, the change to this function is simple; it consists only of adding the WS_CLIPSIBLINGS and WS_CLIPCHILDREN flags to the window style to ensure proper operation of the OpenGL libraries. Much more extensive initialization work is performed in the view classs OnCreate member function. This member function must be added using ClassWizard or the WizardBar, as a

778

Graphics and Multimedia PART VII

handler function for WM_CREATE messages. The implementation of this function, shown in Listing 43.3, creates a rendering context after setting a pixel format for the view windows device context.

Listing 43.3. Implementation of CCubeView::OnCreate.


int CCUBEView::OnCreate(LPCREATESTRUCT lpCreateStruct) { if (CView::OnCreate(lpCreateStruct) == -1) return -1; // TODO: Add your specialized creation code here PIXELFORMATDESCRIPTOR pfd; int iPixelFormat; CDC *pDC; pDC = GetDC(); memset(&pfd, 0, sizeof(pfd)); pfd.nSize = sizeof(pfd); pfd.nVersion = 1; pfd.dwFlags = PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL; pfd.iPixelType = PFD_TYPE_RGBA; pfd.iLayerType = PFD_MAIN_PLANE; pfd.cDepthBits = 16; iPixelFormat = ChoosePixelFormat(pDC->m_hDC, &pfd); SetPixelFormat(pDC->m_hDC, iPixelFormat, &pfd); m_hglrc = wglCreateContext(pDC->m_hDC); ReleaseDC(pDC); return 0; }

The rendering context handle is stored in the member variable m_hglrc. This member variable should be added to the declaration of the view class in the Attributes section, as follows:
class CCUBEView : public CView { ... // Attributes public: CCUBEDoc* GetDocument(); HGLRC m_hglrc; ...

Drawing the Cube


The actual drawing of the cube is performed in the OnDraw member function of the view class. This member function, shown in Listing 43.4, is very similar to the DrawHello function of the C application presented earlier in this chapter. After making the rendering context current, the

The OpenGL Graphics Library CHAPTER 43

779

function performs a series of initializations, including setting the size of the viewport, applying coordinate transformations, and setting up lighting. Afterwards, four quadrilaterals that together comprise the cube are drawn.

Listing 43.4. Implementation of CCubeView::OnDraw.


void CCUBEView::OnDraw(CDC* pDC) { CCUBEDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here CRect clientRect; GLfloat lightPos[4] = {-1.0F, 2.0F, 0.6F, 0.0F}; GetClientRect(&clientRect); wglMakeCurrent(pDC->m_hDC, m_hglrc); glViewport(0, 0, clientRect.right, clientRect.bottom); glLoadIdentity(); glClear(GL_COLOR_BUFFER_BIT); glColor4d(1.0, 1.0, 1.0, 1.0); glRotated(30.0, 0.0, 1.0, 0.0); glRotated(15.0, 1.0, 0.0, 0.0); glEnable(GL_LIGHTING); glEnable(GL_LIGHT0); glLightfv(GL_LIGHT0, GL_POSITION, lightPos); glBegin(GL_QUADS); glNormal3d(0.0, -1.0, 0.0); glVertex3d(0.5, -0.5, 0.5); glVertex3d(-0.5, -0.5, 0.5); glVertex3d(-0.5, -0.5, -0.5); glVertex3d(0.5, -0.5, -0.5); glNormal3d(0.0, 0.0, -1.0); glVertex3d(-0.5, -0.5, -0.5); glVertex3d(-0.5, 0.5, -0.5); glVertex3d(0.5, 0.5, -0.5); glVertex3d(0.5, -0.5, -0.5); glNormal3d(1.0, glVertex3d(0.5, glVertex3d(0.5, glVertex3d(0.5, glVertex3d(0.5, 0.0, 0.0); -0.5, -0.5); 0.5, -0.5); 0.5, 0.5); -0.5, 0.5);

43
THE OPENGL GRAPHICS LIBRARY

glNormal3d(0.0, 0.0, 1.0); glVertex3d(-0.5, -0.5, 0.5); glVertex3d(-0.5, 0.5, 0.5); glVertex3d(0.5, 0.5, 0.5); glVertex3d(0.5, -0.5, 0.5);

continues

780

Graphics and Multimedia PART VII

Listing 43.4. continued


glNormal3d(-1.0, glVertex3d(-0.5, glVertex3d(-0.5, glVertex3d(-0.5, glVertex3d(-0.5, 0.0, 0.0); -0.5, 0.5); 0.5, 0.5); 0.5, -0.5); -0.5, -0.5);

glNormal3d(0.0, 1.0, 0.0); glVertex3d(-0.5, 0.5, 0.5); glVertex3d(0.5, 0.5, 0.5); glVertex3d(0.5, 0.5, -0.5); glVertex3d(-0.5, 0.5, -0.5); glEnd(); glFlush(); wglMakeCurrent(NULL, NULL); }

Note that this implementation does not take into account the fact that the MFC framework also calls the view classs OnDraw function when drawing into a printer-device context. In its present state, attempts to use this application for printing will fail.

Running the Application


Before you can compile the application, you must include two OpenGL header files in your view class implementation file, CUBEView.cpp . Add the following line right after the #include stdafx.h line:
#include <GL/glu.h>

Also dont forget to add the OpenGL library file to the projects settings if you have not done so before. (See Figure 43.3.) To run the application, compile and execute it from the Build menu. The applications window should appear similar to that shown in Figure 43.4. As with our previous OpenGL example, this program also does not provide proper handling of palette messages, which means that its display may appear incomplete or corrupted in 256-color palettized display modes.

The OpenGL Graphics Library CHAPTER 43

781

FIGURE 43.4.
Running the cube.exe MFC application.

43
THE OPENGL GRAPHICS LIBRARY

Summary
OpenGL is a library of high-quality three-dimensional graphics and rendering functions. The librarys device- and platform-independence make it a library of choice for developing portable graphics applications. OpenGL drawings are constructed from primitives; primitives are simple items such as lines or polygons, which in turn are composed of vertices. The OpenGL Library assembles primitives from vertices while taking into account a variety of settings, such as color, lighting, and texture. Primitives are then processed in accordance with transformations, clipping settings, and other parameters; pixel data is deposited into a frame buffer at the end of the rasterization process. The Windows implementation of the OpenGL Library consists of the core library, utility functions (GLU), and auxiliary functions (GLAUX). The auxiliary library can be used to easily create simple standalone OpenGL applications, as it implements a message loop and a window procedure internally. However, due to the simplicity of implementation, this library should not be used in production applications. Windows also provides a set of extension functions (WGL) that facilitate the use of OpenGL functions in the context of the Windows GDI. Furthermore, a set of new functions has been added to the Win32 API to support pixel formats and OpenGL double buffering.

782

Graphics and Multimedia PART VII

The main steps of creating a Windows OpenGL application are as follows: 1. Ensure that your window class is not created with the CS_PARENTDC style. Ensure that your window is created with the styles WM_CLIPCHILDREN and WM_CLIPSIBLINGS set. 2. Create an OpenGL rendering context; a good spot for doing so is in the WM_CREATE handler function for the window that you intend to use with OpenGL. 3. Add appropriate calls in your handler for WM_PAINT messages to draw the OpenGL image. 4. Optionally, add a handler for WM_SIZE messages to reflect changes in the viewport size. Use glViewport to set the viewport size. 5. If you plan to run your application on 256-color devices, add handling for custom palettes.

High-Performance Graphics and Sound: DirectX CHAPTER 44

783

44

High-Performance Graphics and Sound: DirectX


IN THIS CHAPTER
s The DirectX APIs 785 792 s A Working Example

44
GRAPHICS AND SOUND: DIRECTX

784

Graphics and Multimedia PART VII

Early in my career I had the opportunity to spend almost a year as a member of a programmer team working on games for the Commodore 64 home computer. Although we barely earned enough to eat, I have the fondest memories of this period. I dont think I ever had as much fun with computers as I did then, designing green dragons, writing a real-time executive, coding Bartoks Allegro Barbaro on the C64s primitive sound synthesizer, and learning about highperformance drawing algorithms. Game programming is probably the most technically challenging form of programming for desktop systems, and the joy of seeing your creation come to life is a reward by itself. Games under Windows are as old as Windows itself; I vaguely remember playing Reversi with a machine that had Windows 1.0 installed. However, creating true arcade-style games with real-time animation and sound under Windows proved to be an elusive target until recently. That remained the case despite the fact that the benefits of using the Windows platform for game development are enormous. Under Windows, you no longer need to contend with the multitude of graphics accelerators, sound cards, TSR drivers, and the rest of the paraphernalia that make developing even the simplest graphics application under MS-DOS a nightmare. Windows device-independence takes care of it all. One reason game development under Windows has lagged so far behind the DOS platform is speed. Although the GDI is great on device independence, it is not very efficient; updating a window with a relatively large surface area takes a considerable amount of time. In the past, Microsoft has attempted to provide high-efficiency graphics libraries and other tools for game developers. However, nothing compares to the solution that was introduced when I was working on the first edition of this book. Initially known as the Windows 95 Game SDK, the DirectX SDK is a family of libraries for graphics, sound, joystick control, animation, and communications (for multiple-player games). These libraries are based on Microsofts Component Object Model, the foundation of Microsofts ActiveX technology. The latest version of the DirectX SDK and redistributable components are available for downloading at the Microsoft Web site. They are also distributed to Microsoft Developer Network professional-level subscribers. DirectX support is no longer limited to Windows 95 or later. Beginning with version 4, Windows NT also provides support for many key DirectX features. A single chapter can no longer provide in-depth coverage of an API that has grown to be as big as DirectX. However, this chapter provides a comprehensive review of the APIs main features, and also provides a simple working example that incorporates animated graphics and sound.

High-Performance Graphics and Sound: DirectX CHAPTER 44

785

The DirectX APIs


The DirectX SDK consists of a series of APIs. The names of these APIs begin with the word Direct, and thus they are collectively referred to as the DirectX APIs. The first of these APIs, DirectDraw, provides a low-level, high-speed interface to your computers graphics hardware. Through DirectDraw, it is possible to write games that use real-time animation in a window; the API also provides the capability to write full-screen graphics applications. Another member of the DirectX family, Direct3D, provides support for 3D graphics and animation. The DirectSound API provides low-level access to your computers sound hardware. The most important features of this API include the capability to play sounds continuously in a loop and the capability to mix sounds. The DirectPlay API provides a game-oriented communication interface to facilitate multiplayer gaming across a modem, network connection, or through an online service. The DirectInput API provides improved, more reliable access to human interface devices, including new force-feedback input/output devices. The DirectShow API provides high-performance media playback and capture capabilities. The DirectAnimation API provides animation support and strong integration with DHTML. Finally, the DirectSetup API provides a single-function setup capability for DirectX runtime components.

DirectX and the Component Object Model


DirectX services are provided using the Component Object Model (COM). A COM object is exposed to the outside world through one or more interfaces; each interface is essentially a table of function pointers representing the objects methods. Thus, the interface resembles the implementation of a pure virtual class in C++; the actual implementations of these functions are provided by the object itself. The similarity to C++ classes ends, however, when it comes to inheritance. Although it is possible to create a new interface using an old interface, the new interface will not inherit the implementation of the old interfaces methods. For example, although all COM interfaces are derived from the basic interface, IUnknown, they must individually implement mandatory IUnknown methods.

44
GRAPHICS AND SOUND: DIRECTX

786

Graphics and Multimedia PART VII

COM interfaces are compatible with the implementation of C++ classes. The table of methods in a COM interface easily translates into virtual member functions of the C++ class representing the interface. When applications use COM interfaces from the C language, they must refer to the interfaces virtual function table (vtable) explicitly. The DirectX API header files provide convenience macros for all DirectX methods. As I mentioned, all COM interfaces are derived from IUnknown. The IUnknown interface has three methods. The first of these methods, QueryInterface, can be used to determine whether a particular interface is present. AddRef and Release implement reference counting for objects as they are created, referred to from other objects, or released. These methods must be implemented by all IUnknown-derived COM interfaces. By convention, the names of COM interfaces begin with the uppercase letter I. Also by convention, interface methods are referenced using a C++ syntax even though they can also be accessed from C using an explicit vtable reference.

DirectDraw
DirectDraw is a client to the services provided by the DirectDraw HAL (hardware abstraction layer) and HEL (hardware emulation layer). DirectDraw is implemented in the dynamic link library ddraw.dll. The DirectDraw HAL implements device-dependent functions; it only implements functions that are supported by the device. For unsupported functions, the HAL simply reports that they are unavailable. The HAL performs no parameter validation; all parameter validation is performed by DirectDraw. The DirectDraw HEL appears to DirectDraw just like the DirectDraw HAL and provides software emulation for capabilities not implemented in hardware. The DirectDraw HAL and DirectDraw HEL are implemented in the form of driver libraries like ati.vxd and atim32.drv. The DirectDraw API provides access to low-level graphics services through a series of COM (common object model) interfaces. These interfaces and their relationships are schematically depicted in Figure 44.1.

FIGURE 44.1.
DirectDraw COM interfaces.

IDirectDraw2 CreateSurface

IDirectDrawSurface3 SetPalette SetClipper

CreatePalette

IDirectDrawPalette

CreateClipper

IDirectDrawClipper

High-Performance Graphics and Sound: DirectX CHAPTER 44

787

The first of these interfaces, IDirectDraw2, represents the graphics hardware (accelerator card) in your computer. Applications that use DirectDraw services begin their operation by creating a DirectDraw object using the DirectDrawCreate function. DirectDrawCreate returns a pointer to an IDirectDraw interface. Before doing any other work, applications must also call IDirectDraw::SetCooperativeLevel. Applications that request exclusive access to the computers graphics hardware through IDirectDraw::SetCooperativeLevel can change the video mode and implement full-screen graphics and animation; other applications are restricted to the current video mode and should confine their graphics activity to windows they own. The IDirectDraw::CreateSurface method creates a DirectDrawSurface object and returns a pointer to an IDirectDrawSurface interface. DirectDrawSurface objects represent rectangular areas in memory (typically, video memory) where applications can draw. The primary drawing surface is the visible display surface; secondary drawing surfaces can exist in video memory or the computers main memory and are used for offscreen drawing.

NOTE
The IDirectDraw2 interface supersedes the IDirectDraw interface. However, for reasons of compatibility DirectDrawCreate returns a pointer to an IDirectDraw interface. If you must use the capabilities of the IDirectDraw2 interface, this interface can be created by calling IDirectDraw::QueryInterface. Similarly, calling IDirectDraw::CreateSurface returns an IDirectDrawSurface interface pointer; to create an IDirectDrawSurface3, call IDirectDrawSurface::QueryInterface.

Drawing onto a surface is possible through any of a variety of bit blit methods provided by the I D i r e c t D r a w S u r f a c e interface or by using ordinary GDI functions. The IDirectDrawSurface::GetDC method can be used to obtain a device-context handle to the surface. GDI will treat this handle as a handle to a memory-device contexteven when the surface is the primary drawing surface. supports smooth animation with back-buffer surfaces. The method can be used to switch the contents of the primary surface and the back-buffer surface. Applications can use the back-buffer surface to perform offscreen drawing, flip the two surfaces when drawing is finished, and start drawing the next frame in the back buffer. Of course, this is only one of several techniques that can be used to achieve smooth animation. If only small portions of the display surface are updated at any given time, another technique (such as the technique using multiple buffers implemented by the example shown later in this chapter) might be more beneficial.
IDirectDrawSurface IDirectDrawSurface::Flip

44
GRAPHICS AND SOUND: DIRECTX

Another interface, IDirectDrawPalette, provides access to palette services. This interface represents the hardware palette and bypasses Windows palettes. Because of this, use of IDirectDrawPalette requires exclusive access to the video hardware. IDirectDrawPalette can be used for many effects, including palette animation. Note that this interface might not be available on systems that do not support a hardware palette.

788

Graphics and Multimedia PART VII

An IDirectDrawPalette interface is created by a call to IDirectDraw::CreatePalette. The palette must be attached to a display surface through IDirectDrawSurface::SetPalette.
DirectDrawSurface

interface represents clip lists. A clip list can be attached to a object and used during bit blit operations. A window handle can also be attached to a clip list, in which case the clip list represents the clipping rectangles of the window. The
IDirectDrawClipper

An IDirectDrawClipper interface is created by calling IDirectDraw::CreateClipper and attached to a drawing surface by calling IDirectDrawSurface::SetClipper. The DirectDraw API can be used to represent not only the primary video hardware in your computer, but also secondary display devices that are not normally recognized by Windows. For example, consider a development system that has a secondary display card and monitor for testing purposes. An IDirectDraw interface representing this secondary device can be created by a call to DirectDrawCreate. The first parameter of this function represents the GUID (globally unique identifier) of a display driver; when it is set to NULL DirectDrawCreate creates an interface representing the active display driver. However, you can also specify an explicit GUID representing any secondary device that might be installed on your system.

Direct3D
Direct3D consists of a series of COM object types that, tightly integrated with DirectDraw, provide low-level support for three-dimensional graphics and real-time animation. Direct3D itself consists of two APIs. Retained Mode is a high-level API for managing threedimensional objects and creating three-dimensional scenes. Immediate Mode is a low-level API for communicating with accelerator hardware. Direct3Ds Immediate Mode recognizes a family of eight object types. These include interfaces, devices, textures, materials, light sources, viewports, matrices, and execution buffers. The latter contain information on vertices and operation codes that together define rendering. Direct3Ds Retained Mode supports objects that include animations and animation sets, devices, visual objects described in the form of meshes of polygon faces, frames, light sources, materials, shadows, textures, and more.

DirectSound
The DirectSound API provides access to your computers waveform sound hardware. The sound device is represented by the IDirectSound interface; individual buffers are represented by IDirectSoundBuffer.

High-Performance Graphics and Sound: DirectX CHAPTER 44

789

NOTE
DirectSound does not provide MIDI functionality. To utilize your sound hardwares MIDI capability, use the standard Win32 multimedia APIs for MIDI.

The most important DirectSound capability is wave mixing. It is accomplished by using a series of primary and secondary sound buffers. A primary buffer represents the hardware buffer of the sound device; that is, the buffer that is currently playing. Secondary buffers can represent different audio streams that are mixed together into the primary buffer for playback. This mechanism is depicted schematically in Figure 44.2.

FIGURE 44.2.
DirectSound wave mixing.

Secondary buffer 1 (looping)

Secondary buffer 2

44
Primary buffer

GRAPHICS AND SOUND: DIRECTX

An IDirectSound interface is created by calling DirectSoundCreate. Before the interface can be used, you must also call DirectSoundCreate::SetCooperativeLevel to specify the level of access you require to the sound card. For most applications, this should be DSSCL_NORMAL. This level of access ensures smooth cooperation between applications that compete for the same hardware resources. A sound buffer is allocated by calling IDirectSound::CreateSoundBuffer. An interface to the sound buffer is returned in the form of an IDirectSoundBuffer pointer. Applications do not normally need to allocate a primary sound buffer; this buffer is allocated implicitly when the contents of secondary buffers are played back.

790

Graphics and Multimedia PART VII

When a secondary buffer is allocated you must specify the size of the buffer. Afterward, you can use IDirectSoundBuffer::Lock to obtain a pointer to this buffer. You can then use standard C library functions to copy waveform information into this buffer. The content of a buffer is played back using the IDirectSoundBuffer::Play method. Calling this function while a buffer is playing will update playback flags but will not affect playback otherwise (for example, it will not cause playback to restart at the beginning of the buffer). To change the playback position, use IDirectSoundBuffer::SetCurrentPosition.
IDirectSoundBuffer::Play can also be used for continuous (looping) playback. This capability is ideal to provide background sounds, such as the engine sounds in an aircraft simulation game.

In addition to the basic features listed here, DirectSound now also provides the capability to generate three-dimensional stereo sound effects.

DirectPlay
Game-oriented communication services are provided through the DirectPlay API. The DirectPlay API consists of two components: the IDirectPlay interface and the DirectPlay server. Microsoft provides DirectPlay servers for modem and network connections; other servers (for example, servers for online services) are provided by third-party developers. To find out what DirectPlay servers are installed on a computer, use the DirectPlayEnumerate function. A DirectPlay object is created by the DirectPlayCreate function. You must pass the GUID of the selected DirectPlay server to this function. In turn, DirectPlayCreate returns a pointer to an IDirectPlay interface. You can enumerate existing DirectPlay sessions by calling IDirectPlay::EnumSessions. This method must be called after the DirectPlay object has been created. You can create a DirectPlay session or connect to an existing session using the IDirectPlay::Open method. This method actually establishes the communication link. DirectPlay invokes the necessary user interface for configuring the communication protocol; for example, if a modem connection is requested, DirectPlay will invoke a dialog requesting the telephone number and other dialing information. Games using DirectPlay must be identified by a globally unique identifier (GUID). You can generate a GUID using the guidgen. exe utility that is part of Visual C++. After you have connected to a session, you must create players. A player is created through You can obtain the list of players in the session by calling IDirectPlay::EnumPlayers. To actually exchange messages between players, you can use IDirectPlay::Send and IDirectPlay::Receive.
IDirectPlay::CreatePlayer.

High-Performance Graphics and Sound: DirectX CHAPTER 44

791

Other methods exist for managing sessions, players, groups of players, and messages. Players are destroyed using IDirectPlay::DestroyPlayer ; a session is terminated by calling IDirectPlay::Close.

DirectInput
The DirectInput API consists of joystick, mouse, and keyboard related services. Joystick support is provided through the joyGetPosEx function that can return position and button state information for joysticks with six degrees of freedom (axes) and 32 buttons; and through functions for querying device capabilities: joyGetDevCaps, joyGetNumDevs, and joyConfigChanged. Keyboard and mouse support is provided through the COM object types DirectInput and DirectInputDevice.

DirectShow
The DirectShow API provides a rich and complex set of high-performance media playback and capture capabilities. These capabilities include the ability to play back compressed content in a variety of formats (such as MPEG, QuickTime, AVI, and WAV) and to operate video capture devices that are compatible with Video for Windows or the new Windows Driver Model (WDM) used in Windows 98 and Windows NT.

DirectAnimation
The DirectAnimation API provides animation support and strong integration with DHTML. In place of further comment or a tedious enumeration of DirectAnimation classes, here is a simple JScript example:
<html> <body> <object ID=DAControl STYLE=position:absolute; left:30%; top:100;width:300;height:300;z-index: -1" CLASSID=CLSID:B6FFC24C-7E13-11D0-9B47-00C04FC2F51D> </object> <script LANGUAGE=JScript> <!-m = DAControl.PixelLibrary; fillImg = m.Until(m.SolidColorImage(m.Red), m.LeftButtonDown, m.SolidColorImage(m.Green)); DAControl.Image = fillImg; DAControl.Start() //--> </script> </body> </html>

44
GRAPHICS AND SOUND: DIRECTX

As this example reveals, the primary utility of DirectAnimation is in scripting code, although you can also use the DirectAnimation API from Visual C++. The strength of the API lies in objects and types that can change their values in response to events and the passage of time.

792

Graphics and Multimedia PART VII

DirectSetup
The DirectSetup API provides a single-function setup capability for DirectX redistributable components. When distributing applications that use the DirectX APIs, you must distribute with them the DirectX redistributable files. These files must be redistributed in unaltered form, in accordance with the DirectX license agreement. Included with the DirectX files are the DirectSetup DLLs that implement the DirectXSetup function.
DirectXSetup is called with parameters that specify the installation root path and flags that specify

which DirectX components to install. Although it is possible to install only selected components, Microsoft recommends that you do not do so; disk space savings would be minimal due to interdependencies among the DirectX components. The existence of DirectXSetup is more than a convenience. Because of the complexities of DirectX installation, developers should not attempt to perform a manual installation.

A Working Example
The application shown in Listing 44.1 combines the services of DirectDraw and DirectSound to implement a noisy bouncing ball.

Listing 44.1. Simple DirectX application.


#include <windows.h> #include <ddraw.h> #include <dsound.h> IDirectDraw *dd; IDirectDrawSurface *dds0, *dds1, *dds2, *dds3; IDirectDrawClipper *ddc; IDirectSound *ds; IDirectSoundBuffer *dsb1, *dsb2; int x = 20, y = 20; int vx = 5, vy = 3; void MoveBall(HWND hwnd, BOOL bMove) { BOOL bBounce = FALSE; RECT rectSrc, rectDest; int ox, oy, nx, ny; GetClientRect(hwnd, &rectDest); ClientToScreen(hwnd, (POINT *)&rectDest.left); ClientToScreen(hwnd, (POINT *)&rectDest.right); if (bMove) { ox = rectDest.left +

High-Performance Graphics and Sound: DirectX CHAPTER 44


MulDiv(rectDest.right - rectDest.left - 32, x, 500); oy = rectDest.top + MulDiv(rectDest.bottom - rectDest.top - 32, y, 500); x += vx; y += vy; if (x < 0) { x = 0; vx = -vx; bBounce = TRUE; } if (x >= 500) { x = 1000 - x; vx = -vx; bBounce = TRUE; } if (y < 0) { y = -y; vy = -vy; bBounce = TRUE; } if (y >= 500) { y = 1000 - y; vy = -vy; bBounce = TRUE; } if (bBounce) { dsb1->SetCurrentPosition(0); dsb1->Play(0, 0, 0); } } nx = rectDest.left + MulDiv(rectDest.right - rectDest.left - 32, x, 500); ny = rectDest.top + MulDiv(rectDest.bottom - rectDest.top - 32, y, 500); rectSrc.left = rectSrc.top = 0; rectSrc.right = rectSrc.bottom = 32; if (bMove) { rectDest.left = rectDest.top = 0; rectDest.right = rectDest.bottom = 32; dds2->Blt(&rectDest, dds3, &rectSrc, DDBLT_WAIT, NULL); if (abs(nx - ox) < 32 && abs(ny - oy) < 32) { if (nx < ox) { rectSrc.left = ox - nx; rectSrc.right = 32; rectDest.left = 0; rectDest.right = 32 - rectSrc.left; } else { rectDest.left = nx - ox; rectDest.right = 32; rectSrc.left = 0; rectSrc.right = 32 - rectDest.left; } if (ny < oy) { rectSrc.top = oy - ny; rectSrc.bottom = 32; rectDest.top = 0; rectDest.bottom = 32 - rectSrc.top; } else {

793

44
GRAPHICS AND SOUND: DIRECTX

continues

794

Graphics and Multimedia PART VII

Listing 44.1. continued


rectDest.top = ny - oy; rectDest.bottom = 32; rectSrc.top = 0; rectSrc.bottom = 32 - rectDest.top; } dds2->Blt(&rectDest, dds1, &rectSrc, DDBLT_WAIT, NULL); } rectSrc.left = rectSrc.top = 0; rectSrc.right = rectSrc.bottom = 32; rectDest.left = ox; rectDest.top = oy; rectDest.right = rectDest.left + 32; rectDest.bottom = rectDest.top + 32; dds0->Blt(&rectDest, dds2, &rectSrc, DDBLT_WAIT, NULL); } rectDest.left = nx; rectDest.top = ny; rectDest.right = rectDest.left + 32; rectDest.bottom = rectDest.top + 32; dds0->Blt(&rectDest, dds1, &rectSrc, DDBLT_WAIT, NULL); } LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { HDC hDC; PAINTSTRUCT paintStruct; switch(uMsg) { case WM_PAINT: hDC = BeginPaint(hwnd, &paintStruct); if (hDC != NULL) { MoveBall(hwnd, FALSE); EndPaint(hwnd, &paintStruct); } break; case WM_TIMER: MoveBall(hwnd, TRUE); break; case WM_KEYDOWN: switch (wParam) { case VK_LEFT: vx--; break; case VK_UP: vy--; break; case VK_RIGHT: vx++; break; case VK_DOWN: vy++; break; case VK_ESCAPE: PostMessage(hwnd, WM_CLOSE, 0, 0); } break; case WM_DESTROY: PostQuitMessage(0); break;

High-Performance Graphics and Sound: DirectX CHAPTER 44


default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow) { MSG msg; HWND hwnd; WNDCLASS wndClass; DDSURFACEDESC ddsd; DSBUFFERDESC dsbd; HDC hddDC; RECT rect; HRSRC hrsrc; HGLOBAL hRData; DWORD *pRData; LPBYTE pMem1, pMem2; DWORD dwSize1, dwSize2; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = BOUNCE; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(BOUNCE, BOUNCE, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); DirectDrawCreate(NULL, &dd, NULL); dd->SetCooperativeLevel(hwnd, DDSCL_NORMAL | DDSCL_NOWINDOWCHANGES); memset(&ddsd, 0, sizeof(DDSURFACEDESC)); ddsd.dwSize = sizeof(DDSURFACEDESC); ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE; ddsd.dwFlags = DDSD_CAPS; dd->CreateSurface(&ddsd, &dds0, NULL); dd->CreateClipper(0, &ddc, NULL); dds0->SetClipper(ddc); ddc->SetHWnd(0, hwnd); ddsd.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN; ddsd.dwHeight = 32; ddsd.dwWidth = 32; ddsd.dwFlags = DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH; dd->CreateSurface(&ddsd, &dds1, NULL); dd->CreateSurface(&ddsd, &dds2, NULL); dd->CreateSurface(&ddsd, &dds3, NULL);

795

44
GRAPHICS AND SOUND: DIRECTX

continues

796

Graphics and Multimedia PART VII

Listing 44.1. continued


dds1->GetDC(&hddDC); SaveDC(hddDC); rect.left = rect.top = 0; rect.right = rect.bottom = 32; FillRect(hddDC, &rect, (HBRUSH)(COLOR_WINDOW + 1)); SelectObject(hddDC, GetStockObject(BLACK_BRUSH)); SelectObject(hddDC, GetStockObject(BLACK_PEN)); Ellipse(hddDC, 0, 0, 32, 32); RestoreDC(hddDC, -1); dds1->ReleaseDC(hddDC); dds3->GetDC(&hddDC); FillRect(hddDC, &rect, (HBRUSH)(COLOR_WINDOW + 1)); dds3->ReleaseDC(hddDC); DirectSoundCreate(NULL, &ds, NULL); ds->SetCooperativeLevel(hwnd, DSSCL_NORMAL); memset(&dsbd, 0, sizeof(DSBUFFERDESC)); dsbd.dwSize = sizeof(DSBUFFERDESC); dsbd.dwFlags = DSBCAPS_STATIC | DSBCAPS_CTRLDEFAULT; hrsrc = FindResource(hInstance, BOUNCE.WAV, WAVE); hRData = LoadResource(hInstance, hrsrc); pRData = (DWORD *)LockResource(hRData); dsbd.dwBufferBytes = *(pRData + 10); dsbd.lpwfxFormat = (LPWAVEFORMATEX)(pRData + 5); ds->CreateSoundBuffer(&dsbd, &dsb1, NULL); dsb1->Lock(0, dsbd.dwBufferBytes, &pMem1, &dwSize1, &pMem2, &dwSize2, 0); memcpy(pMem1, (LPBYTE)(pRData + 11), dwSize1); if (dwSize2 != 0) memcpy(pMem2, (LPBYTE)(pRData + 11) + dwSize1, dwSize2); dsb1->Unlock(pMem1, dwSize1, pMem2, dwSize2); hrsrc = FindResource(hInstance, HUM.WAV, WAVE); hRData = LoadResource(hInstance, hrsrc); pRData = (DWORD *)LockResource(hRData); dsbd.dwBufferBytes = *(pRData + 10); dsbd.lpwfxFormat = (LPWAVEFORMATEX)(pRData + 5); ds->CreateSoundBuffer(&dsbd, &dsb2, NULL); dsb2->Lock(0, dsbd.dwBufferBytes, &pMem1, &dwSize1, &pMem2, &dwSize2, 0); memcpy(pMem1, (LPBYTE)(pRData + 11), dwSize1); if (dwSize2 != 0) memcpy(pMem2, (LPBYTE)(pRData + 11) + dwSize1, dwSize2); dsb2->Unlock(pMem1, dwSize1, pMem2, dwSize2); dsb2->Play(0, 0, DSBPLAY_LOOPING); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd);

High-Performance Graphics and Sound: DirectX CHAPTER 44


SetTimer(hwnd, 1, 100, NULL); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); KillTimer(hwnd, 1); return msg.wParam; }

797

The applications execution begins in WinMain, where first a perfectly ordinary window class is registered and the applications window is created. Next, DirectDrawCreate is called to create an IDirectDraw interface to the display hardware. This interface is used to create a primary display surface. Then a clip list is created using IDirectDraw::CreateClipper and attached to this primary surface. The window handle of the applications window is attached to this clip list, ensuring that subsequent bit blit operations into the primary surface will operate correctly when the window is partially covered. Next, a series of secondary display surfaces is created. These secondary surfaces are used in a special algorithm that ensures smooth repainting of the bouncing ball. Two of these surfaces are initialized, one with the image of the ball (created through the GDI Ellipse function), the other with the background color. When initialization of the drawing surfaces is complete, an interface to the computers sound hardware is created using DirectSoundCreate. Two secondary sound buffers are then created using IDirectSound::CreateBuffer. One is used to store the audio stream for the background hum; the other stores the stream for the bouncing noise. Continuous playing of the background sound is started by calling IDirectSoundBuffer::Play with the DSBPLAY_LOOPING parameter. Before you enter the applications main message loop, a call is made to SetTimer to set up a 100-millisecond timer. This timer is used to periodically update the bouncing balls position and redraw the display. When a WM_TIMER event is received, the applications window procedure calls the MoveBall function. This function calculates the balls new position based on the global velocity values vx and vy. The balls position is expressed in the form of integers running between 0 and 500; these are then translated into screen coordinates using the MulDiv function. Redrawing the ball is a complex process involving three secondary buffers. Why is this complexity necessary? Obviously, if you are to update the balls position on the screen, you must do two things: erase the ball at its old position and draw it at its new position. However, unless the ball moves very fast, chances are that its old and new positions overlap. Simply erasing the ball at its old position would cause unwanted flicker because the screen pixels that are covered by the ball at both its old and new position also turn briefly white. To avoid this, instead of simply erasing the ball at its old position, create a secondary buffer that represents the balls old position. Into this buffer draw portions of the ball that will be

44
GRAPHICS AND SOUND: DIRECTX

798

Graphics and Multimedia PART VII

visible after the ball is at its new position. Copy this buffer at the balls old position; also copy the entire ball at its new position. Figure 44.3 shows a schematic representation of this multistep process. When your application updates only small areas of its window, this procedure might yield better performance than updating a back-buffer and using the IDirectDrawBuffer::Flip function.

FIGURE 44.3.
Smooth screen update using multiple buffers.

Primary surface

Secondary surface 1 Primary surface

Secondary surface 2 Primary surface

The application also responds to keyboard events. You can use the arrow keys to accelerate or decelerate the ball in the horizontal or vertical direction. Pressing Escape terminates the application. The two sounds that the application uses are in the waveform files hum.wav and bounce.wav. These files are referenced in the applications tiny resource file, shown in Listing 44.2.

Listing 44.2. Simple DirectX application resource file.


BOUNCE.WAV HUM.WAV WAVE WAVE DISCARDABLE DISCARDABLE bounce.wav hum.wav

High-Performance Graphics and Sound: DirectX CHAPTER 44

799

Note that in order to keep the application simple, some explicit assumptions were made as to the contents and structure of these waveform files. If you want to utilize another file, you might be well advised to look at some of the DirectX samples for ideas of how to provide a generic parsing capability. The application can be compiled from the command line using the C/C++ compiler of Visual C++:
rc bounce.rc cl bounce.cpp bounce.res user32.lib gdi32.lib ddraw.lib dsound.lib

The completed application looks similar to Figure 44.4.

FIGURE 44.4.
The bouncing ball application.

Summary
Recognizing the need for a high-performance interface for real-time Windows multimedia applications, Microsoft developed the DirectX SDK. This SDK provides a family of APIs, collectively referred to as the DirectX APIs, for access to the computers video, sound, communication, and joystick hardware. The basis for many elements in the DirectX APIs is Microsofts Component Object Model. Several interfaces in the DirectX APIs are derived from the COM IUnknown interface. Video hardware is represented by DirectDraw objects, accessed through the IDirectDraw interface. Through a DirectDraw object, applications can create drawing surfaces, palettes, and clip lists. The interface to drawing surfaces, IDirectDrawSurface, provides high-performance bit blit capabilities. The IDirectDraw interface can be used to manipulate the display hardware in a variety of ways; this includes exclusive access to the display hardware for games that provide full-screen graphics and animation. A special use of IDirectDraw is to provide access to secondary video hardware not normally recognized by Windows itself.

44
GRAPHICS AND SOUND: DIRECTX

800

Graphics and Multimedia PART VII

Relying on DirectDraw services, the Direct3D API provides a set of advanced interfaces for three-dimensional graphics and animation. This API is itself subdivided into two distinct components. Direct3D Immediate mode provides low-level access to accelerator hardware; Direct3D Retained mode provides higher-level three-dimensional rendering concepts. Access to sound hardware is provided through the IDirectSound interface. Applications typically use the sound hardware by creating a series of secondary sound buffers; the contents of these buffers are mixed into the primary buffer and played back by DirectSound. This wave mixing capability as well as the capability to play back in looping mode are the key gamerelated capabilities of DirectSound. DirectPlay provides communication services for multiplayer applications. Such applications can communicate through networks, modems, or online services. DirectPlay provides a simple send-receive functionality for game programs that participate in game sessions. The DirectInput API provides services for joystick, keyboard, and mouse input. Joystick support is provided through functions such as joyGetPosEx that can return position and other information for multibutton, multiaxis control devices. Keyboard and mouse support is provided through COM objects. Setting up the DirectX API is a complex but necessary task: not all users have the latest version of Windows that installs these libraries by default. A single-function interface that performs DirectX setup from the standard redistributable directories is provided in the form of the DirectXSetup function.

IN THIS PART
s Implementing Context-Sensitive Help 803 s Creating Installation Programs s User Interface Extensions 847 827

VIII
PART

s Localization: Creating International Applications 865

Other Topics

Implementing Context-Sensitive Help CHAPTER 45

803

45

Implementing Context-Sensitive Help


IN THIS CHAPTER
s Help File Development 805 s The Microsoft Help Workshop 813 s AppWizard-Generated Help File Skeletons 822

45
IMPLEMENTING CONTEXTSENSITIVE HELP

804

Other Topics PART VIII

Users of Windows applications expect intuitive user interfaces that enable them to work productively even when they are using an application for the first time. Lengthy learning periods and large, complex user manuals are no longer acceptable features of a new application. A significant element of a well-thoughtout user interface is integrated help. Although many application features are obvious to the experienced user, others are not; furthermore, inexperienced users may not immediately grasp the significance of elements in a menu, window, or dialog. Windows provides many features that assist the development of professional-quality, contextsensitive help files. Among these is the WinHelp program, the WinHelp API, and Windows capability to assign help context identifiers to various user-interface elements.

NOTE
With the advent of HTML and the ubiquitous Web browsers, using HTML documents for context-sensitive help has become a viable alternative. This is exactly the format Microsoft uses for most of the documentation that comes with Visual C++ 6. However, there are many advantages to using the old WinHelp format. The viewer executable is installed on all 32-bit Windows systems by default. The viewer is also smaller, which results in significatly reduced loading times. Because of these reasons and the fact that the MFC AppWizard creates skeletal help support in the Windows Help format, this chapter focuses on creating traditional WinHelp help content. However, the basics of creating HTML-based help are also covered toward the end of the chapter.

For MFC applications, AppWizard can be instructed to create a skeleton help file. AppWizardgenerated help can also be added to applications that were initially created without contextsensitive help. Developing help files has never been easier. The Visual C++ development system now includes the Microsoft Help Workshop, an integrated tool that helps you maintain and manage help projects. In this section, we first examine the features of WinHelp, the Windows help application. Next, we review the rules for constructing help topic files. Lastly, we examine the Help Workshop and touch on the subject of advanced topics. When authoring a help file, you should always keep in mind your intended audience. Despite a frequently heard myth to the contrary, I do not believe that the contents of help files and printed documentation should be identical. The purpose of these items is fundamentally different. While printed documentation is most often used by users who are new to your application and wish to receive an introduction to its concepts and ideas, help files are more likely to

Implementing Context-Sensitive Help CHAPTER 45

805

be accessed by experienced users wishing to obtain specific reference information on a particular topic. The format of the presentation (static, printed text versus online text with hypertext links) also prescribes a different authoring approach.

Help File Development


Help files are hypertext files that are presented by a special Windows application, WinHelp.exe (or winhlp32.exe on NT sytems). Constructing a help file is a process of several steps. First, help topic files need to be authored using an editor that is capable of handling rich text format (RTF) files. Next, a help project (HPJ) file must be created; this file identifies the help topic files and other components that comprise a help file project. Next, any additional files (for example, bitmap images) must be created. Finally, the help project file must be compiled using the Microsoft Help Compiler. The final version of the help system may also include dynamic link libraries and a help contents file. Figure 45.1 provides an illustration of this process, identifying the various elements of a help project and the tools used to process them.

FIGURE 45.1.
Help file development.

Resources identifiers (resource.h)

RTF word processor

Bitmap editor

C/C++ source files

makehm.exe

Help topic (RTF) files

Bitmap files

C/C++ compiler

Help mapping (HM) file

Help project (HPJ) file

Help compiler

Dynamic link libraries (DLLs)

45
IMPLEMENTING CONTEXTSENSITIVE HELP

Help file

Help contents (CNT) file

806

Other Topics PART VIII

Help files can be invoked from an application in a variety of ways. Applications may invoke help in response to the users selection of Help menu items, pressing the F1 key, or entering help mode by pressing Shift+F1. In all of these cases, help is actually invoked by a call to the WinHelp function.

Help Topics and the Rich Text Format


I purchased my first copy of the Windows SDK back in 1990 or thereabouts. I was a bit disappointed to learn that I needed to make an additional purchase in order to be able to edit rich text format help topic files. Nevertheless, I swallowed the bullet and went out and bought Word for Windows 1.1. What a long way Word has come in seven years! Working with Word 1.1 and Windows 3.0 taught me to save my work after every paragraph or so; otherwise, it fell prey to those dreaded Unidentified Application Errors, or UAEs, that seemed to crash Windows every 15 minutes. As it turns out, although I never regretted this purchase, there was no need for me to buy Word for Windows to edit help topics. RTF files consist of printable ASCII characters; the file format is not exactly user-friendly, but it is human-readable. It is possible to edit RTF help topic files using any ASCII text editor. Regardless of which editor you use, the Windows Help Compiler treats RTF files the same way. It accepts formatted text as its input; however, it interprets certain text attributes in a special way. For example, text that is marked with a double underline is interpreted as text marking a jump to another topic. Listing 45.1 shows a very simple help topic file consisting of only two topics.

Listing 45.1. A simple help topic file.


{\rtf1\ansi {\fonttbl\f0\fswiss MS Sans Serif;} \deff0\fs20 #{\footnote #H_CONTENTS} ${\footnote $Help Contents} K{\footnote KContents} \keepn\brdth\fs28 Help Contents \par\pard\fs20\sb30 This is the Help Contents page. To see more help, click on {\uldb Overview}{\v H_OVERVIEW}. \par\pard \page #{\footnote #H_OVERVIEW} ${\footnote $Help Overview} K{\footnote KOverview} !{\footnote !PositionWindow(256,256,512,512,1,main)} \keepn\brdrb\fs28 Help Overview \par\pard\fs20\sb30

Implementing Context-Sensitive Help CHAPTER 45


This is the Help Overview page. To see more help, click on {\uldb Contents}{\v H_CONTENTS}. \par To see the a macro at work, click on {\ul About}{\v !About()}. }

807

At the beginning of this file, \rtf1\ansi identifies the RTF version number and the ANSI character set. This part is mandatory for all help topic files. The next line specifies a font table. The font table of this simple help file consists of only one font. This font is selected as the default font by the \deff0 statement in line three. The \footnote statements in the next three lines are of crucial importance in help topic files. The help system uses footnotes of different types to index help text and to identify topicspecific information. The type of information specified by the footnote depends on the footnote character. In the present example, the pound sign (#) specifies a context string; the context string is used to refer to the current topic (as in a hypertext link). The K footnote specifies a keyword; the $ footnote specifies the topic title that will be displayed by the help system as the help window caption. The topic text that follows is broken by a variety of formatting codes into a scrolling and nonscrolling part. The nonscrolling region appears on top and can be used, for example, as a topic subtitle (Figure 45.2).

FIGURE 45.2.
Help topic example.

45
IMPLEMENTING CONTEXTSENSITIVE HELP
The second part of the topic text contains a hypertext reference to the second topic. The help system recognizes two types of links. Pop-up links are marked by the \ul statement and represent links to topics that appear in a pop-up window. Links marked with the \uldb statement represent links to topics that are displayed in the help systems main window. The link itself is identified by the subsequent \v statement.

808

Other Topics PART VIII

These codes and most other codes recognized by the help system can easily be generated by an RTF editor. For example, the \ul command represents underlined text; \uldb represents double underline. The help topics presented in Listing 45.1 appear, when edited using Microsoft Word 7.0, as shown in Figure 45.3.

FIGURE 45.3.
Editing a help topic using Word.

The help system also recognizes commands that are not part of the standard RTF format. These commands are related to bitmaps and DLLs. For example, the bmc command can be used to embed a bitmap in text. These help-only commands use a special syntax. The curly braces that normally enclose RTF commands are to be preceded by the backslash character. Furthermore, the backslash character that normally precedes RTF commands should be omitted, as in this example:
\{bmc bitmap.bmp\}

This special syntax enables such commands to be included when using an RTF word processor. If you are using Word, for instance, simply include the command with the curly braces but without the backslashes:
{bmc bitmap.bmp}

When this text is saved as part of the RTF file, the curly braces are automatically preceded by backslashes to distinguish them from RTF commands.

Implementing Context-Sensitive Help CHAPTER 45

809

NOTE
Although the RTF format recognizes embedded pictures through the \pict command, in my experience you are much better off using commands such as bmc to specify bitmaps or metafile pictures in external files. I repeatedly experienced problems with RTF files that grew too large because of the large amounts of picture data embedded within them.

The RTF file shown in Listing 45.1 also contains some macro references. We discuss help macros later in this chapter.

The Help Project File


Every help project must have a help project file. It is this file that is specified when the help compiler is invoked. Help project files usually have the .hpj extension. A typical help project file, the one used to compile the help topics demonstrated earlier, is shown in Listing 45.2.

Listing 45.2. A simple help project file.


[OPTIONS] CONTENTS=H_CONTENTS TITLE=Test Help File COMPRESS=OFF [WINDOWS] main=,,,(255,255,192),(192,192,192) [FILES] HLP.RTF [CONFIG] CreateButton(about, About, About()) [MAP] H_CONTENTS H_OVERVIEW

10000 10100

Perhaps the most important element in this file is the [FILES] section, where the names of the help topic files are specified. Other sections, such as the [OPTIONS] and the [WINDOWS] sections, provide additional parameters specifying the help files appearance and behavior. The [WINDOWS] section can also be used to define secondary help windows. Such windows can be referred to in topic files by name in order to display specific topics in a separate window. A possible use of this capability is to define a window type where sample code can be displayed.

45
IMPLEMENTING CONTEXTSENSITIVE HELP

810

Other Topics PART VIII

The [CONFIG] section contains macros that are executed when the help file is first opened. (More about this a little later in this chapter.) The [MAP] section associates context strings with context identifiers. Context identifiers can be passed to WinHelp by the application invoking help. Mappings can be created manually, or, in the case of MFC applications, automatically by using the makehm.exe utility. This utility uses the MFC projects resource.h file to create a mapping for all resource elements in the project. Help project files can contain #include directives. This is, for example, how AppWizardgenerated help project files refer to context mappings that are stored in separate files.

The Help Contents File


New versions of WinHelp (versions 4.0 or later), can use a separate Table of Contents property page (see Figure 45.4). The table of contents is created from a separate file, one with a name identical to that of the help file but having the .cnt extension.

FIGURE 45.4.
Help table of contents.

The table of contents file used to generate the table of contents in Figure 45.4 is shown in Listing 45.3.

Listing 45.3. A simple help table of contents file.


:Base HLP.hlp 1 Help Topics 2 Contents=H_CONTENTS 2 Overview=H_OVERVIEW

Implementing Context-Sensitive Help CHAPTER 45

811

In addition to specifying the items to appear in the table of contents, this file can contain additional options. For example, it can be used to specify the name of DLLs that specify additional property pages to appear in the Help Topics property sheet next to Contents, Index, and Find.

Compiling Help
The preceding sections review the elements that comprise a help project. All these elements come together when you invoke the Windows Help Compiler to generate the applications help file. The name of the new help compiler is hcrtf.exe. This program can be invoked by itself, or as part of the Microsoft Help Workshop. To compile a help file from the command line, type a command similar to the following:
hcrtf -x hlp.hpj

NOTE
Although the hcrtf utility can be invoked from the command line (this is how it is invoked from the makehelp.bat file generated for MFC application skeletons by AppWizard), it is usually a much better idea to invoke it from within the Microsoft Help Workshop. For one thing, when this utility encounters any errors, it uses a window within the Microsoft Help Workshop to report those.

Macros and DLLs


The Windows help system offers a variety of macros that can be used to enhance the appearance and functionality of a help file. Macros can be utilized in three ways: They can be executed when the help file is first loaded, when a help topic is displayed, or when a hotspot is selected by the user. In addition, some macros (for example, CreateButton, AppendItem) take macros as arguments; these macros are executed when the item is activated. To execute a macro at the time the help file is loaded, add the macro to the [CONFIG] section of your help project file. For example, the following [CONFIG] section in a help project file creates a button that invokes the About dialog:
[CONFIG] CreateButton(about, About, About())

45
IMPLEMENTING CONTEXTSENSITIVE HELP

To invoke a help macro when a topic is selected, add a footnote to the topic with the exclamation mark (!) as the footnote character. For example, the following footnote, when added to a topic, causes the help window to be centered on the screen every time the topic is invoked:
!{\footnote !PositionWindow(256,256,512,512,1,main)}

812

Other Topics PART VIII

The third method of invoking a help macro is by adding a macro reference to a hotspot. The following example invokes the Help About dialog when a hotspot is selected by the user:
To see the a macro at work, click on {\ul About}{\v !About()}.

Wherever you need to specify a macro, you can also specify a macro string. A macro string consists of several macros, separated by semicolons (;). In addition to macros, help files can also call functions in dynamic link libraries. DLL functions must be registered before use; for registration, use the RegisterRoutine macro. Typically, you would invoke this macro from the [CONFIG] section of your help project file; this way, the DLL will be registered when the help file is loaded. For example, if you wish to add a hotspot to your project that plays back a video file, add the following to the help project file:
[CONFIG] RegisterRoutine(MSVFW32.DLL, MCIWndCreate, UUUS);

Afterwards, you can add a hotspot to a help topic as follows:


Click {\ul here}{\v !MCIWndCreate(0,0,0,myvideo.avi)} for video.

Needless to say, you can use functions from system DLLs as well as DLLs that you develop yourself. DLLs can also be invoked in the context of embedded window references, through the statements ewc, ewl, or ewr that are to be included in your help topic files.

Invoking Help from Applications


Now that we have seen most aspects of help file development, only one question remains: How is help invoked from a C/C++ application? Regardless of what specific user action causes it, help is always invoked through the WinHelp Windows function. When invoking WinHelp through this function, applications can specify a variety of commands, causing help to appear in a regular or pop-up window, causing it to display the help contents, a specific topic, and so on. The simplest way to invoke help is to call WinHelp when the user selects a help command from the applications menu. In this case, simply invoke help by calling WinHelp from the handler of the appropriate WM_COMMAND message. Handling the F1 key is somewhat more difficult. In order to handle this key regardless of which user-interface element is active, applications used to require a message hook function. In Windows 95 or later, or Windows NT 4.0 or later, handling F1 is somewhat easier; no hook function is needed because whenever the user presses the F1 key, the operating system sends a WM_HELP message to the application.

Implementing Context-Sensitive Help CHAPTER 45

813

Windows 95, Windows NT 4.0, and later versions assist the development of context-sensitive help in other ways. For example, they send WM_CONTEXTMENU messages whenever the user rightclicks a window. In response to this message, applications can invoke WinHelp with the HELP_CONTEXTMENU command; this causes the pop-up menu with the Whats This? menu item to be displayed. To handle context-help mode, applications must implement a flag indicating that the user has pressed the F1 key or invoked context help through some other meansselecting a toolbar button, for example. When this flag is set, applications must process messages such as command messages differently, invoking WinHelp instead of invoking a command function. If you want to provide context help on nonclient area components of your applications window, you may also need to intercept the appropriate nonclient area messages as well. If the application implementing context-help mode is an in-place OLE server or client, care must be taken to ensure that the context-help mode of the server and the client are synchronized. If either the client or the server does not support context-help mode, context-help mode cannot be made available during an in-place session. For MFC applications, the framework automates invoking the help system with appropriate parameters. Unless you wish to add special effects, you do not need to worry about calling WinHelp yourself.

The Microsoft Help Workshop


The Microsoft Help Workshop is a Windows-based utility for creating and managing help project files and help content files. Using this utility in conjunction with an RTF-capable word processor greatly simplifies help file development.

Editing a Help Project


Figure 45.5 shows the Microsoft Help Workshop with a help project file open. This is the same help project file shown in Listing 45.2. As you can see, the Microsoft Help Workshop adds a series of standard sections and comments to the manually generated file. On the right side of the help project window, a series of buttons corresponds to sections of the help project file. Project settings can be modified by either using one of these buttons or by double-clicking on the appropriate item itself. For example, clicking on the Options button or double-clicking on the [OPTIONS] section in the help project file invokes the Options property sheet, shown in Figure 45.6. The Files button can be used to identify the help topic (RTF) files that comprise the project (Figure 45.7). You can add or remove files from the file list; you can also specify files that are to be included using the #include directive. A file included here with this directive should contain a list of topic files.

45
IMPLEMENTING CONTEXTSENSITIVE HELP

814

Other Topics PART VIII

FIGURE 45.5.
The Microsoft Help Workshop.

FIGURE 45.6.
Help project options: the General page.

Window options can be set or modified by clicking on the Windows button. The property sheet that is displayed (Figure 45.8), can be used to modify window properties such as size, color, caption, standard buttons, and any macros that are to be invoked when the window is displayed. To modify the properties for the main help window, use the window type main; however, you can also define additional window types by using the Add button in the General property page.

Implementing Context-Sensitive Help CHAPTER 45

815

FIGURE 45.7.
Adding and removing topic files.

FIGURE 45.8.
Help window options: the General page.

45
IMPLEMENTING CONTEXTSENSITIVE HELP
The Bitmaps button can be used to specify the location of bitmap files that are to be included with your help project (Figure 45.9). This option can be used as an alternative to including full pathnames in your help topic filesan approach that would make moving your help project to a different directory difficult.

816

Other Topics PART VIII

FIGURE 45.9.
Bitmap directories.

Help topic mappings (associations of context strings with numeric context identifiers) are added or modified using the Map button (Figure 45.10). A notable feature of this dialog is the Overview button, which invokes the Microsoft Help Workshop help file.

FIGURE 45.10.
Help topic mappings.

Implementing Context-Sensitive Help CHAPTER 45

817

The Alias button can be used to set up topic aliases (Figure 45.11). Topic aliases are particularly useful if you wish to map several user-interface elements to the same topic (for example, a catchall topic for error messages).

FIGURE 45.11.
Editing a topic alias.

The Config button can be used to add or modify configuration macros (Figure 45.12). These macros, added to the [CONFIG] section of your help project file, are called when the help file is first loaded.

FIGURE 45.12.
Configuration Macros dialog.

45
IMPLEMENTING CONTEXTSENSITIVE HELP

818

Other Topics PART VIII

The Data Files button is used to add files to the help project files [BAGGAGE] section (see Figure 45.13). Files listed in this section are added to the help files internal file system. WinHelp exposes a set of baggage-handling functions that DLLs can use to access these internally stored files. The advantage of using internal files is improved access speed when the help file is loaded from slower media (for example, CD-ROM). A typical use of this capability is to store multimedia files in the help file that are later retrieved and played back by a DLL.

FIGURE 45.13.
Baggage files.

When you want to recompile a help file, use the Save and Compile button. Clicking on this button causes your modifications to the help project file to be saved and the help compiler, hcrtf.exe, to be started. Because the Microsoft Help Workshop displays any compiler messages during compilation (Figure 45.14), this is the preferred method for compiling help files. When invoking the help compiler, hcrtf.exe, from the command line, compiler warnings and errors may be lost.

Editing a Help Contents File


The second function of the Microsoft Help Workshop is to maintain help table of contents files (Figure 45.15). These files specify the appearance and contents of the table of contents property page that WinHelp displays when a table of contents file is available and the user selects the Contents button. The Edit, Remove, Add Above, Add Below, Move Right, and Move Left buttons can be used to add and remove items as well as to modify the hierarchy of the help table of contents.

Implementing Context-Sensitive Help CHAPTER 45

819

FIGURE 45.14.
Help file compilation.

FIGURE 45.15.
Editing a help contents file.

45
IMPLEMENTING CONTEXTSENSITIVE HELP
The Link Files and the Index Files buttons specify additional help files that are to be included in index searches. It is possible to add new property pages to the Help Index property sheet by using the Tabs button. A new property page is represented by a DLL that exports the following function:
HWND WINAPI OpenTabDialog(HWND, DWORD, DWORD);

820

Other Topics PART VIII

When called, this function should create a dialog. The first parameter to this function identifies the dialogs parent; the remaining parameters are reserved. The dialog should be based on a template that is visible, has no borders, and has the WS_CHILD and DS_CONTROL styles set. You can use the AppWizard to create such a DLL. Create a regular DLL that is statically linked with the MFC LibraryMFC AppWizard (DLL) Step 1. Create a new dialog and add a class for the dialog. Next, add an OpenTabDialog function in a separate file (for example, opentab.cpp) that looks similar to the one shown in Listing 45.4.

Listing 45.4. A possible OpenTabDialog implementation file.


#include stdafx.h #include resource.h #include HlpDlg.h HWND WINAPI OpenTabDialog(HWND hwnd, DWORD, DWORD) { CHlpDlg *pDlg; pDlg = new CHlpDlg; pDlg->Create(CHlpDlg::IDD, CWnd::FromHandle(hwnd)); return pDlg->m_hWnd; }

In order to export this function, you must have the following line in the EXPORTS section of your module definition file:
OpenTabDialog=?OpenTabDialog@@YGPAUHWND__@@PAU1@KK@Z

Without this or a similar entry, the functions decorated name will be exported, which is not recognized by the Windows help system. If your DLL is functioning properly, a new tab is added to the help topics property sheet (Figure 45.16). You can communicate with WinHelp from your DLL by sending messages to it.

Testing and Running Help


The Microsoft Help Workshop can also be used to test help files. To start WinHelp with a specific help file, use the Run WinHelp command from the File menu. If you select this command, a dialog is displayed (Figure 45.17) where a variety of options can be selected. The File menu also contains the special Help Author option. When this flag is set, help files are run with debugging enabled. While debugging is enabled, WinHelp displays extra error messages; it also displays the topic identifier in window captions instead of the topic title.

Implementing Context-Sensitive Help CHAPTER 45

821

FIGURE 45.16.
Adding a custom tab to the Help Topics property sheet.

FIGURE 45.17.
Starting WinHelp from the Help Workshop.

45
IMPLEMENTING CONTEXTSENSITIVE HELP
Another useful feature is the WinHelp Messages command in the View menu. This command displays a variety of debugging messages generated by WinHelp while executing a help file. Finally, several useful test commands are available under the Test menu. The Contents File command tests a help contents file for link integrity. The Close All Help command closes all instances of WinHelp. The Send a macro command executes a macro in the context of the specified help file. Lastly, the WinHelp API command can be used to invoke the WinHelp function with a variety of parameter settings.

822

Other Topics PART VIII

AppWizard-Generated Help File Skeletons


For MFC projects that are created with AppWizard with help support requested, the AppWizard can be instructed to create a skeleton set of help files. The files created include afxcore.rtf, afxprint.rtf, and a variety of bitmap files. The bitmap files represent toolbar buttons, window portions, and other graphics used in the help topic files. Many of the help topics in afxcore.rtf and afxprint.rtf are complete; others may require modifications. Your application will probably also require a variety of application-specific topics; these you would probably implement in a separate topic file. In addition to creating the help topic files and bitmaps, the AppWizard also creates a help project file and a help content file for your application. Furthermore, it creates a batch file, makehelp.bat, which can be utilized to recompile the help project. This batch file is automatically invoked from your applications make file. However, makehelp.bat will not report any errors; if you wish to recompile your help project after extensive changes, I recommend that you use the Microsoft Help Workshop.

HTML Help
The advantages of the Windows help system notwithstanding, many authors might want to take advantage of HTML when creating help content. Indeed, Microsoft also has chosen this route: Most help content in Windows 98 and Windows NT 5 is HTML-based. The creation of HTML content is a complex subject that is far beyond the scope of this chapter. It is also covered extensively elsewhere, in publications dedicated to Web authoring. Here, our focus is the HTML Help API in Windows and the help authoring tool provided by Microsoft, the HTML Help Workshop.

The HTML Help System


HTML help files can contain any HTML content. This content is displayed using the same engine that is part of Microsofts Internet Explorer. As a consequence, HTML help requires that Internet Explorer, preferably version 4 or later, be installed on the end users system.

NOTE
Because the HTML help system might not be available on all your end users computers, it might be necessary to package and redistribute its components along with your application.

Implementing Context-Sensitive Help CHAPTER 45

823

The HTML help system provides a navigational interface. This interface can display a table of contents, an index, and a full-text search facility. Help content can be stored either in individual files or in a compressed format. In compressed form, HTML help files can be as economical as traditional Windows help files. Compressed help files can be created using the HTML Help Workshop.

Table of Contents and Index


Both the table of contents and the index are created as HTML files. Both use the sitemap format proposed by Microsoft to the WWW Consortium; both allow the display of a tree hierarchy of titles. For example, the following contents file defines a table of contents with two headingsthe first with one, the second with two entries:
<HTML> <HEAD> <!-- Sitemap 1.0 --> </HEAD><BODY> <OBJECT type=text/site properties> <param name=ImageType value=Folder> </OBJECT> <UL> <LI> <OBJECT type=text/sitemap> <param name=Name value=CONTENTS> </OBJECT> <UL> <LI> <OBJECT type=text/sitemap> <param name=Name value=NH Series> <param name=Local value=html\content.htm> </OBJECT> </UL> <LI> <OBJECT type=text/sitemap> <param name=Name value=REFERENCE> </OBJECT> <UL> <LI> <OBJECT type=text/sitemap> <param name=Name value=API functions> <param name=Local value=html\api.htm> </OBJECT> <LI> <OBJECT type=text/sitemap> <param name=Name value=Data Types> <param name=Local value=html\types.htm> </OBJECT> </UL> </UL> </BODY></HTML>

45
IMPLEMENTING CONTEXTSENSITIVE HELP

Content and index files can be created by hand, or they can be generated using the HTML Help Workshop or third-party authoring tools.

824

Other Topics PART VIII

The HTML Help API


The HTML Help API is modeled after the WinHelp API, and contains a small set of functions that applications can use to invoke the HTML help system. The primary interface to HTML Help is through the HtmlHelp function. Like its older counterpart, WinHelp, this function takes four parameters which identify the calling window, specify the name of the help file, and supply additional parameters. Unlike WinHelp, HtmlHelp is not (yet?) part of the standard set of Windows API functions. It resides in the file hhctrl.ocx. The easiest way to use this function is by linking your application with hhctrl.lib or htmlhelp.lib. If you link with hhctrl.lib, the hhctrl.ocx library and all other components required to display HTML help content will be loaded when your application starts. If you use htmlhelp.lib, the loading of these components will be delayed until the first call to HtmlHelp.

The HTML Help Workshop


In the preceding paragraphs, Microsofts HTML Help Workshop has been mentioned several times. This utility provides a graphical user interface for organizing HTML help content. It can also be used to create compressed HTML help files and test your help system. A particularly useful feature of the HTML Help Workshop is its capability to convert traditional help source files into HTML help source files. If you are thinking of switching to HTML help while working on an existing project, this capability may prove to be invaluable.

Summary
Online context-sensitive help is a mandatory feature for most new Windows applications. Context-sensitive help is developed using a variety of tools and displayed using the WinHelp.exe utility. The files that make up a Windows help project include a help project file, help topic files, and other, optional file components. Help topic files are in the Microsoft rich text format. As RTF files consist of printable ASCII characters, they can be edited by most ASCII editors; alternatively, and for convenience, an RTF-compatible word processor, such as Word for Windows, can be used. The help system interprets specific RTF formatting codes. For example, underlined text represents a hotspot, a hypertext link to another topic in a help topic file. In addition to standard RTF formatting codes, the help system also uses a few help-only codes. The help project file specifies the files that make up the help project. In addition, this file also specifies a series of settings and attributes for the help file, its windows, and its behavior.

Implementing Context-Sensitive Help CHAPTER 45

825

Another help project component is a table of contents file. If such a file exists, the help system uses a special property page for displaying a help files table of contents. The help system offers a series of macros. Macros can be nested and chained into macro strings. Macros can be invoked when the help file is first loaded, when a particular help topic is displayed, or when a specific hotspot is selected by the user. In addition to macros, applications can also invoke functions from dynamic link libraries, assuming that those functions have been previously registered using the RegisterRoutine macro. Applications invoke help using the WinHelp Windows function. This function is invoked when the user selects a Help menu command, the F1 key, or when context help mode is selected through the Shift+F1 key. Of these, processing the F1 key requires the use of hook functions, or, alternatively, you can rely on the WM_HELP message that is sent to your application under Windows 95, Windows NT 4.0, or later versions. For MFC applications created with AppWizard, AppWizard automatically adds the functionality to invoke help as appropriate. The Microsoft Help Workshop, an application introduced in Visual C++ Version 4, provides a graphics-editing facility for help project and help content files. Use of this application in conjunction with an RTF-capable word processor greatly simplifies the task of help file development. For MFC application skeletons created with help support, AppWizard creates a series of helprelated files. These files contain many complete topics and several other topics that require completion in accordance with the specifics of your application. AppWizard also creates a file, makehelp.bat, that is invoked from your applications make file. However, as makehelp.bat does not report any errors, it may be useful to compile your help project from within the Microsoft Help Workshop after any significant changes to the help project files. It is now also possible to create help content using HTML. HTML help files can use any HTML content, and are displayed by the same engine that powers Internet Explorer. To display HTML help content, the HTML help system must be installed on the users computer. Help is displayed in a split window, along with a hierarchical table of contents, a hierarchical index, and a search facility. The HTML help system offers an API similar to the traditional WinHelp API. A single API function provides access to most help system functionality. To help author HTML help content, Microsoft provides the HTML Help Workshop tool. In addition to the ability of creating compressed HTML content, this tool can also be used to convert existing help projects into the new format.

45
IMPLEMENTING CONTEXTSENSITIVE HELP

826

Other Topics PART VIII

Creating Installation Programs CHAPTER 46

827

Creating Installation Programs

46
CREATING INSTALLATION PROGRAMS

46

IN THIS CHAPTER
s Installation Program Requirements s InstallShield 5 830 828

828

Other Topics PART VIII

Back in the golden days of MS-DOS programming, the programmers task was usually finished the moment the final version of the programs executable file was compiled and linked. The executable was copied to a single floppy disk and was often executed from there; the written manual might have contained a few superficial comments about copying the executable to the hard disk for better performance. As the size of applications grew, however, things quickly became more complex. End users received packages consisting of multiple files or files that were larger than the capacity of floppy disks used to distribute them. As the complexity of the installation task increased, fewer and fewer end users possessed the necessary skills to successfully execute them; the need for automated installation became clear. The first installation programs were often simple MS-DOS batch files. These sufficed when installation meant merely copying the correct files to the right location. However, Windows programs required more, such as changes to shared .INI files, that could no longer be handled by batch files. Microsofts first response to this challenge was in the form of the Windows Setup Toolkit, distributed as part of the Windows SDK. Although this setup toolkit implemented most of the functionality essential for professional setup, it was very cumbersome and difficult to use. Many of the steps toward creating distribution images were manual and error-prone. In recent years, a third-party product, InstallShield Software Corporations InstallShield application, became the setup toolkit of choice for many developers. Its success is doubtless due, in part, to the fact that a light version of the product has been included on all recent Visual C++ CD-ROM releases. Visual C++ 5 is no exception; its CD-ROM includes InstallShield 5 Free Edition, a lightweight but fully functional system for creating professional installations. In the typical development life cycle, often inadequate time is allocated for the development of setup programs. Frequently, the need for a setup program that can be used efficiently by end users is completely forgotten until the product is almost ready for release; by that time, especially if the project is late or over budget, installation programs are created hastily, with inadequate testing. What is all too often forgotten is something I cannot sufficiently emphasize: The setup program is the first piece of software from your company that will run on your end users computers. Any first impression they get of your work will be based on the way your installation works. If your installation doesnt work at all, your new application will never be installed and all your work will have been in vain. Need I say more? This chapter reviews the requirements that professional-quality installation programs must satisfy, then presents a simple hands-on example.

Installation Program Requirements


Present-day installation programs must do much more than merely copy the executable files of your program to the end users hard disk. Several requirements are part of Microsofts logo

Creating Installation Programs CHAPTER 46

829

compliance guidelines; others represent common expectations by end users for installation program features that can be considered industry standard.

46
CREATING INSTALLATION PROGRAMS

Media Types
Applications today are distributed in many forms. Some are distributed on one or more CD-ROMs; others are distributed on floppy disks. Still others are distributed as single, selfextracting files suitable for downloading across the Internet. Most professional applications are distributed in multiple formats to match the requirements imposed by different distribution channels and the varieties of hardware configurations that end users might have. A professionalquality installation toolkit must be able to handle multiple media types and produce installation images for each requested type.

Multiple Pieces of Media


Floppy-diskbased distribution usually requires several disks. The installation toolkit must provide a way to manage disks. The setup program run by end users must be aware that the installation is contained on multiple pieces of media and request the user to insert them as necessary.

Efficient Storage
Windows applications are very space-consuming; even simple programs often require that several megabytes worth of dynamic link libraries be distributed with them. If distribution is done over the Internet or on floppy disks, compression becomes essential.

User-Selectable Options
Most complex applications can be installed in a variety of configurations. Some users do not require installation of your example or tutorial files, for example. Installation of rarely used components can also be made optional. The user may determine the destination location for installation. The installation program, therefore, must provide a user interface where these choices can be made and perform certain related tasks (such as verifying disk space requirements).

Registration and/or Authentication


In the past 15 years, end users repeatedly rejected copy protection methods which, while often causing headaches for legitimate users, didnt really prevent unauthorized distribution. These days, most software manufacturers resort to other copy disincentive methods. These include saving the users name (perhaps in an unalterable, encrypted form) or requesting the user to authenticate himself, perhaps through supplying a registration number. Naturally, support for these capabilities must be provided by the installation program.

830

Other Topics PART VIII

Conditional Copying
A well-written installation program must be capable of detecting the presence of components that are already installed. It should not overwrite any such components with an older or incompatible version.

Configuration Updates
Installation programs must be capable of making changes to the Windows configuration. In other words, they must be able to read and modify the contents of the Registry and .INI files.

Shared Components
Most Windows programs these days are distributed along with a variety of shared components. These may include the MFC and Visual C++ runtime libraries, ODBC, ActiveX controls, or other redistributable components from Microsoft or third parties. A well-designed installation program must provide a solid mechanism for updating such components, including components that might be in use at the time of installation.

Uninstallation
An important feature (and one that is part of the Windows logo requirements) is the capability for an application to uninstall itself. Uninstallation is a complex procedure with many pitfalls. During an uninstallation, an application must completely remove itself, its configuration files, and Registry entries. However, care should be taken not to remove any shared components that might be in use by other applications. Furthermore, special rules govern what are recognized by Microsoft as Windows core components; these components should not be removed under any circumstances.

InstallShield 5
InstallShield 5 is the latest in a series of high-quality, professional toolkits from InstallShield Corporation. If you are familiar with earlier versions of InstallShield, such as InstallShield 3, InstallShield 5 should represent a natural evolution; it adds a graphical interface to InstallShields scripting language and automates many of the tasks that were previously executed manually. A single glance at the InstallShield 5 user interface with a project open (see Figure 46.1) immediately reveals the obvious goal of its designers: to make the InstallShield user interface as similar in appearance to Visual Studio as possible. This makes the product very easy to use for the Visual C++ programmer, although the subtle differences can be a bit unnerving or annoying at times. Perhaps one day in the not-too-distant future Microsoft will incorporate setup toolkit functionality into Visual Studio. Until then, InstallShield provides a very acceptable compromise.

Creating Installation Programs CHAPTER 46

831

FIGURE 46.1.
The InstallShield user interface.

46
CREATING INSTALLATION PROGRAMS

Creating a Project with the Project Wizard


The rest of this chapter reviews the use of InstallShield 5 through the building of an installation program and disk set for an MFC-based Hello, World application. Suppose that the applications executable is named HELLO.EXE and that it is located in the C:\HELLO directory. You create an InstallShield 5 project using the Project Wizard. The Project Wizard is visible as an icon in the Projects window of InstallShield 5. When you first start InstallShield 5, this window with this single icon will be the only items visible inside the InstallShield 5 main window. Double-clicking the Project Wizard icon invokes the first in a series of Project Wizard dialogs (see Figure 46.2). In this dialog, you can specify the applications name and some of its basic characteristics. Enter HELLO as the applications name and specify the location of the applications executable. Also set the applications type to Software Development Application, and leave the remaining settings unchanged. Click the Next button. The second Project Wizard dialog in Figure 46.3 enables you to select which dialogs your setup program should display when it is being run. The retail version of InstallShield 5 also allows custom dialogs; the Free Edition that is packaged with Visual C++ and other development tools allows only a selection from existing choices.

832

Other Topics PART VIII

FIGURE 46.2.
Project Wizard Welcome.

FIGURE 46.3.
Project Wizard Choose Dialogs.

For the Hello, World application you need the Welcome Message, User Information, Choose Destination Location, Select Program Folder, Start Copying Files, and Setup Complete dialogs. Make sure these are checked and the remaining are cleared before clicking the Next button. In the next Project Wizard dialog you can pick the target platforms for your application (see Figure 46.4). InstallShield 5 must know the target platform in order to exercise platformspecific features during its interaction with the end user. You might also need to know which platform is being used, for example, if your application requires the installation of libraries that are platform-specific.

Creating Installation Programs CHAPTER 46

833

FIGURE 46.4.
Project Wizard Choose Target Platforms.

46
CREATING INSTALLATION PROGRAMS

The Free Edition allows three Intel-based 32-bit platforms as targets for your installation: Windows 95 (Windows 98), Windows NT 3.5x, and Windows NT 4.0 (Windows NT 5.0). For the Hello, World application, all three must be selected; this is the default setting. In other words, you can click the Next button to proceed to the next Project Wizard dialog without changing anything here. The fourth Project Wizard dialog, shown in Figure 46.5, lets you choose the target language of your setup program. The Free Edition only allows the English language, so you might as well click the Next button right away.

FIGURE 46.5.
Project Wizard Specify Languages.

834

Other Topics PART VIII

In the next dialog you specify setup types (see Figure 46.6). Complex applications usually offer several setup types to the user; for example, a Compact setup might be best when space is at a premium (on a notebook computer, for example), whereas a Custom setup is preferred by professional users who want to have full control over what components are installed and where.

FIGURE 46.6.
Project Wizard Specify Setup Types.

In this example, because your application is but a single executable, you wont bother the user with multiple setup types. The only setup type you need is the Typical type. In dialog six (shown in Figure 46.7) you can specify the components of your setup. Each setup type is associated with a set of components; when the user selects the Custom setup type, he will be presented with a list of components to choose from for installation. Components, in turn, will be associated with file groups later. Initially, the Project Wizard presents four components. However, you need only two: Program Files and Shared DLLs. You can delete the Example Files and Help Files components before clicking Next. Next is the dialog for specifying file groups (see Figure 46.8). Files that are installed with your application are organized in groups based on common characteristics. For example, you might want to use a file group for files that are specific to an installation platform, or files that are installed conditionally, only if the user selects a certain setup type or custom installation component. Our Hello, World application requires only two file groups: Program Executable Files and Shared DLLs. You can delete the remaining three file groups offered by the Project Wizard. Click Next to activate the final Project Wizard dialog, shown in Figure 46.9. This dialog simply presents a summary of the choices that were made with the Project Wizard. Click Finish and the Project Wizard creates the new project with the requested settings.

Creating Installation Programs CHAPTER 46

835

FIGURE 46.7.
Project Wizard Specify Components.

46
CREATING INSTALLATION PROGRAMS

FIGURE 46.8.
Project Wizard Specify File Groups.

This project, however, is not yet finished. Several steps must be completed before the project can be compiled and disk images can be made. As a minimum, file groups must be populated with files, and the projects script and resource settings must be changed.

Adding Files
You add file groups by selecting the File Groups tab in the InstallShield 5 project workspace pane (see Figure 46.10). For this project, you must add three files: the applications executable to the Program Executable Files group, and the Visual C++ and MFC DLLs to the Shared DLLs group.

836

Other Topics PART VIII

FIGURE 46.9.
Project Wizard Summary.

FIGURE 46.10.
File groups.

InstallShield 5 maintains links to selected files rather than copies. To insert a link into a particular group, click the Links icon under that groups name in the File Groups pane. The File Groups window will appear, into which you can drag files from the Windows Explorer or insert files by right-clicking in this window and selecting the Insert Files command from the pop-up menu. You can insert multiple files at once. Insert the file HELLO.EXE into the Program Executable Files group as in Figure 46.11, and the files MFC42.DLL and MSVCRT.DLL into the Shared DLLs group as in Figure 46.12. The latter two files can be found in the DEVSTUDIO\VC\Redist subdirectory on your Visual C++ CD-ROM. Before the file groups are of any use, they must also be added to the appropriate project components. To do so, click the Components tab in the workspace pane (see Figure 46.13). When you click any of the components in your project, the Components window appears, in which settings specific to that component can be added or modified.

Creating Installation Programs CHAPTER 46

837

FIGURE 46.11.
Adding the applications executable file.

46
CREATING INSTALLATION PROGRAMS

FIGURE 46.12.
Adding shared DLLs.

FIGURE 46.13.
Components.

To the Program Files component (see Figure 46.14), you must add the Program Executable Files group. Double-click the Included File Groups field to invoke a dialog where the file groups can be added. One other setting must be changedthe Installation setting. This setting controls under what conditions a file that might already exist on the users hard disk is going to be replaced. The most appropriate setting for our applications executable is to replace files only if both their version and their date are newer than those of the file on the users system (see Figure 46.15).

838

Other Topics PART VIII

FIGURE 46.14.
The Program Files component.

FIGURE 46.15.
Installation properties.

For the Shared DLLs component (see Figure 46.16) the file group that is to be included is the Shared DLLs file group. The Installation setting must also be modified for this component; in addition, you should also change the destination to the WINSYSDIR directory, which is where DLLs such as the MFC or Visual C++ DLLs must be deposited during installation. There are many other component settings that you might want to alter. For example, if your setup program includes a Custom setup type, you might want to ensure that all components have proper description, display text, and visibility settings. However, because your simple installation will not allow the selection of individual components, you need not worry about these settings.

Creating Installation Programs CHAPTER 46

839

FIGURE 46.16.
The Shared DLLs component.

46
CREATING INSTALLATION PROGRAMS

Editing Resources
Before the installation script is ready to be compiled, it must be customized. Customization entails changing the splash screen bitmap, altering resource strings, and modifying the setup script itself if any changes are required in its behavior. To change the splash screen bitmap, select the Setup Files tab in the project workspace pane (see Figure 46.17). The default splash screen bitmap can be found under the Splash Screen/ Language Independent icon in the Setup Files tree. If you click this icon, the Setup Files window will appear, in which you can select the file setup.bmp. Double-clicking the filename invokes the program associated with the .BMP extension.

FIGURE 46.17.
Setup files.

The Resources tab shown in Figure 46.18 enables you to add and edit string resources.

840

Other Topics PART VIII

FIGURE 46.18.
Resources.

A series of string resources are defined by the Project Wizard; all you must do is change their values to values appropriate to your application (see Figure 46.19).

FIGURE 46.19.
String table.

The Setup Script


The key component to an InstallShield 5 setup project is the setup script. The script uses InstallShields own language, a scripting language that is loosely based on the syntax of C. Installation of the Hello, World application requires a few changes to the setup script supplied by the Project Wizard. First, the script must be updated to install the applications icon in the Start menu. Second, a few changes must be made to display a meaningful summary when the program files are about to be copied. The first change is performed by modifying the SetupFolders function. The Project Wizard has thoughtfully placed a TODO comment into this function, marking it as something you must complete. The modified function, shown in Listing 46.1, calls the AddFolderIcon InstallShield function to install the icon. The command string associated with the icon is created by concatenating the target directory and the executable filename; the InstallShield function LongPathToQuote is used to ensure that if the resulting command line contains any spaces, it is enclosed in quotes.

Creating Installation Programs CHAPTER 46

841

Listing 46.1. The SetupFolders function.


function SetupFolders() STRING szCommandLine; begin szCommandLine = TARGETDIR ^ HELLO.EXE; LongPathToQuote( szCommandLine, TRUE ); AddFolderIcon (svDefGroup, @PRODUCT_NAME, szCommandLine, TARGETDIR, , 0, , REPLACE); return 0; end;

46
CREATING INSTALLATION PROGRAMS

The second function that must be modified is the DialogShowSdStartCopy function. The default version of this function displays a meaningless string; you want to replace it with information about the choices the user made in the previous dialogs. The modified function, shown in Listing 46.2, displays the target directory and Start Menu folder that the user selected.

Listing 46.2. The DialogShowSdStartCopy function.


function DialogShowSdStartCopy() NUMBER nResult; STRING szTitle, szMsg; begin listStartCopy = ListCreate( STRINGLIST ); ListAddString(listStartCopy, , AFTER); ListAddString(listStartCopy, Target Directory, AFTER); ListAddString(listStartCopy, + TARGETDIR, AFTER); ListAddString(listStartCopy, , AFTER); ListAddString(listStartCopy, Program Folder, AFTER); ListAddString(listStartCopy, + svDefGroup, AFTER); szTitle szMsg = szMsg = szMsg = = ; Setup has enough information to start copying files. ; szMsg + If you want to review or change any settings, click Back. ; szMsg + If you are satisfied with the settings, click Next.;

nResult = SdStartCopy( szTitle, szMsg, listStartCopy ); return nResult; end;

To compile the script, invoke the Compile command from the Build menu. Errors and warnings are displayed at the bottom of the InstallShield 5 screen in the Output pane. Dont worry about the warnings that are displayed during compilation; these merely reflect the fact that certain standard features are not used in all installation scripts.

842

Other Topics PART VIII

Building Distribution Sets


Your setup script is compiled, your applications files are identified. What remains is putting it all together by creating installation disk images. To create disk images, invoke the Media Build Wizard from the Build menu. You can name your media in the first dialog of the Media Build Wizard (see Figure 46.20). A default for CDROM media is provided; if you would like to create disk images for floppy disks instead, you might want to add a new media entry labeled Floppy Disk.

FIGURE 46.20.
The Media Build WizardMedia Name.

Click Next to activate the Disk Type dialog shown in Figure 46.21, where you can select the distribution disk type. You need 3.5-inch floppy disks. If you select CD-ROM or Custom Size, the Custom Size and Data as Files settings also become enabled in this dialog.

FIGURE 46.21.
The Media Build WizardDisk Type.

Creating Installation Programs CHAPTER 46

843

The next step is where you specify a full build or quick build (see Figure 46.22). Quick builds use references to files as opposed to copying them, which facilitates faster testing but cant be used for distribution.

46
CREATING INSTALLATION PROGRAMS

FIGURE 46.22.
The Media Build WizardBuild Type.

In the fourth dialog of the Media Build Wizard (see Figure 46.23) you can specify the contents of a tag file. Tag files are placed on distribution media by InstallShield to identify your application.

FIGURE 46.23.
The Media Build WizardTag File.

Next, you can specify installation platforms that this media will support (see Figure 46.24). Only files specific to the selected platforms or files common to all platforms are copied when the distribution disk set is created. The default selection includes all platforms, which is exactly what you need for the Hello, World application.

844

Other Topics PART VIII

FIGURE 46.24.
The Media Build WizardPlatforms.

The last Media Build Wizard dialog, shown in Figure 46.25, provides a review of all the build selections you made. To build the distribution images, click the Finish button.

FIGURE 46.25.
The Media Build WizardSummary.

Testing and Debugging


Now that everything is finished, the new installation is ready to be tested. The simplest test, of course, is to just run it on a variety of platforms; this can help you verify compatibility with the target platforms that your application is supposed to be compatible with. But what if something goes wrong? Few things are more infuriating than error messages like The installation has failed. Tell me something I didnt know!

Creating Installation Programs CHAPTER 46

845

Fortunately, InstallShield 5 comes with a comprehensive debug facility that provides most features one would expect from modern debuggers, including the ability to single-step through script code and examine the values of variables. A script is started for debugging with the Debug Setup command under the Build menu. After the initial decompression and initialization, the script begins executing and is immediately interrupted by the debugger. The debugger displays two windows that can be used to view the code being debugged and control the debugging process. The InstallShield Debugger window in Figure 46.26 provides the capability to watch and manipulate variables. It also contains command buttons that can be used to single-step through code, interrupt or resume script execution, or terminate the debugging process.

46
CREATING INSTALLATION PROGRAMS

FIGURE 46.26.
The InstallShield 5 Debugger.

The other window, shown in Figure 46.27, displays the source file of the script currently being debugged. This window can also be used to install breakpoints; to add or remove a breakpoint, simply double-click on the script line at which you want script execution to be interrupted. All lines containing breakpoints are shown with a red background.

846

Other Topics PART VIII

FIGURE 46.27.
Viewing a script in the InstallShield 5 Debugger.

Summary
Installation programs are complex nowadays. They must satisfy several requirements, including the following: s s s s s s s s s Distribution using different media types Handling multiple pieces of media Efficient storage of large installation file sets User-selectable installation options Registration and/or authentication Conditional copying Configuration updates Shared components Uninstallation

InstallShield has become an industry standard for creating installation programs. Its latest version, InstallShield 5, is packaged with many development tools, including Visual C++ 5. It provides a user interface similar to Visual Studio, a powerful C-like scripting language and debugger, and many automated features that can be used to create skeleton installation projects, identify and select files, and creating distribution media.

User Interface Extensions CHAPTER 47

847

User Interface Extensions

47

IN THIS CHAPTER
s Interfacing with the Shell 848 s Examples 850

47
USER INTERFACE EXTENSIONS

848

Other Topics PART VIII

One of the key features of the user interface that was introduced with Windows 95 is the capability to extend the standard functionality of the Windows Explorer and related components, such as the Taskbar or the Quick View utility. Many newer applications take advantage of these capabilities and install, for example, custom context menus or property pages for their file types. This chapter briefly reviews the shell extension capabilities that applications can take advantage of. I also present two simple examples that demonstrate some of the features discussed.

Interfacing with the Shell


The Windows shell is nothing else but the well-known Windows Explorer (or Internet Explorer 4 on Windows 98, Windows NT 5, or older systems with the desktop integration of this browser enabled). The Explorer provides a number of capabilities, each of which can be extended or customized by applications. Applications can interface with the shell using a combination of Win32 API functions (the Shell Library) and Component Object Model (COM) interfaces.

Namespace
The Explorer maintains a hierarchy of objects. Similar to the hierarchical structure of files and directories, this object hierarchy also includes networks, computers, storage devices, printers, shortcuts, desktop icons, and other shell objects. Objects are organized into folders (which are themselves objects); at the root of all objects and folders is a single root folder: the desktop. The collection of symbols that the shell maintains is called the shell namespace. You can obtain an IShellFolder interface to the desktop object using the SHGetDesktopFolder API function. This interface offers the EnumObjects and BindToObject member functions; using these functions, you can navigate through the entire shell namespace.

Application Desktop Toolbars (Appbars)


Application Desktop Toolbars, or Appbars for short, are toolbars similar to the standard Taskbar; they can align themselves with the edges of the desktop, and they can cause the usable area of the desktop to be automatically reduced. Appbars can contain buttons or other dialog controls. Appbars are created, destroyed, or manipulated using the SHAppBarMessage function.

Interacting with the Taskbar


Applications can also interact with the standard Windows taskbar. The function SHAppBarMessage can be used to obtain information about any desktop toolbar in the system, including the Taskbar. The function Shell_NotifyIcon can be used to place, modify, or delete icons on the Taskbars tray.

User Interface Extensions CHAPTER 47

849

Creating File Viewers


Windows Explorer provides a Quick View command for many file types. When Quick View is selected from the Explorers context menu, a special program, quikview.exe, is invoked; this program, in turn, peruses the Registry to find which, if any, DLLs can be used to parse and display the contents of the current file. File parser DLLs are used to parse the contents of a file. These DLLs must implement a series of functions that are called the display engine, as implemented by quikview.exe and file viewers. File viewers are COM objects. They implement the IFileViewer interface, through which data that needs to be displayed or printed is communicated to the file viewer.

47
USER INTERFACE EXTENSIONS

Writing Shell Extension DLLs


Shell extension DLLs directly enhance the capabilities offered by the Windows Explorer. These DLLs can offer the following extension functionality: s s s s s Additional commands in the Explorers context menus Additional property pages in the Explorers property sheet Dynamically managed icons Type-specific copy behavior Customized drag-and-drop behavior

All these shell extensions are implemented in the form of COM objects. Most shell extension objects implement the IShellExtInit interface, which is used to initialize shell extensions. Some extensions also require implementation of the IPersistFile interface. Context menu handlers implement the IContextMenu interface. Member functions of this interface are called when the DLL is to insert its menu commands into the shells context menu, or when a command has been selected by the user. Icon handlers determine the icons that are to be displayed by the shell for a file type. Unlike the default behavior, which associates icons with file types, icon handlers can specify icons depending on the contents or other characteristics of specific files. Icon handlers implement the IExtractIcon interface. Property page handlers add extra property pages to be displayed when the user selects the Properties command on an item in the shell. This is accomplished by implementing the IShellPropSheetExt interface. A file type can have multiple property page handlers installed in the Registry. Copy hook handlers dont require implementation of the IShellExtInit interface. They must implement the ICopyHook interface; its single member function, CopyCallBack, is called every time the shell is about to move, rename, copy, or delete a file.

850

Other Topics PART VIII

Customized drag-and-drop behavior is implemented through drag-and-drop handlers, data handlers, and drop handlers. Drag-and-drop handlers are nearly identical to context menu handlers; they implement the IContextMenu interface to add menu items to the context menu that is displayed when the user drags a file with the right mouse button. Data handlers implement the IDataObject interface, through which they can add custom clipboard types when a file object is dragged from the shell or copied to the clipboard. Data handlers must also implement the IPersistFile interface. Lastly, drop handlers can turn a file into a drop target by implementing the IPersistFile and IDropTarget interfaces. Shell extensions are installed through the Registry. A key under HKEY_CLASSES_ROOT identifies the file extension with a file type; additional keys under the file type key identify the shell extension DLLs that implement specific extension functionality. Suppose, for instance, that you have an icon handler for documents with the extension .MFT (My File Type). You would then have the following entries in the Registry:
HKEY_CLASSES_ROOT\.mft\[Default]=MyFileType HKEY_CLASSES_ROOT\MyFileType\[Default]=My File Type HKEY_CLASSES_ROOT\MyFileType\shellex\IconHandler\[Default]= {12345678-abcd-1234-cdef-1234567890ab}

Usually, these entries will be made by your applications installation program and removed when the application is uninstalled. The shell extension DLL must reside in the Windows SYSTEM (or NT) directory.
SYSTEM32

under Windows

Other Extensions
In addition to modifying the Explorers behavior, there are other ways to enrich the standard user interface functionality. Control Panel applets can be used to add new functionality to the Windows Control Panel. The behavior of existing items in the Control Panel can also be modified by implementing the IShellPropSheetExt interface. Briefcase Reconcilers add file-type specific functionality to the Windows Briefcase. Reconciliation of Briefcase contents is accomplished through a series of COM interfaces implemented by the Briefcase and by reconcilers. Screen savers are implemented using screen saver library functions in the Win32 API. Screen savers are installed by simply copying their executable file to the Windows system directory.

Examples
Programming Windows shell extensions appears tricky at first sight. However, as the following examples show, practical implementation is often simpler than many think. The first of

User Interface Extensions CHAPTER 47

851

these examples demonstrates a Windows program that installs an icon in the Taskbar; the second example is more complex, and it implements something that may even be considered to be of practical use, a property page for Aldus placeable metafiles that displays metafile header information.

Installing a Taskbar Icon


To install an icon on the Taskbar, an application must do two things: The icon must be added and managed using the Shell_NotifyIcon function, and notification messages from the icon must be processed. This is demonstrated by the HELLO.C sample in Listing 47.1. To compile this sample from the command line, enter cl hello.c user32.lib shell32.lib.

47
USER INTERFACE EXTENSIONS

Listing 47.1. HELLO.C: A Taskbar icon example.


#include <windows.h> LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { NOTIFYICONDATA nid; switch(uMsg) { case WM_CREATE: nid.cbSize = sizeof(nid); nid.hWnd = hwnd; nid.uID = 1; nid.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP; nid.uCallbackMessage = WM_USER; nid.hIcon = LoadIcon(NULL, MAKEINTRESOURCE(IDI_WINLOGO)); strcpy(nid.szTip, Hello); Shell_NotifyIcon(NIM_ADD, &nid); break; case WM_USER: if ((UINT)lParam == WM_LBUTTONDOWN) MessageBox(NULL, Hello!, Hello, MB_OK); break; case WM_DESTROY: nid.cbSize = sizeof(nid); nid.hWnd = hwnd; nid.uID = 1; nid.uFlags = 0; Shell_NotifyIcon(NIM_DELETE, &nid); PostQuitMessage(0); break; default: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return 0; } int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR d3, int nCmdShow)

continues

852

Other Topics PART VIII

Listing 47.1. continued


{ MSG msg; HWND hwnd; WNDCLASS wndClass; if (hPrevInstance == NULL) { memset(&wndClass, 0, sizeof(wndClass)); wndClass.style = CS_HREDRAW | CS_VREDRAW; wndClass.lpfnWndProc = WndProc; wndClass.hInstance = hInstance; wndClass.hCursor = LoadCursor(NULL, IDC_ARROW); wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wndClass.lpszClassName = HELLO; if (!RegisterClass(&wndClass)) return FALSE; } hwnd = CreateWindow(HELLO, HELLO, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL); ShowWindow(hwnd, nCmdShow); UpdateWindow(hwnd); while (GetMessage(&msg, NULL, 0, 0)) DispatchMessage(&msg); return msg.wParam; }

This application displays a featureless window as its main user interface component. When it receives the WM_CREATE message for its window, it installs the Taskbar icon (Figure 47.1) using a call to Shell_NotifyIcon with the NIM_ADD parameter.

FIGURE 47.1.
A Taskbar icon.

Whenever the mouse is over the surface area of the Taskbar icon, the application that created the icon receives notification messages. The message type has been set to WM_USER in the initial call to Shell_NotifyIcon. Whenever a message of this type is received, you can determine the actual message received by the icon by examining the lParam parameter; if it is a WM_LBUTTONDOWN message, a simple dialog (Figure 47.2) is displayed.

User Interface Extensions CHAPTER 47

853

FIGURE 47.2.
Clicking on the Hello icon in the Taskbar.

Adding a Shell Property Page


Adding a property page to the shell is a somewhat more involved task than installing a Taskbar icon. It requires the implementation of several COM interfaces, creation of a dialog, and a number of other, supplementary tasks. The next example implements a single property page for Windows metafiles with the .wmf extension. Many metafiles, although saved with this extension, are not standard in format; the metafile content is preceded by a 22-byte header that contains information about the horizontal and vertical size of the metafile. This information would otherwise be obtainable only by parsing the metafile and determining the extent of every object in it. Metafiles of this type are called Aldus placeable metafiles, or placeable metafiles for short. They can be easily distinguished from ordinary metafiles because the 22-byte header contains both a 4-byte magic word and a 2-byte checksum. Our shell extension will examine the first 22 bytes of the selected metafile, verify whether it is a valid metafile header, and display header parameters. The user will also be able to modify these parameters and write them back to the metafile (Figure 47.3). The definitions for the template of this property page are shown in Listing 47.2; the actual resource script is in Listing 47.3. Note that the edit fields in the dialog template are marked as disabled, while the text field IDC_NOTPLACEABLE is visible; this is the default state of the dialog that will be changed only if a file that is verified as a placeable metafile is encountered.

47
USER INTERFACE EXTENSIONS

Listing 47.2. Shell Extension DLL: RESOURCE.H.


#define #define #define #define #define #define #define #define IDD_WMFPROPS IDC_NOTPLACEABLE IDC_PIXELSPERINCH IDC_LEFT IDC_RIGHT IDC_TOP IDC_BOTTOM IDC_STATIC 106 1000 1001 1002 1003 1004 1005 -1

854

Other Topics PART VIII

FIGURE 47.3.
Property page for placeable metafiles.

Listing 47.3. Shell Extension DLL: WMFPRPSH.RC.


#include <windows.h> #include resource.h IDD_WMFPROPS DIALOG DISCARDABLE 0, 0, 186, 105 STYLE WS_CHILD | WS_VISIBLE FONT 8, MS Sans Serif BEGIN LTEXT The selected file is not a placeable metafile., IDC_NOTPLACEABLE,10,10,138,8 LTEXT Pixels per inch:,IDC_STATIC,10,25,48,8 EDITTEXT IDC_PIXELSPERINCH,60,25,40,14,ES_AUTOHSCROLL | WS_DISABLED GROUPBOX Coordinates in pixels,IDC_STATIC,10,45,165,50 LTEXT Left:,IDC_STATIC,15,60,15,8 EDITTEXT IDC_LEFT,40,60,40,14,ES_AUTOHSCROLL | WS_DISABLED LTEXT Right:,IDC_STATIC,15,74,20,8 EDITTEXT IDC_RIGHT,40,74,40,14,ES_AUTOHSCROLL | WS_DISABLED LTEXT Top:,IDC_STATIC,90,60,16,8 EDITTEXT IDC_TOP,120,60,40,14,ES_AUTOHSCROLL | WS_DISABLED LTEXT Bottom:,IDC_STATIC,90,74,25,8 EDITTEXT IDC_BOTTOM,120,74,40,14,ES_AUTOHSCROLL | WS_DISABLED END

This was easy; now comes the hard part. We must implement two COM objects; one will expose an IClassFactory interface and will be used for object creation, while the other will expose the IShellExtInit and IShellPropSheetExt interfaces necessary for a property sheet extension to work. Listing 47.4 shows the declaration of two object types, CShellExtClassFactory and CShellExt. The header file also declares the METAFILEHEADER type and the shell extensions GUID, used to

User Interface Extensions CHAPTER 47

855

identify this shell extension in the Registry. The GUID is defined using the DEFINE_GUID macro; if you use this application as the basis for creating shell extensions on your own, dont forget to generate a new GUID using uuidgen.exe.

Listing 47.4. Shell Extension DLL: WMFPRPSH.H.


DEFINE_GUID(CLSID_ShellExtension, 0x7fabf160L, 0xa7b2, 0x11d0, 0xb7, 0xd1, 0x02, 0x07, 0x01, 0x1c, 0x43, 0x7c); typedef struct { #pragma pack(push) #pragma pack(1) DWORD key; WORD hmf; SMALL_RECT bbox; WORD inch; DWORD reserved; WORD checksum; #pragma pack(pop) } METAFILEHEADER, *LPMETAFILEHEADER; class CShellExtClassFactory : public IClassFactory { protected: ULONG m_nRefCount; public: CShellExtClassFactory(); ~CShellExtClassFactory(); //IUnknown members STDMETHODIMP QueryInterface(REFIID, LPVOID FAR *); STDMETHODIMP_(ULONG) AddRef(); STDMETHODIMP_(ULONG) Release(); //IClassFactory members STDMETHODIMP CreateInstance(LPUNKNOWN, REFIID, LPVOID FAR *); STDMETHODIMP LockServer(BOOL); }; class CShellExt : public IShellExtInit, /* IPersistFile, */ IShellPropSheetExt { public: char m_szPropSheetFileUserClickedOn[MAX_PATH]; protected: ULONG m_nRefCount; LPDATAOBJECT m_pDataObj; public: CShellExt(); ~CShellExt(); //IUnknown members STDMETHODIMP QueryInterface(REFIID, LPVOID FAR *); STDMETHODIMP_(ULONG) AddRef(); STDMETHODIMP_(ULONG) Release();

47
USER INTERFACE EXTENSIONS

continues

856

Other Topics PART VIII

Listing 47.4. continued


//IShellExtInit methods STDMETHODIMP Initialize(LPCITEMIDLIST pIDFolder, LPDATAOBJECT pDataObj, HKEY hKeyID); //IShellPropSheetExt methods STDMETHODIMP AddPages(LPFNADDPROPSHEETPAGE lpfnAddPage, LPARAM lParam); STDMETHODIMP ReplacePage(UINT uPageID, LPFNADDPROPSHEETPAGE lpfnReplaceWith, LPARAM lParam); };

The implementation of these the CShellExtClassFactory and CShellExt classes, as well as the DLLs main functions and a few utility functions, are shown in Listing 47.5.

Listing 47.5. Shell Extension DLL: WMFPRPSH.CPP.


#define STRICT #define INC_OLE2 #include #include #include #pragma <windows.h> <windowsx.h> <shlobj.h> data_seg(.text

#define INITGUID #include <initguid.h> #include <shlguid.h> #include wmfprpsh.h #pragma data_seg() #include resource.h UINT g_nRefCount HINSTANCE g_hInstance; // Helper functions void SetText(HWND hWnd, SHORT wParam) { char sz[17]; _itoa(wParam, sz, 10); SetWindowText(hWnd, sz); } void GetText(HWND hWnd, SHORT &wParam) { char sz[17]; GetWindowText(hWnd, sz, 17); wParam = atoi(sz); } // DLL functions extern C int APIENTRY DllMain(HINSTANCE hInstance, DWORD dwReason, LPVOID lpReserved) {

User Interface Extensions CHAPTER 47


if (dwReason == DLL_PROCESS_ATTACH) g_hInstance = hInstance; return 1; } STDAPI DllCanUnloadNow(void) { return g_nRefCount <= 0 ? S_OK : S_FALSE; } STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, LPVOID *ppvOut) { *ppvOut = NULL; if (IsEqualIID(rclsid, CLSID_ShellExtension)) { CShellExtClassFactory *pcf = new CShellExtClassFactory; return pcf->QueryInterface(riid, ppvOut); } else return CLASS_E_CLASSNOTAVAILABLE; } // CShellExtClassFactory implementation CShellExtClassFactory::CShellExtClassFactory() { m_nRefCount = 0; g_nRefCount++; } CShellExtClassFactory::~CShellExtClassFactory() { g_nRefCount--; } STDMETHODIMP CShellExtClassFactory::QueryInterface(REFIID riid, LPVOID FAR *ppv) { *ppv = NULL; if (IsEqualIID(riid, IID_IUnknown) || IsEqualIID(riid, IID_IClassFactory)) { *ppv = (LPCLASSFACTORY)this; AddRef(); return NOERROR; } else return E_NOINTERFACE; } STDMETHODIMP_(ULONG) CShellExtClassFactory::AddRef() { return ++m_nRefCount; } STDMETHODIMP_(ULONG) CShellExtClassFactory::Release() { if (--m_nRefCount <= 0) delete this; return m_nRefCount; } STDMETHODIMP CShellExtClassFactory::CreateInstance(LPUNKNOWN pUnkOuter, REFIID riid, LPVOID *ppvObj)

857

47
USER INTERFACE EXTENSIONS

continues

858

Other Topics PART VIII

Listing 47.5. continued


{ *ppvObj = NULL; if (pUnkOuter) return CLASS_E_NOAGGREGATION; CShellExt *pShellExt = new CShellExt(); //Create the CShellExt object if (NULL == pShellExt) return E_OUTOFMEMORY; return pShellExt->QueryInterface(riid, ppvObj); } STDMETHODIMP CShellExtClassFactory::LockServer(BOOL fLock) { return NOERROR; } // CShellExt implementation CShellExt::CShellExt() { m_nRefCount = 0; m_pDataObj = NULL; g_nRefCount++; } CShellExt::~CShellExt() { if (m_pDataObj) m_pDataObj->Release(); g_nRefCount--; } STDMETHODIMP CShellExt::QueryInterface(REFIID riid, LPVOID FAR *ppv) { *ppv = NULL; if (IsEqualIID(riid, IID_IShellExtInit) || IsEqualIID(riid, IID_IUnknown)) { *ppv = (LPSHELLEXTINIT)this; } else if (IsEqualIID(riid, IID_IShellPropSheetExt)) { *ppv = (LPSHELLPROPSHEETEXT)this; } if (*ppv) { AddRef(); return NOERROR; } else return E_NOINTERFACE; } STDMETHODIMP_(ULONG) CShellExt::AddRef() { return ++m_nRefCount; } STDMETHODIMP_(ULONG) CShellExt::Release() { if (--m_nRefCount <= 0) delete this; return m_nRefCount; }

User Interface Extensions CHAPTER 47


STDMETHODIMP CShellExt::Initialize(LPCITEMIDLIST pIDFolder, LPDATAOBJECT pDataObj, HKEY hRegKey) { if (m_pDataObj) m_pDataObj->Release(); if (pDataObj) { m_pDataObj = pDataObj; pDataObj->AddRef(); } return NOERROR; } BOOL CALLBACK WMFPageDlgProc(HWND hDlg, UINT uMessage, WPARAM wParam, LPARAM lParam) { LPPROPSHEETPAGE psp=(LPPROPSHEETPAGE)GetWindowLong(hDlg, DWL_USER); CShellExt *lpcs; LPMETAFILEHEADER lpmfh; HANDLE hFile; DWORD cb; switch (uMessage) { case WM_INITDIALOG: SetWindowLong(hDlg, DWL_USER, lParam); psp = (LPPROPSHEETPAGE)lParam; lpcs = (CShellExt *)psp->lParam; lpmfh = new METAFILEHEADER; memset(lpmfh, 0, sizeof(METAFILEHEADER)); SetProp(hDlg, MFH, lpmfh); if (*(lpcs->m_szPropSheetFileUserClickedOn)) { hFile = CreateFile(lpcs->m_szPropSheetFileUserClickedOn, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (hFile == INVALID_HANDLE_VALUE) break; if (!ReadFile(hFile, lpmfh, sizeof(METAFILEHEADER), &cb, NULL)) goto INITDONE; if (cb != sizeof(METAFILEHEADER)) goto INITDONE; if (lpmfh->key != 0x9AC6CDD7) goto INITDONE; SetText(GetDlgItem(hDlg, IDC_LEFT), lpmfh->bbox.Left); SetText(GetDlgItem(hDlg, IDC_RIGHT), lpmfh->bbox.Right); SetText(GetDlgItem(hDlg, IDC_TOP), lpmfh->bbox.Top); SetText(GetDlgItem(hDlg, IDC_BOTTOM), lpmfh->bbox.Bottom); SetText(GetDlgItem(hDlg, IDC_PIXELSPERINCH), lpmfh->inch); ShowWindow(GetDlgItem(hDlg, IDC_NOTPLACEABLE), SW_HIDE); EnableWindow(GetDlgItem(hDlg, IDC_LEFT), TRUE); EnableWindow(GetDlgItem(hDlg, IDC_RIGHT), TRUE); EnableWindow(GetDlgItem(hDlg, IDC_TOP), TRUE); EnableWindow(GetDlgItem(hDlg, IDC_BOTTOM), TRUE); EnableWindow(GetDlgItem(hDlg, IDC_PIXELSPERINCH), TRUE); INITDONE: CloseHandle(hFile); } break; case WM_DESTROY: lpmfh = (LPMETAFILEHEADER)RemoveProp(hDlg, MFH); delete lpmfh; break;

859

47
USER INTERFACE EXTENSIONS

continues

860

Other Topics PART VIII

Listing 47.5. continued


case WM_COMMAND: if (HIWORD(wParam) == EN_KILLFOCUS) { lpmfh = (LPMETAFILEHEADER)GetProp(hDlg, MFH); switch (LOWORD(wParam)) { case IDC_LEFT: GetText((HWND)lParam, lpmfh->bbox.Left); break; case IDC_RIGHT: GetText((HWND)lParam, lpmfh->bbox.Right); break; case IDC_TOP: GetText((HWND)lParam, lpmfh->bbox.Top); break; case IDC_BOTTOM: GetText((HWND)lParam, lpmfh->bbox.Bottom); break; case IDC_PIXELSPERINCH: GetText((HWND)lParam, (SHORT &)lpmfh->inch); break; } SendMessage(GetParent(hDlg), PSM_CHANGED, (WPARAM)hDlg, (LPARAM)0); } break; case WM_NOTIFY: switch (((NMHDR FAR *)lParam)->code) { case PSN_APPLY: lpmfh = (LPMETAFILEHEADER)GetProp(hDlg, MFH); if (lpmfh->key == 0x9AC6CDD7) { lpmfh->checksum = 0; for (int i = 0; i < 10; i++) lpmfh->checksum ^= *((WORD *)lpmfh + i); lpcs = (CShellExt *)psp->lParam; hFile = CreateFile(lpcs->m_szPropSheetFileUserClickedOn, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (hFile == INVALID_HANDLE_VALUE) break; WriteFile(hFile, lpmfh, sizeof(METAFILEHEADER), &cb, NULL); CloseHandle(hFile); } break; default: break; } break; default: return FALSE; } return TRUE; } UINT CALLBACK WMFPageCallback(HWND hDlg, UINT uMessage, LPPROPSHEETPAGE lppsp) { CShellExt *lpcs=(CShellExt *)lppsp->lParam;

User Interface Extensions CHAPTER 47


if (uMessage == PSPCB_RELEASE) lpcs->Release(); return TRUE; } STDMETHODIMP CShellExt::AddPages(LPFNADDPROPSHEETPAGE lpfnAddPage, LPARAM lParam) { PROPSHEETPAGE psp; HPROPSHEETPAGE hpage; FORMATETC fmte = {CF_HDROP, (DVTARGETDEVICE FAR *)NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL}; STGMEDIUM medium; HRESULT hres = 0; if (m_pDataObj) hres = m_pDataObj->GetData(&fmte, &medium); if (SUCCEEDED(hres)) { UINT cbFiles = 0; if (medium.hGlobal) cbFiles = DragQueryFile((HDROP)medium.hGlobal, (UINT)-1, 0, 0); if (cbFiles < 2) { if (cbFiles) DragQueryFile((HDROP)medium.hGlobal, 0, m_szPropSheetFileUserClickedOn, sizeof(m_szPropSheetFileUserClickedOn)); psp.dwSize = sizeof(psp); psp.dwFlags = PSP_USEREFPARENT | PSP_USETITLE | PSP_USECALLBACK; psp.hInstance = g_hInstance; psp.pszTemplate = MAKEINTRESOURCE(IDD_WMFPROPS); psp.hIcon = 0; psp.pszTitle = WMF; psp.pfnDlgProc = WMFPageDlgProc; psp.pfnCallback = WMFPageCallback; psp.pcRefParent = &g_nRefCount; psp.lParam = (LPARAM)this; AddRef(); hpage = CreatePropertySheetPage(&psp); if (hpage) { if (!lpfnAddPage(hpage, lParam)) DestroyPropertySheetPage(hpage); } } } return NOERROR; } STDMETHODIMP CShellExt::ReplacePage(UINT uPageID, LPFNADDPROPSHEETPAGE lpfnReplaceWith, LPARAM lParam) { return E_FAIL; }

861

47
USER INTERFACE EXTENSIONS

862

Other Topics PART VIII

The first two functions defined here, SetText and GetText, are simple utility functions used to convert numbers into text and vice versa and transfer the data to or from edit boxes. Next are a series of DLL functions. DllMain is called whenever a process attaches to, or detaches from, our DLL; we are only interested in the first case, as that is when the DLLs instance handle can be obtained and saved for later use. DllCanUnloadNow is a function required for shell extensions, and it is used to determine whether the DLL can be removed from memory; that is the case if the DLLs reference count, maintained globally in g_nRefCount, is at or below zero (actually, it should never drop below zero). The third function here, DllGetClassObject, creates a class factory object and returns an interface pointer to it; this can then be used by the shell to create shell extension objects that we support. The implementation of CShellExtClassFactory is straightforward and standard (for COM applications, that is). Objects of type CShellExt are created when the CreateInstance function is called. Implementation of CShellExt also begins with the implementation of the standard IUnknown methods AddRef , Release , and QueryInterface , followed by an implementation of IShellExtInit::Initialize. These functions are simple, straightforward, and self-explanatory. Not so is the method IShellPropSheetExt::AddPages. Because this function references the dialog callback functions WMFPageDlgProc and WMFPageCallback, those are defined first, so lets take a look at them. a fairly standard-looking dialog procedure. What is worthy of note is how a pointer to a PROPSHEETPAGE structure is stored and communicated across invocations of this function: It is added as a user-defined parameter to the window using the SetWindowLong function when the dialog is being initialized. In subsequent calls it is retrieved using GetWindowLong. The dialog procedure responds specifically to WM_INITDIALOG, WM_DESTROY, WM_COMMAND, and messages. In the handler for WM_INITDIALOG, the selected metafile is opened and the header is read. If the metafile is identified as a placeable metafile (the magic word in the handler is correct) then the dialog fields are populated with data and enabled. A pointer to the metafile header structure is also saved for later use using the SetProp function.
WM_NOTIFY WMFPageDlgProc is

The handler for WM_DESTROY destroys this property and also frees the memory allocated for the metafile header. A handler for WM_COMMAND is used to process EN_KILLFOCUS notifications. These notifications are sent when the focus leaves an edit control; for the sake of simplicity, we assume that this means that the user edited the contents of the control. We notify the parent of the property page by sending it a PSM_CHANGED message, which should cause it to enable the Apply button in the property sheet dialog.

User Interface Extensions CHAPTER 47


WM_NOTIFY messages are sent by the property sheet to the property page when the user performs a specific action, such as closing or canceling the dialog or clicking the Apply button. Of these, we are interested in PSN_APPLY notifications, which tell us that the user wishes to apply changes to the metafile. In response, we write the header back to the metafile.

863

The function WMFPageCallBack exists so we can maintain a correct reference count on the CShellExt object. Now we come to CShellExt::AddPages. In this method, we first check how many files are selected by the user; our property page is meaningful only if a single metafile is selected; otherwise, we simply dont display it. If we do decide to display the property page, that is accomplished through a call to CreatePropertySheetPage function. Lastly, there is the implementation of ReplaceFile. This method is used only for property page handlers in the Control Panel; we can simply return an error, as this method should never be called in our DLL. Before we can compile our shell extension, we must also supply a module definition file, as shown in Listing 47.6. This file defines the DLL entry points that the shell will reference.

47
USER INTERFACE EXTENSIONS

Listing 47.6. Shell Extension DLL: WMFPRPSH.DEF.


LIBRARY DESCRIPTION WMFPRPSH Windows Metafile Property Sheet

EXPORTS DllCanUnloadNow @2 PRIVATE DllGetClassObject @3 PRIVATE

To compile this extension, we must compile the resource script and the C++ source file separately and link them with the appropriate libraries to form a DLL. It is best accomplished using a simple make file, shown in Listing 47.7. To use this make file to compile the library, simply type nmake -f wmfprpsh.mak.

Listing 47.7. Shell Extension DLL: WMFPRPSH.MAK.


wmfprpsh.dll: wmfprpsh.obj wmfprpsh.res wmfprpsh.def link /SUBSYSTEM:windows /DLL wmfprpsh.obj wmfprpsh.res /DEF:wmfprpsh.def \ user32.lib shell32.lib uuid.lib oleaut32.lib comctl32.lib del wmfprpsh.lib del wmfprpsh.exp wmfprpsh.obj: wmfprpsh.cpp wmfprpsh.h resource.h cl -c wmfprpsh.cpp wmfprpsh.res: wmfprpsh.rc resource.h rc wmfprpsh.rc

864

Other Topics PART VIII

Now that our DLL is compiled and ready to go, only one question remains: How do we go about installing it? The answer is simple: We need to copy the DLL to the Windows system library (SYSTEM32 on Windows NT), and we need to create the necessary Registry entries. These Registry entries are shown in Listing 47.8. This file is in a format that can be readily understood by the Registry editor. You can type regedit wmfprpsh.reg on the command line, or simply double-click this file in the Windows Explorer; either way, the files contents will be added to the Registry and the new property page should become visible if you right-click on any file with the .wmf extension and select the Properties command.

Listing 47.8. Shell Extension DLL Registry entries.


REGEDIT4 [HKEY_CLASSES_ROOT\CLSID\{7fabf160-a7b2-11d0-b7d1-0207011c437c}] @=WMF Property Page Shell Extension [HKEY_CLASSES_ROOT\CLSID\{7fabf160-a7b2-11d0-b7d1-0207011c437c}\InProcServer32] @=wmfprpsh.dll ThreadingModel=Apartment

[HKEY_CLASSES_ROOT\.wmf] @=WMFFile [HKEY_CLASSES_ROOT\WMFFile] @=Windows Metafile [HKEY_CLASSES_ROOT\WMFFile\shellex\PropertySheetHandlers] @=WMFPage [HKEY_CLASSES_ROOT\WMFFile\shellex\PropertySheetHandlers\WMFPage] @={7fabf160-a7b2-11d0-b7d1-0207011c437c}

Summary
Beginning with Windows 95 and Windows NT 4.0, Windows offers a variety of ways to customize standard interface behavior. Applications can interact with desktop icons and objects representing items such as computers, networks, files, and folders by traversing the shell namespace using a COM interface. Applications can install their own toolbars much like the standard Taskbar. They can also interact with the Taskbar itself and place icons in the system tray. The Windows Explorer can be extended in a variety of ways using shell extension DLLs. These DLLs expose COM interfaces that the Explorer can utilize to modify its context menus, the icons it displays, property pages, drag and drop, and copy behavior. Additional extensions to the user interface include Control Panel applets, screen savers, and Briefcase reconcilers.

Localization: Creating International Applications CHAPTER 48

865

48

Localization: Creating International Applications


IN THIS CHAPTER
s Preparing for Nationalization: Programming Practices 867 s Tools for International Programming 870 s Writing a Unicode Application 874 s Multilanguage Resources 877

48
LOCALIZATION

866

Other Topics PART VIII

My first professional encounter with a computer occurred in 1979, when I wrote some FORTRAN simulation code on a Control Data 3300 mainframe in Budapest, Hungary. Although the Hungarian language uses a variety of vowels with accents and umlauts, nobody raised an eyebrow about the fact that the computers of the day were limited to the 26 letters of the English alphabet. Computers were supposed to be like that; besides, when you communicate with the computer by using punch cards and receive results in the form of difficult-to-read line printer output (typically in uppercase only), what difference does the absence of a few accent marks make? When proper spelling was important, accented characters were simply transcribed. We often saw printouts with transcriptions like SZAAMIITOOGEEP, which is Hungarian for computer. (Szmtgp is also, ironically, one of the few Hungarian words in which every single vowel has a diacritical mark.) This kind of spelling was also familiar to anyone who ever used a Telex machine or sent a postal telegram. What a difference a few years can make! By early 1983, as the popular Commodore 64 home computer began to appear in ever-increasing numbers in Hungary, the need for proper Hungarian language support became evident. More and more computers were used not by professionals but by small entrepreneurs, store clerks, and administrative personnel everywhere. For these non-experts, a Hungarian language interface was essential, and the lack of accented characters was often the cause of confusing and sometimes hilarious errors. (One anecdote was about an electrical company that couldnt determine why it had in its inventory thousands of meters of fkabl, meaning seals intestines. Eventually, staff found out that the database record in question referred to a quantity of f kbel, or main cable. Without the diacritical marks the two words look identical.) Even on a primitive 8-bit computer like the Commodore 64, the solution to the language problem was far from simple. The Commodore 64 had the capability to use a user-defined character set, so you could change the set of graphic symbols, part of the machines standard character set, to represent the required accented characters instead. So far, so good. But how about keyboard input? The machines keyboard was an American QWERTY keyboard, and it wasnt interchangeable (the keyboard was the machine). That problem can be solved by redefining the keyboard layout and supplying the user with keycaps or a keyboard overlay. But additional problems abound. For example, how do you sort alphabetically when the collating sequence no longer coincides with the characters ASCII codes, or when the collating rules of the language prescribe special treatment for digraphs? After you solve the problems of the screen and keyboard, how do you provide national language support for other peripheral devices, such as the printer? I left Hungary in the mid-80s, but I did not leave the problem behind. During my work in Austria and later in my new home, bilingual Canada, the issue of supporting a non-English user interface was one I had to deal with repeatedly. Today, the tools are improved and standardized solutions have emerged; however, as operating systems provide more features that can have national language support implications, the task of nationalization also becomes more complex.

Localization: Creating International Applications CHAPTER 48

867

For the longest time the majority of American software developers enjoyed the luxury of being able to ignore the international market. The vast majority of software sales occurred in English-speaking North America; thus, international versions of even the most popular applications were either nonexistent or appeared much like an afterthoughtbuggy, expensive, unreliable code that was often one or more major version numbers behind the English language version. Today this is no longer the case. The international software marketplace is huge, and its size is increasing rapidly. Furthermore, even in North America, multilanguage support (French in Canada and Spanish in the United States and Mexico) has become a common requirement at many commercial or government organizations. International support can take many forms. Your application might have a single set of executable and library files, capable of operating in many different languages. Alternatively, you can release different language versions as separate products. You may opt for this solution when the difference is more than just a matter of translation. For example, an accounting application might also need to take into account the differences in taxation and accounting laws. The rest of this chapter examines the various aspects of nationalization in the context of Visual C++ software development.

Preparing for Nationalization: Programming Practices


Rare are the programs that are developed in multiple national languages from the beginning. (Doing so would mean an incredible waste of effort and a project management nightmare.) Far more likely, applications are first developed in a single language (English or a common tongue spoken proficiently by all members of the development team), and when the development effort is largely complete, the user interface is translated (often by professional translators). Unfortunately, this approach has a major flaw: Even experienced programmers often make implicit assumptions that are specific to the language used for development.

48
LOCALIZATION

Text Size
The assumption that is the easiest to correct concerns the size of the text. English text generally tends to be shorter than text in most other European languages, so if your dialog boxes are sized to contain just barely the text within, dont be surprised if the same text no longer fits after it has been translated into German or French (see Figure 48.1).

868

Other Topics PART VIII

FIGURE 48.1.
How not to size dialog boxes.

Depending on the Grammar


Somewhat more complex are assumptions that relate to English grammar. Compactness isnt the only remarkable feature of the English language; so is its capability to do away with most word endings, leaving a solitary s to indicate a third person singular verb or the plural form of nouns. This characteristic makes English especially well suited for programming languages and other text-based computer applications (text adventure games are a prime example). Other languages are less lucky. Hungarian, for example, uses hundreds of different word endings to express the person, plural forms, or even the tense of a verbwhat English expresses with pronouns, prepositions, and auxiliary verbs. What is the consequence? Program code that constructs sentences on the fly is out! Consider, for example, the following clever program line:
printf(Customer %s %s %s\n, szCst, bOrd ? placed an order for : requested information about szPrd);

Assuming that szCst is John Doe and szPrd is Turbo Gizmo, then, depending on the value of bOrd, the sentence constructed by this instruction reads either Customer John Doe requested information about Turbo Gizmo or Customer John Doe placed an order for Turbo Gizmo. So far, so good. Your program works wonderfully and your users just love the plain English informational messages. Now suppose that your company is rapidly expanding and opens its Eastern European office in Budapest, and that your flagship product urgently must be translated into Hungarian. Guess what? A translation will not work. The order in which the various elements of the sentence are put together is different. Words need word endings, often depending on the vowels in the words themselves; and in this particular example, one of the word endings even depends on the value of bOrd. Without attempting to present a mutilated version of this code that works for Hungarian, let me just show you the expected output: John Doe vsrl a Turbo Gizmo-rl informcit krt or John Doe vsrl feladott egy rendelst a Turbo Gizmo-ra. Chances are that the Budapest translation firm your company hired to translate your application doesnt know how to edit C code.

Localization: Creating International Applications CHAPTER 48

869

Isolating Language-Specific Information


Speaking of code editing, I follow a fundamental rule in all my development projects, whether the application I am working on is intended to be localized or not: I never put visible text or other locale-dependent material into the source code. Keeping such material separate from the source code is exactly why resource files were invented. You might also find that many translation companies can work with resource files quite comfortably and will be able to take your .rc file and return an equivalent file in the target language with few, if any, errors. If you really must deal with language dependencies in your code (for example, if you want your users to be able to type commands in English), compartmentalize natural language dependencies as much as possible and assume that this part of your code will need to be redeveloped. Language-specific elements in your programs user interface are not limited to text. They also include date and currency formats, thousands separators, the decimal point, and more. Windows 95/98 and Windows NT, like all modern operating systems, support the customization of these elements, although use of these features might not always be trivial. In a perfect world, we would always use functions such as GetNumberFormat to ensure that our applications output appears in the format expected by the international user. But the world isnt perfect: programmers are lazy, deadlines are looming, and few of us would be willing to use monsters like this every time we want properly formatted output:
char szNum[]= 3141592.65358979; char *pszBuf; int nSize; nSize = GetNumberFormat(LOCALE_USER_DEFAULT, 0, szNum, NULL, NULL, 0); pszBuf = new char[nSize]; GetNumberFormat(LOCALE_USER_DEFAULT, 0, szNum, NULL, pszBuf, nSize); MessageBox(NULL, pszBuf, The number:, 0); delete[] pszBuf;

48
LOCALIZATION

Still, if your application is meant to be used in multiple international markets, this is exactly what you should use to avoid a vastly increased localization effort later.

User Input
Localization isnt limited to changing an applications output and user interface. Programs must also accommodate input in foreign languages. Clearly, if your application interprets the users textual input in any way (for example, if it accepts keyed-in commands), the input can be in the users language of preference. Therefore, the localized version of your application must be capable of accepting foreign language input. But there is a catch, a trap that even the mighty giant Microsoft fell for occasionally. Suppose your application provides a macro language. If the command set is translated, must the macro language be changed as well? If so, how will you maintain macro compatibility across localized versions? (Microsofts Excel spreadsheet had localized versions that sported a localized macro language; consequently, although you were able to use Hungarian Excel to open a spreadsheet that was originally created with the U.S. version, macros no longer ran.)

870

Other Topics PART VIII

Another aspect of accepting textual input concerns the use of different character sets. If your application expects input using the ANSI character set, the input will be limited to the characters found in that set and might not be suitable for use with non-Western European languages.

Non-European Languages
I began this section by covering implicit language-specific assumptions that should be avoided. Yet I was guilty of making such an assumption myselfnamely, that all text a program must output can be presented using the extended ASCII (ANSI) character set. Obviously, this isnt the case unless the language is a typical Western European language that this character set is supposed to support, such as English, French, and German. Eastern European languages and Turkish use accented characters that are not part of the ANSI character set; Greek, Russian, and Ukrainian use a different alphabet; Arabic and Hebrew use a different alphabet and rightto-left script; most Asian languages use syllabic script consisting of thousands of symbols. If you expect your application to be translated into a non-Western European language, you must ensure that it can correctly handle character sets that differ from the default Windows character set.

Tools for International Programming


Clearly, writing applications for the international market takes more than just observing a few prudent programming practices. The Win32 API offers several international programming features. Everybody familiar with the Windows 95 or Windows NT Control Panel knows that under these operating systems you can customize several locale-dependent functions and features. This possibility exists because Windows recognizes the concept of a locale, a collection of settings and information that describes a national language environment.

Locales
A locale in the Windows programming environment describes all settings specific to a national language. These settings include the decimal point, thousands separator, collation sequence, date, time, currency formats, and more. Two sets of locale-related functions are available to the C/C++ programmer. One set includes ANSI-compatible functions that control the behavior of the standard C/C++ libraries. Another set of functions form part of the Win32 API; these functions control the locale-specific behavior of graphical applications. The function setlocale is an ANSI-compatible function that determines the locale-specific behavior of many other ANSI functions, including such commonly used ones as printf. Table 48.1 shows the localization categories that can be affected by setlocale; these categories are represented by manifest constants that are defined in locale.h.

Localization: Creating International Applications CHAPTER 48

871

Table 48.1. Localization categories used by setlocale. Category Description


LC_ALL LC_COLLATE LC_CTYPE LC_MONETARY LC_NUMERIC LC_TIME

All categories Collating sequence Character classification Currency formatting Number formatting Time and date formatting

Consider a simple example. The program in Listing 48.1 takes a locale name as a commandline parameter, sets the locale, and outputs a double precision number using printf. To compile locout.c, just type cl locout.c on the command line.

Listing 48.1. Using the setlocale function.


#include <stdio.h> #include <locale.h> void main(int argc, char *argv[]) { if (argc != 2) { printf(Usage: %s locale-name\n, argv[0]); exit(1); } if (setlocale(LC_ALL, argv[1]) == NULL) { printf(Could not set locale.\n); exit(1); } printf(%f\n, 31415.92654); }

48
LOCALIZATION

Invoking the program with English or German as the command-line parameter produces the following results:
C:\TEMP>locout English 31415.926540 C:\TEMP>locout German 31415,926540 C:\TEMP>

The Visual C++ documentation lists a large number of locale strings accepted by the setlocale function. However, even though a string is on the list, it might not represent a valid locale on your system. For example, if you try to use Chinese as the locale string on a computer with a North American version of Windows and with no Far Eastern character support installed, setlocale will likely return an error.

872

Other Topics PART VIII

The function setlocale controls the behavior of the standard C/C++ library, but another function, SetThreadLocale, controls the locale-specific behavior of the Win32 API. SetThreadLocale takes a locale identifier as its parameter; such identifiers consist of a language identifier and a sorting identifier and can be constructed with the MAKECLID macro. Several functions assist in constructing strings representing time, date, number, and currency values according to the current local settings: GetTimeFormat, GetDateFormat, GetNumberFormat, and GetCurrencyFormat. You can also override specific aspects of locale behavior by using the SetLocaleInfo function. Listing 48.2 contains a simple Windows programming example that uses the SetThreadLocale function to change the locale and GetNumberFormat to format a number in accordance with the new locale setting. To compile this program, use the following command line: cl winloc.cpp user32.lib. The program will display a number, using the number format that is applicable to the German language.

Listing 48.2. Using the SetThreadLocale function.


#include <iostream.h> #include <windows.h> void main(void) { char szNum[]= 3141592.65358979; char *pszBuf; int nSize; LCID lcID = MAKELCID(LANG_GERMAN, SORT_DEFAULT); nSize = GetNumberFormat(lcID, 0, szNum, NULL, NULL, 0); pszBuf = new char[nSize]; GetNumberFormat(lcID, 0, szNum, NULL, pszBuf, nSize); MessageBox(NULL, pszBuf, The number is:, 0); delete[] pszBuf; }

So when would you use these functions to set and manipulate locales explicitly? The answer is simple: when your application must use a locale other than the default system locale. If you want to use the default system locale (presumably representing the users preference), do nothing. However, if your application is capable of operating in a language other than the system default, you might need to explicitly control the locale.

Character Sets
Locales solve one key aspect of the localization problem: country-specific behavior of many functions that do collating, number formatting, date, time, and currency output. However, locales dont provide a solution for non-Western European languages that use characters not found in the extended ASCII (ANSI) set.

Localization: Creating International Applications CHAPTER 48

873

Over the years several solutions that deal with this problem have been proposed and implemented. Among their objectives are compatibility with 7-bit ASCII, compatibility with existing applications, and versatility that allows the use of many world languages. Often, these objectives are contradictory, and none of the solutions are perfect; they all represent compromises. Three methods for representing non-English character sets are commonly used on IBMcompatible personal computers: code pages, multiple-byte character sets, and Unicode. Code pages redefine the symbol set associated with byte codes; multiple-byte character sets extend the single-byte symbol set; and Unicode defines an internationally standardized double-byte character set. These methods are not mutually exclusive. Windows NT, for example, implements Unicode but provides both code pages and multiple-byte character sets to represent specific locales and to maintain compatibility with existing applications. Code pages are simple to use and dont require changes in applications. However, code pages have a major disadvantage: only one code page can be used at a time, so text written in different languages cannot be mixed. Furthermore, using the wrong code page can result in incorrectly interpreted data (for example, wrong characters are displayed or an invalid collating sequence is used). The ASCII character set and many extended character sets use single bytes to represent characters; these character sets are known as (not surprisingly) single-byte character sets (SBCS). Multiple-byte character sets (MBCS) specify one or more lead bytes that precede multiple-byte codes. MBCSs are used to represent syllabic languages for which the 256 8-bit symbols of an SBCS would be insufficient to represent all the characters in a language. Under Windows 95 and Windows NT, a code page specifies, among other things, the SBCS or MBCS that is used on the system. Programming with an MBCS is just plain old C/C++ programming with character strings. The recommended way to deal with syllabic script is to use Unicode; the use of MBCS (and programming in Far Eastern languages in general) is far beyond the scope of this chapter. Unicode has many advantages over the SBCS/MBCS solution but also has a major disadvantage: Because all characters are represented by two bytes, Unicode no longer is compatible with 7-bit ASCII. Standard library functions (such as strcpy) that work well with SBCS/MBCS must be rewritten for Unicode. However, this incompatibility might well be a price worth paying for the capability to represent text that is written in multiple languages. The next section looks at a few simple Unicode programming examples. But first, consider another aspect of non-European scripts: not all languages write from left to right. Some, most notably Arabic and Hebrew, write from right to left; an application that offers a user interface in these languages must take this into account. The latest versions of the Win32 API, as implemented in Windows 95/98 and Windows NT, offers several functions that can facilitate rightto-left output.

48
LOCALIZATION

874

Other Topics PART VIII

Writing a Unicode Application


The major difference between Unicode and non-Unicode applications is that Unicode uses two-byte characters. Clearly, this renders Unicode strings incompatible with C and Windows library functions that handle normal (8-bit) character strings. In fact, the incompatibility begins right with the main (or WinMain) function: because these functions use character pointers as parameters, equivalent Unicode functions are required.

NOTE
Unicode applications are supported only under Windows NT. Windows 95 provides only minimal Unicode support.

Typically, Unicode versions of system functions are differentiated from the normal versions by the letter w that is added to their names. Thus, printf becomes wprintf, strcpy becomes wstrcpy, and so on. These functions use the short type in place of char to represent characters; because variables of type short are at least 16 bits wide, this type is sufficient to hold Unicode character values. The C/C++ syntax for character constants and string constants is also inadequate when it comes to Unicode. Constants such as A and Hello denote 8-bit characters and strings. Unicode character and string constants can be represented by preceding the character or string constant with an uppercase L, as in LA and LHello.

Unicode in Console Applications


Under Windows NT, Unicode can be used both in console applications and in graphical programs. The first Unicode example presented here (Listing 48.3) uses Unicode functions to output its own name (obtained through the parameters of the wmain function) as part of a Unicode string on standard output.

Listing 48.3. A simple Unicode program.


#include <stdio.h> void wmain(int argc, wchar_t *argv[]) { wprintf(L%s greets the world: Hello, World!\n, argv[0]); }

Compiling this program is as simple as compiling a non-Unicode C program: at the command line, type cl whello.c. The compiler recognizes that this is a Unicode application when it encounters the wmain function, and links the compiled program with the appropriate libraries.

Localization: Creating International Applications CHAPTER 48

875

If your source code is in C++, you must also declare wmain as extern C. Otherwise, you might receive a complaint from the linker about a missing main function. Instead of using functions specific to Unicode, you can use generic mappings defined in the header file tchar.h. This header file defines symbols and type names that can be used in both Unicode and non-Unicode applications, thus improving the portability of your code. For example, this file maps the symbol _tprintf into either printf or wprintf, depending on whether the preprocessor symbol _UNICODE is defined. By using tchar.h, you can rewrite the simple program in Listing 48.3 to look like the one in Listing 48.4.

Listing 48.4. Portable program that uses generic mappings.


#include <tchar.h> #include <stdio.h> void _tmain(int argc, TCHAR *argv[]) { _tprintf(_T(%s greets the world: Hello, World!\n), argv[0]); }

To compile this program as a non-Unicode application, enter cl line; to compile as Unicode, enter cl /D_UNICODE hello.c.

hello.c

on the command

Unicode in Windows Applications


Writing Unicode applications that use the Windows graphical user interface is just as simple as writing a Unicode console application. As you guessed, WinMain must be replaced with wWinMain; however, it isnt necessary to replace Win32 API calls with Unicode equivalents. Instead, it is sufficient to define the symbol UNICODE.

48
LOCALIZATION

NOTE
No, this isnt a typo. The symbol this time is indeed UNICODE, without the leading underscore. Because some header files expect UNICODE and others expect _UNICODE, the best approach is to define both of them at the same time. Some header files (in particular, the MFC header files) automatically define one if the other has been defined.

This simple approach works because the Win32 API header files declare Win32 functions conditionally. Consider, for example, the MessageBox function. It is declared in winuser.h as follows:
WINUSERAPI int WINAPI MessageBoxA( HWND hWnd, LPCSTR lpText,

876

Other Topics PART VIII


LPCSTR lpCaption, UINT uType); WINUSERAPI int WINAPI MessageBoxW( HWND hWnd, LPCWSTR lpText, LPCWSTR lpCaption, UINT uType); #ifdef UNICODE #define MessageBox MessageBoxW #else #define MessageBox MessageBoxA #endif // !UNICODE

From this, it should be clear that the Win32 API actually contains two functions, MessageBoxA and MessageBoxW. The symbol MessageBox is a generic macro set to reference either one or the other, depending on whether UNICODE is defined. Listing 48.5 shows a simple Windows program (displaying the contents of its command line in a message box) that uses these generic macros and can be compiled both with and without Unicode support.

Listing 48.5. A simple Unicode program for Windows.


#include <tchar.h> #include <windows.h> int WINAPI _tWinMain(HINSTANCE hI, HINSTANCE hPrevI, LPTSTR lpCmd, int nCmd) { MessageBox(NULL, lpCmd, _T(Message), 0); }

To compile this application from the command line without Unicode support, enter the following:
cl winhello.c user32.lib

To compile with Unicode support, use the following command line:


cl /DUNICODE /D_UNICODE winhello.c user32.lib

Unicode and MFC


With the exception of ODBC database classes (ODBC doesnt support Unicode), the Microsoft Foundation Classes library also provides Unicode support. Creating a Unicode-enabled MFC application requires two steps: defining the _UNICODE preprocessor constant and specifying to the linker the program entry point wWinMainCRTStartup.

Localization: Creating International Applications CHAPTER 48

877

Both these changes can be made through the Project Settings dialog box (see Figures 48.2 and 48.3). Dont forget to set the changes for both the debug and release configurations of your project. Alternatively, if you want to create both a Unicode and a non-Unicode version of your application, you might consider creating new configurations with Unicode settings.

FIGURE 48.2.
Adding _UNICODE to preprocessor definitions.

FIGURE 48.3.
Setting the entry-point symbol for Unicode applications.

48
LOCALIZATION

Multilanguage Resources
You can avoid much of the grief that accompanies localization by ensuring that all languagedependent elements of your code are located in resource files. However, you must still create, maintain, and use resource files that contain the resources of your project in multiple languages. Several strategies are available, each with advantages and drawbacks. Which strategy you choose is a function of your projects requirements.

878

Other Topics PART VIII

Before reviewing these strategies, you should know something about the translation process itself.

Translating the User Interface into a Foreign Language


With large projects, the job of translating the user interface or end-user documentation is usually contracted to a professional translation agency. Many of these agencies have experienced translators who can work with resource (.rc) files, help file source (in rich text format), or other formats such as HTML. In contrast, small projects are often translated in-house, whether the project is a helper application that your company wants to send off to its branch office in Mexico City, or perhaps a shareware application that you wrote in your spare time and want to market in Europe. The language of computers is changing rapidly. Hundreds, if not thousands, of new terms and phrases have been introduced to the English language over the past few decades, most of them during the past 10 years. It stands to reason that other languages have experienced the same fatethey, too, have absorbed a multitude of technical expressions. What does this mean for translating an applications user interface? The answer is simple: even being a native speaker of the target language doesnt guarantee that you will be aware of the latest technical terminology. I experience this regularly with my own mother tongue. Not only do I encounter new phrases every day in material coming out of Hungary, but I also get to see how my fellow expatriate Hungarians mutilate the language when they try to talk about computer-related problems. Inappropriate translations and the excessive use of English language terms abound; the latter is especially harmful in professional translations because the overuse of English creates an impression of snobbery. Fortunately, there is a solution. Microsoft published a series of foreign-language glossaries as part of its Developer Network, Professional Edition subscription. Thousands of words and phrases from the user interfaces of most popular Microsoft applications can be found here in easy-to-use, comma-delimited text files. The glossaries not only reflect current usage in the chosen language, but also (and more important) represent usage sanctioned by Microsoft that is, phrases and expressions that your users encounter daily simply by using the localized version of Windows itself. In fact, I used these glossaries to create the multilingual example presented later in this chapter. Neither my German nor my Russian can be called fluent, so without the glossaries I would have felt quite helpless. (Even with the glossaries, I am likely to have made a mistake or two, for which I apologize to the native speakers of these languages.) But never mind German or Russian; I admit I even used the Hungarian glossaries to verify my translation choices in my own mother tongue.

Localization: Creating International Applications CHAPTER 48

879

Multiple Application Versions


After a translated version of your application becomes available, the next question is how your application will be distributed to end users. The least innovative solution is to create separate executables for each national language. The major advantage of this solution is that it allows more than just resource localization. Your application can also contain conditionally compiled code that is specific to a certain locale. Indeed, if localized versions have extensive differences, this approach might be the only feasible one. The disadvantage of this solution is that you must maintain locale-specific executables, distribution sets, and more. With several target languages, this approach can quickly become a serious maintenance problem.

Satellite DLLs
If the difference between localized versions is limited to the user interface, you can use another approach: instead of linking your applications executable with its resources, they can be linked into a separate dynamic-link library file, known as a satellite DLL. There will be a DLL corresponding to each of the languages your application supports. At startup time, your application can determine the current system locale and select the corresponding DLL as the source for its resources. This solution has the advantage of limited overhead: only locale-dependent components are duplicated, not the entire application. It is also possible to locate, in addition to resources, locale-dependent code (such as conversion routines) in the satellite DLL. Yet another advantage is that you can install multiple satellite DLLs, allowing the user to make a language selection at runtime. Disadvantages of this approach include the need for a relatively complex installation script and the need to maintain several satellite DLL projects.

48
LOCALIZATION

Multiple-Language Resources
Windows NT and Windows 95/98 offer a far more elegant solution for resource localization. How would you like your application to have a single executable file that automatically adapts itself to the current locale on the target machine, displaying information in that language or falling back to a default language if a user interface in the system locale language isnt available? This is exactly what you can accomplish using multiple-language resource files. The idea behind this solution is that a resource file in the 32-bit environment can contain multiple copies of the same resource, distinguished only by a locale identifier. Here is a fragment from a multiple-language resource file as generated by Visual Studio:

880

Other Topics PART VIII


LANGUAGE LANG_GERMAN, SUBLANG_GERMAN STRINGTABLE PRELOAD DISCARDABLE BEGIN AFX_IDS_APP_TITLE HELLO AFX_IDS_IDLEMESSAGE Hilfe erhalten Sie mit F1 AFX_IDS_HELPMODEMESSAGE Klicken Sie auf das Element, zu dem Sie Hilfe wnschen END LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US STRINGTABLE PRELOAD DISCARDABLE BEGIN AFX_IDS_APP_TITLE HELLO AFX_IDS_IDLEMESSAGE For Help, press F1 AFX_IDS_HELPMODEMESSAGE Select an object on which to get Help END

As you can see, two nearly identical blocks of resource compiler statements are here. The same set of symbols is defined in both blocks, which would normally be an error; however, the blocks are preceded by LANGUAGE statements that specify the language of all resource compiler statements that follow. When a resource file like this is compiled, all localized copies of resources are included in the compiled result. Which resource is loaded is determined at runtime, depending on the current locale setting. The end result is an application whose user interface pops up in the users choice of language without any programming effort on your behalf. This is, if I may allow a colloquialism here, powerful stuff. Using multiple-language resources is trivially easy with MFC applications (although there is a catchmore about it later under Installation Issues). I prepared a simple example, an international version of my MFC Hello, World program, that can operate in four languages: English, German, Russian, and Hungarian. This multilingual project is created like any other MFC application. I chose a single document interface (SDI) application for simplicity. The only function that is modified is the OnDraw member function of the view class, as seen in Listing 48.6.

Listing 48.6. Adding drawing code to the multilingual Hello program.


void CHELLOView::OnDraw(CDC* pDC) { CHELLODoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here CString str; str.LoadString(IDS_HELLO); CFont font; font.CreateStockObject(DEVICE_DEFAULT_FONT); CFont *pOldFont = pDC->SelectObject(&font); pDC->TextOut(10, 10, str); if (pOldFont) pDC->SelectObject(pOldFont); }

Localization: Creating International Applications CHAPTER 48

881

The identifier IDS_HELLO represents a string that must be added to the string table in the applications resource file (see Figure 48.4).

FIGURE 48.4.
Adding a string to the resource files string table.

So far, so good. Compiling this application produces the expected result, a simple MFC program that displays the string Hello, World in the otherwise blank client area of its main window. How would this program become a multilingual application? To create a foreign-language copy of an existing resource, simply right-click the resource identifier in ResourceView and select the Insert Copy command (see Figure 48.5). Visual Studio displays the Insert Resource Copy dialog box, as shown in Figure 48.6, in which you can specify a condition (that is, the name of a symbol that must be defined in order for this resource to be included), a language, or both.

48
LOCALIZATION

FIGURE 48.5.
Inserting a copy of a resource.

FIGURE 48.6.
The Insert Resource Copy dialog box.

882

Other Topics PART VIII

In the case of this simple application, the following resources must have foreign-language copies: the IDD_ABOUTBOX dialog box, the IDD_MAINFRAME menu, and the string table. You might also want to add foreign-language sections to the applications default version resource. Adding a resource in German can be done without any difficulties (assuming you know German of course), but Hungarian presents a problem. Some characters in the Hungarian language cannot be represented using the standard ANSI character set. Clearly, Unicode would be the answer except that the resource compiler doesnt support Unicode! What is a minor problem in the case of Hungarian becomes a major one with Russian. How can we, without Unicode support, create and edit Russian-language resources? The answer is surprisingly simple. Although the resource compiler doesnt support Unicode, the language setting determines how the compiler interprets the characters in a resource file. Thus, for any resources defined as Russian, the resource compiler will interpret characters in accordance with the appropriate Russian-language code page. What does this behavior of the resource compiler mean in practice? If your version of Windows is Russian or, in the case of Windows NT, is configured with Russian as the default locale, you will be able to edit Russian-language resources directly. If your Windows version isnt configured with Russian, you can still edit Russian-language resources, but they will all look weird because characters will not be mapped correctly (see Figure 48.7). Dont worry, they will actually appear normal on a Russian system. Clearly, editing resources directly this way isnt practical, but you can copy and paste text from an editor (such as WordPad) that can handle Unicode text.

FIGURE 48.7.
How Russian resources appear on a US-English system.

Translating all resources, even for a simple application like this example, takes some time. After all your translations are complete, as shown in Figure 48.8, you can recompile and test the application. As long as you test it on an English system, you will find no difference: it will behave exactly as before.

Localization: Creating International Applications CHAPTER 48

883

Testing the application on a non-English system might not be an option available to everyone. Fortunately, there is an alternative: if you are using Windows NT, you can switch the default system locale to Russian, and after rebooting the system, it will behave similarly to native Russian versions. To switch the default input locale, invoke the Regional Settings applet from the Control Panel, select the Russian locale, and click the Set as System Default Locale check box (see Figure 48.9). You must reboot the system (and you might need your Windows NT CD-ROM because some files might need to be copied from there). Dont be surprised at the different look and feel: even the Latin characters of the Russian character set appear different in dialog boxes and on the desktop and taskbar.

FIGURE 48.8.
Multiple-language resources.

48
LOCALIZATION

If you start the application after changing the default system locale to Russian, it will automatically start up using Russian-language resources (see Figure 48.10). What happens if the default system locale isnt one that your application supports? In that case, it will use its default locale (English in this case). Sometimes, however, this isnt good enough; you might want to give your users explicit control over the language used by your applications user interface, regardless of the system default locale they might be using. Although your clients might want to be able to change languages on the fly, it can add an unpredictable amount of complexity to your application because locale-dependent resources that are already in memory must be reloaded. Because language changes are not expected to happen routinely, requiring the user to restart the application after the change is much simpler. The selected language setting can be saved in the Registry and applied in the InitInstance member function of your programs application class, as shown in Listing 48.7.

884

Other Topics PART VIII

FIGURE 48.9.
Changing the default system locale.

FIGURE 48.10.
Greeting the world in Russian.

Listing 48.7. Changing the locale at application initialization.


BOOL CHELLOApp::InitInstance() { SetThreadLocale(MAKELCID(LANG_GERMAN, SORT_DEFAULT)); ...

The end result is an application that comes up using the selected language, regardless of the default system locale setting. Note that this solution can be used to switch between languages that are compatible with the default system locale, but not for other languages. For example, if you switch the applications

Localization: Creating International Applications CHAPTER 48

885

language to Russian on an English system, the result will be unreadable (see Figure 48.11). The reason is that the fonts required to display menus and dialog boxes in Russian will not have been installed. This problem is expected to be addressed by the upcoming version of Windows NT 5.0.

FIGURE 48.11.
Running a program in Russian when the default system locale is English.

Help Files
The solutions described so far can be used to localize your application. What about online documentation? The answer is simple: you will need to create multiple copies of all online documentation for the languages you support. You must also ensure that, depending on the system default setting or the users locale preference, the correct documentation file will be loaded.

48
LOCALIZATION

Installation Issues
As mentioned earlier, there is a catch when you use multiple-language resource files, especially in the context of MFC applications. The catch is that a typical application uses user-interface resources from three different sources: its own resource files, the operating system, and additional DLLs (such as the MFC DLLs). Using a multiple-language resource file will not change the language of resources coming from the latter two sources. Changing the language of resources supplied by the operating system is neither possible nor desirable; such a change would necessarily be global, affecting all applications. The last thing you want is an angry user who complains that ever since he tried to install your program, his system displays error messages in Swahili! If you simply must display system resources in a language other than that of the Windows version you are using, one option is to use customized versions of these dialog boxes in your application. You can easily override common system resources such as the File Open dialog box. Messages might still appear in the language of the Windows installation, but these will be relatively few and far between (and might not be under your applications control at all).

886

Other Topics PART VIII

The case of MFC resources is more complex. Most applications link with MFC dynamically. Localized versions of MFC are in separate DLLs (such as MFC42DEU.DLL) that you might need to install along with your application; furthermore, you must take steps to ensure that your application links with the correct DLL. Because the presence of multiple localized versions of MFC can add considerable complexity to the installation process, Microsoft recommends that you add localized versions of MFC resources to your projects resource files instead. You can do this easily if you link statically with MFC; if you link dynamically, the solution is somewhat trickier, as described in Technical Note 57 in the Visual C++ documentation. Finally, you might also want to localize your installation program itself. If you use InstallShield, you might need to purchase a localized version of the InstallShield product.

Summary
Creating applications in a language other than English takes more than merely translating the applications user interface: it requires that the character sets, device settings (keyboard and printer), collating sequence, currency, date/time, and number formatting all match the conventions of the target language. The localization exercise is eased greatly if you follow proper programming practices from the beginning. In particular, applications should not contain locale-specific information (such as text strings) that is hard-coded as part of the source code. In the case of Windows applications, locale-specific components should, whenever possible, be stored in the applications resource file. The Win32 API offers two key tools that facilitate localization. Locales are collections of settings that describe the conventions of a particular language. Using locales is somewhat different for console applications (where ANSI-compatible locale functions control the behavior of the standard C/C++ library) and graphical applications that use the locale functions of the Win32 API. The other key tool to internationalization is the Unicode standard of double-byte characters. This format, supported under Windows NT, allows multiple languages to coexist, even when using different scripts. However, because Unicode represents characters using two bytes, using different library functions to perform common text-related tasks is necessary. It is also possible to use a set of generic macros that maps into single-byte or Unicode function implementations, depending on whether a compile-time symbol (_UNICODE) is defined. For Windows applications, the compile-time symbol is UNICODE, and the mapping takes place automatically.

Localization: Creating International Applications CHAPTER 48

887

Translating an applications resources into a foreign language is made easier by the international glossaries published by Microsoft. These glossaries contain current technical terminology sanctioned by Microsoft and used in Microsoft products. Multiple-language resources can be implemented through separate executables, through satellite DLLs (containing localized resources and, optionally, other locale-dependent components), or through the use of multiple-language resources. Such resources enable an application to present its user interface in the language specified by the default system locale, with no need to recompile or in any way change the application.

48
LOCALIZATION

888

Other Topics PART VIII

Bibliography

889

Bibliography

890

Bibliography

Black, Uyless. TCP/IP & Related Protocols, Second Edition. McGraw-Hill, New York, New York, 1994. Bronstein, I.N., and K.A.Semendayev. Spravotchnik po Matematike, Gostechizdat, Moscow, 1954; Hungarian translation, Matematikai Zsebknyv, M szaki Knyvkiad, Budapest, 1963. Date, C.J. A Guide to the SQL Standard, Second Edition. Addison-Wesley Publishing Company, Reading, MA, 1989. Ellis, Margaret A. and Bjarne Stroustrup. The Annotated C++ Reference Manual. Addison-Wesley Publishing Company, Reading, MA, 1991. Kernighan, Brian W. and Dennis M. Ritchie. The C Programming Language, Second Edition. Prentice Hall, Englewood Cliffs, NJ, 1988. Microsoft Corporation, Microsoft Developer Network Developer Library. January, 1997. Nelson, Mark, C++ Programmers Guide to the Standard Template Library. IDG Books Worldwide, Foster City, CA, 1995. Press, William H., Saul A. Teukolsky, William T. Vettering, and Brian P. Flannery. Numerical Recipes in C, Second Edition. Cambridge University Press, Cambridge, United Kingdom, 1992. Sams Publishing, multiple authors. Programming Windows 95 Unleashed. Sams Publishing, Indianapolis, IN, 1995. Sams Publishing, multiple authors. Windows NT Workstation 4 Unleashed. Sams Publishing, Indianapolis, IN, 1996. Shirley, John, and Ward Rosenberry. Microsoft RPC Programming Guide. OReilly & Associates, Sebastopol, CA, 1995. Stevens, W. Richard. UNIX Network Programming. Prentice Hall, Englewood Cliffs, NJ, 1990. Stroustrup, Bjarne. The C++ Programming Language, Second Edition. Addison-Wesley Publishing Company, Reading, MA, 1991. Toth, Viktor T. Visual C++ 5 Unleashed, Second Edition. Sams Publishing, Indianapolis, IN, 1997.

891

INDEX

892

Symbols UNLEASHED

Symbols
, (comma), dial modifier character, 715 ! (exclamation mark), dial modifier character, 715 $ (dollar sign), dial modifier character, 715 -1 filter expression value, 221 -debug, command-line option, 651 -install, command-line option, 651 -remove, command-line option, 651 ... (ellipses), exception handler, 228-230 ; (semicolon), dial modifier character, 715 << operator, type archiving, 359-360 >> operator, 359-360 @ (at symbol), dial modifier character, 715 [CONFIG] section (help project files), 810 [MAP] section (help project files), 810 0 (zero) filter expression value, 221 1 filter expression value, 221 16-bit programs library functions, 163 looping threads, 41 multitasking, 143 32-bit programs, 161 address calculations, 163 integer size, 162 library functions, 163 memory models, 163 selector functions, 163 type modifiers, 162

A
Abort function, 354 AbortDoc function, 345 acceleration tables, 446 accelerators, 101 accessing physical memory, 175 properties, ActiveX controls, 329 Registry keys, 212-213 ACM (Audio Compression Manager), 764 Activate function, 509 ACTIVATE window mangement message, 73 activating OLE servers, 440 ActiveDocument property (application objects), 507 ActiveX, 426, 512 ADO, 633 Command objects, 634 Connection objects, 634 Error objects, 634 Field objects, 635 Parameter objects, 634 Property objects, 634 Recordset objects, 634 sample application, 635-637 applications, converting existing OLE servers, 569 ATL advantages, 538-539 building controls, 539 drawing functions, 546-547 event handling, 551-557 methods, 543-546 Object Wizard, 542-543 properties, 543-546 property pages, 547-550 Proxy Generator, 552 skeleton project, 540-542

controls, 322-323 bitmaps, 524 code overview, 515-523 containers, 323 creating, 512-513 distributing, 534 drawing, 530-531, 546-547 event handling, 550-557 events, 528-529 inserting, 323-324 member variables, 326-328 message handling, 329 methods, 528, 545-546 properties, 325-329, 524-528, 543-545 property page interfaces, 531-533, 547-550 skeleton controls, 513-515 testing, 534 usage tips, 535 ControlWizard, 513-515 documents, 560 containers, 562-567 servers, 567-569 containers, 562 interfaces, 561 servers, 561 events, 329-330 editing, 26 ActiveX Control Test Container, 556 ActiveX Control Text Container, 11 ActiveX Data Objects, see ADO ActiveX Template Library, see ATL ADAO application, 608 classes, CDAOSet, 613-615 customizing, 617-619

APIs (application programming interfaces) INDEX database, 609-612 building, 609 tables, 610 DFX functions, 615-616 skeleton application, 612-613 Add Member Variable dialog box, 23, 277 Add Method dialog box, 501 Add Property to Interface dialog box, 543 adding components to Component Gallery, 28 Adding a Class dialog box, 275 address books, MAPI, 668 Address Resolution Protocol (ARP), 681 address spaces, 158-159 Compatibility Arena, 159 reserved system arenas, 159 shared arenas, 159 addresses calculating, 32-bit applications, 163 host addresses, 683 logical, 158 network addresses, 683 physical, 158 TAPI, 714-715 canonical address format, 715 dialable addresses, 715 AddSegmentAtMessagePos function, 60 AddView function, 260 ADO (ActiveX Data Objects), 626, 633-635 sample application, 635-637 Advanced Options dialog box (MFC AppWizard), 16 afxdocob.h, 569 AfxEnableControlContainer function, 323 aggregate functions, 592 Aldus placeable metafiles, 853 Alias button (Microsoft Help Workshop), 817 allocating memory, 164 COM, 419-420 malloc, 164 ODBC, 585 shared memory, 165 stray pointers, 165 virtual, 167 ODBC environments, 584 AllocConsole function, 190 ALTER TABLE statements, 593 AMFC application, 602 AngleArc function, 135 AnimatePalette function, 126 animation DirectX Direct3D, 788 resource files, 798 sample application, 792-793, 797-798 palette animation, 126-129 animation controls, 93, 306 ANISOTROPIC mapping mode, 115, 340 anonymous pipes creating, 732 data transfers, 734 ANSI character set, nonsupported languages, 870 antimonikers, 422 apartment model (threads), 422 APIs (application programming interfaces), 46 COM API, 46 DirectX, 785 Direct3D, 788 DirectAnimation, 791 DirectDraw, 786-788 DirectInput, 791 DirectPlay, 790-791 DirectSetup, 792 DirectShow, 791 DirectSound, 788-790 function calls, 42 MAPI, 46, 670 applications, 668 architecture, 666-667 CMC, 673-674 Extended MAPI, 675 MFC support, 677-678 OLE messaging, 675-677 profiles, 669 service providers, 668-669 Simple MAPI, 670-673 ODBC, 583-586 TAPI, 46, 712, 719 addresses, 714-715 architecture, 716-719 Assisted TAPI, 712-713 Basic Telephony, 719 data communication program, 723-728 devices, 714 Extended TAPI, 720 media modes, 722 multiple applications, 722-723 programming model, 720-721 services, 713 Supplementary Telephony, 720 Win32 API, NT services, 641-642 WinInet, 700, 705-706 asynchronous mode, 709

893

894

APIs (application programming interfaces) UNLEASHED cache management functions, 709 FTP sessions, 706-707 Gopher sessions, 707-708 HTTP sessions, 708-709 WinSock, 685-686 asynchronous socket functions, 690 blocking, 689 byte ordering, 687 initialization, 686 name service, 687 sample application, 690-692 socket communication, 687-689 sockets, 686 Appbars, 848 APPINIT_DLLS Registry key, 79 Application Architecture classes, 237 Application Configuration files, 738 Application Desktop Toolbars (Appbars), 848 application global classes, 69 application programming interfaces, see APIs applications 32-bit, 161 address calculations, 163 integer size, 162 library functions, 163 memory models, 163 selector functions, 163 type modifiers, 162 ActiveX controls adding, 542-543 inserting, 323 ADAO, 608 CDAOSet class, 613-615 customizing, 617-619 database, 609-612 DFX functions, 615-616 skeleton application, 612-613 AppWizard, 10 classes, adding, 27 client/server, 660 clipboard viewers, 198-199 clipboard-aware header file, 202 resource file, 202 source code, 199-202 COM automation server, 426-427 accessing, 436-437 functions, 427 Regstry entry, 435-436 source code, 428-435 data transfers, 423-424 DCOM, DCOMCFG.EXE, 574 debugging, preparation, 29-30 dialog-based, 15, 270-271 creating, 19 DirectX, 792, 797-799 DLLs, creating, 20 exception handling, 398 exceptions CArchiveException class, 402 CDaoException class, 404 CDBException class, 404 CException class, 399-400 CFileException class, 401-402 CInternetException class, 404-405 CMemoryException class, 401 CNotSupportedException, 403 COleDispatchException, 405 COleException class, 405 CResourceException class, 403 CUserException class, 405 exception-handlng macros, 398-399 raising, 224 exiting, 508 FTP sample application, 706 generic Windows application, 56-57 gettime MFC, 692-694 WinSock, 690-692 Gopher sample application, 707 guidgen.exe, 420, 739 Hello, World, 50-51, 238 application objects, 239-243 AppWizard options, 239 compiling, 50 customizing, 253-255 document objects, 248-249 executing, 50 frame window, 245-246 graphics version, 58-60 message loops, 51-53 message map, 243-245 resources, 252-253 source code, 50 string resources, 253 view window, 250-252 window procedures, 53-56 help files, 804-805 AppWizard-generated, 822 compiling, 811 developing, 805-806

asynchronous operations, TAPI INDEX DLLs, 812 footnotes, 807 hyperlinks, 807 invoking, 812-813 macros, 811-812 Microsoft Help Workshop, 813-814, 818-821 project files, 809-810 RTF files, 806-809 tables of contents, 810-811 topics, 806-809 HTTP sample application, 708-709 installation programs creating, 831-835 customizing, 839-840 debugging, 845 distributing, 842-844 file groups, 835-838 InstallShield, 830 requirements, 828-830 setup scripts, 840-841 testing, 844 MAPI support, 668 MDI classes, 18-19 creating, 15-19 database support, 15 OLE support, 15 project files, 15 memory, sharing, 165, 171-173 MFC, 236 multimedia playback, 750-751 NT services architecture, 643-644 compiling, 651 creating, 642-651 CSlot class, 649-651 CSrv class, 647-649 installing, 651 main service module, 644-647 removing, 651 running, 651 SDK Service example, 642-643 starting, 641 stopping, 641 Win32 API, 641-642 objects, 507-508 OCON containers, 456-475 drag sources, 483-484 drop targets, 484-491 positioning support, 479-481 selection support, 482 serialization, 480 Registry reader program, 214-218 SDI classes, 18-19 creating, 15-19 database support, 15 OLE support, 15 project files, 15 Spy++, 36 TAPI data communication program, 723-728 tapiexe.exe, 716 threads, 36 uuidgen.exe, 420 AppWizard help files, generating, 822 MAPI support, 677 MFC, 14 OLE server applications, creating, 441-442 project templates, 10 toolbar bitmaps, 554 APSTUDIO_INVOKED symbol, 100 Arc function, 134 arcs, drawing, 134 ArcTo function, 134 ARP (Address Resolution Protocol), 681 array collections CPtrArray class, 374 CStringArray class, 375 CUIntArray class, 374 arrays collection classes CObArray, 371-373 CPtrArray, 374 CStringArray, 375 CUIntArray, 374 integral array classes, 374-375 creating, 371 elements, adding/ removing, 372 sizing, 372 sparce matrices, 167 ARSV server application skeleton, 495 automation support, 499-503 calculator, 495-499 standard objects application objects, 507-508 document objects, 508-509 methods, 506 properties, 507 testing, 504-505 type library, 503-504 user interface, 494-495 ASCII character set, SBCS, 873 Assisted TAPI, 712-713 associating sockets, 686 asynchronous I/O operations, 181-186 asynchronous mode, WinInit, 709 asynchronous operations, TAPI, 716-717

895

896

asynchronous socket UNLEASHED asynchronous socket functions, 690 ATL (ActiveX Template Library), 538 ActiveX controls adding to projects, 542-543 bitmaps, adding, 554-555 building, 539 drawing, 546-547 methods, 545-546 properties, 543-545 property pages, 547-550 skeleton project, 540-542 testing, 556 advantages, 538-539 event handling, 550-554, 557 Proxy Generator, 552 atomic transactions, 660 Attach member function, 357 attaching, CDC objects to device contexts, 335 attributes, device contexts, 338 audio ACM, 764 DirectX, 784-785 COM interfaces, 785-786 DirectPlay, 790-791 DirectSetup, 792 DirectSound, 788-790 sample application, 792, 797-799 formats AVI, 753 MIDI, 753 WAV, 752 mixer devices, 765 playback, 752 PlaySound function, 764 Audio Compression Manager (ACM), 764 Audio Video Interleaved (AVI) format, 753 authenticating, installation programs, 829 Author property (document objects), 508 automatic events, 409 automation (OLE), 425-426, 494 properties, editing, 25 servers, 494 application skeleton, 495 calculators, 495-499 standard objects, 506-509 supporting, 499-503 testing, 504-505 type libraries, 503-504 user interfaces, 494-495 AVI (Audio Video Interleaved) format, 753 AVIFile functions, 763 AVIStream functions, 763 Binder, ActiveX documents, 560 binding dynamic, 621 handles, 739 BitBlt function, 136, 343 BITMAP clipboard format, 195 Bitmap Image object, 458 BITMAP statement, 100 BITMAP structure, 348 bitmaps, 129, 135-136 ActiveX controls, 524 adding, 554-555 CDC class functions, 343-344 color depth, 129 copying, 136 creating, 129 DIBs, 129 creating, 136 filling, 343 loading, 129 MFC support, 348 sizing, 348 Bitmaps button (Microsoft Help Workshop), 815 blocking calls, 181, 689 bmc command, 808 bouncing ball application (DirectX), 792-799 breaking pipe connections, 734 breakpoints, 9 Briefcase, reconcilers, 850 brushes, 123-124 MFC support, 347-348 buffers, clearing, 191 bugs, debugging, 29-30 Build menu commands, Settings, 29 Build Type dialog box, 843 BUTTON class subclassing, 77-79 superclassing, 80-81

B
based pointers, 173 Basic Telephony service, 719 BEGIN-END keyword pair, 101 BeginPaint function, 71 BeginPath function, 137, 345 Bzier curves, drawing, e 134 big-endian byte ordering, 687 BINARY Registry value type, 207 bind function, 686

CDC class INDEX buttons, 92 controls, defining, 103 Microsoft Help Workshop Alias, 817 Bitmaps, 815 Config, 817 Data Files, 818 Edit Files, 819 Files, 813 Link Files, 819 Map, 816 Options, 813 Save and Compile, 818 Tabs, 819 Windows, 814 byte ordering, 687 call control application (TAPI), 713 call monitors, 722 call ownership, 722 CallNamedPipe function, 734 CAMFCSet class declaration, 598-599 implementation, 599-600 CancelToClose function, 291 CAnimateCtrl class, 306 canonical address format, 715 CAPTION option (DIALOG statement), 102 Caption property (application objects), 507 CArchive class error handling, 361 objects, 238 creating, 358-359 reading, 359-360 writing to, 359-360 operators, 359-360 sample application, 361-362 CArchiveException class, 402 CArray class, 382-383 case sensitivity, Registry keynames, 206 CASRVDoc class declaration, 502-503 header file, 501 CAsynchSocket class sample application, 692-694 synchronous operations, 694 catchall function, 230 catching exceptions ellipses handler, 228-230 filter expressions, 221-224 floating point, 223 guard blocks, 221 nested handlers, 222-223 sample program listing, 221 CBitmap class, 348 CBrush class, 347 CClientDC class, 236, 337 CCmdTarget class, 265-266 CCmdUI class, 412 CColorDialog class, 238, 299 CCommandLineInfo class, 412 CCommonDialog class, 298-299 CCreateContext class, 412 CCriticalSection class, 409 CD-ROMs, file systems, 179 CDaoDatabase class, 621 CDaoException class, 404, 622 CDaoFieldExchange class, 622 CDaoQueryDef class, 622 CDaoRecordset class, 620-621 CDAOSet class declaration, 613-615 implementation, 615 CDaoTableDef class, 622 CDaoWorkspace class, 621-622 CDatabase class, 603 CDataExchange class, 412 CDBException class, 404 CDC class, 335 attributes, 338 clipping operations, 345 coordinate mapping, 339-341 drawing functions, 341 bitmaps, 343-344 lines/shapes, 342-343 text, 344-345

897

C
C applications, OpenL, 775-776 exceptions, 220-224 translating, 230-231 keywords, volatile, 151 OpenGL applications, 772-775 compiling, 776 initialization, 775 window procedures, 775-776 stream I/O, 188 termination handling, 224-226 C++, compiling Unicode applications, 875 cache, management, WinInet functions, 709 CADCCntrItem class, 562 calculating addresses, 32-bit applications, 163 calculators (automation servers), 495-499

898

CDC class UNLEASHED objects addresses, 336 attaching, 336 path functions, 345-346 printing functions, 345 CDialog class, 274 CDialog-derived class, 19 CDocItem class, 265-266 CDocObjectServerItem class, 569 CDocument class data elements, 262-264 declaration, 258-259 MAPI support, 677 member functions, 259-260 message handling, 261 overridable functions, 261-262 CEvent class, 409 CException class, 399-400 CFieldExchange class, 604 CFile class error handling, 355 file locking, 355 file management, 354 hierarchy, 352 initialization, 354 objects associating, 356 reading to, 354 sample console application, 355-356 subclasses CInternetFile, 356 CMemFile, 356-357 COleStreamFile, 357 CSocketFile, 358 CStdioFile, 356 CFileDialog class, 299-300 CFileException class, 401-402 CFileStatus structure, 412 CFindReplaceDialog class, 300-301 CFont class, 348 CFontDialog class, 302 CFtpConnection class, 390 CGdiObject class, 346 CGopherConnection class, 390-391 CGopherLocator class, 392 change notification objects, 153 character sets, international applications, 872-873 CHeaderCtrl class, 307 check boxes, 92 child windows, 64 Choose Color common dialog, 85 Choose Dialogs dialog box (Project Wizard), 831 Choose Target Platform dialog box, 832 ChooseColor function, 85 CHOOSEFONT structure, 86 ChoosePixelFormat function, 775 CHotKeyCtrl class, 307-308 CHttpConnection class, 391 CInPlaceFrame class, 442 CInternetException class, 392, 404-405 CInternetFile class, 356, 391 CInternetSession class, 389 CLASS option (DIALOG statement), 102 classes adding to applications, 27 CAMFCSet declaration, 598-599 implementation, 599-600 CAnimateCtrl, 306 CArchive creating objects, 358-359 error handling, 361 overloaded operators, 359-360 reading objects, 359-360 sample application, 361-362 writing to objects, 359-360 CArchiveException, 402 CArray, 382-383 CASRVDoc declaration, 502-503 header file, 501 CAsynchSocket sample application, 692-694 synchronous operations, 694 CBitmap, 348 CBrush, 347 CClientDC, 337 CCmdTarget, 265-266 CCmdUI, 412 CColorDialog, 238, 299 CCommandLineInfo, 412 CCommonDialog, 298-299 CCreateContext, 412 CCriticalSection, 409 CDaoDatabase, 621 CDaoException, 404, 622 CDaoFieldExchange, 622 CDaoQueryDef, 622 CDaoRecordset, 620-621 CDAOSet declaration, 613-615 implementation, 615 CDaoTableDef, 622 CDaoWorkspace, 621-622 CDatabase, 603 CDataExchange, 412 CDBException, 404 CDC, 335 attributes, 338 clipping operations, 345

classes INDEX coordinate mapping, 339-341 drawing functions, 341-345 path functions, 345-346 printing operations, 345 CDialog, 274 CDocItem, 265-266 CDocObjectServerItem, 569 CDocument data elements, 262-264 declaration, 258-259 MAPI support, 677 member functions, 259-260 message handling, 261 overridable functions, 261-262 CEvent, 409 CException, 399-400 CFieldExchange, 604 CFile error handling, 355 file locking, 355 file management, 354 hierarchy, 352 initialization, 354 object associations, 356 reading from/writing to objects, 354 sample console application, 355-356 CFileDialog, 299-300 CFileException, 401-402 CFindReplaceDialog, 300-301 CFont, 348 CFontDialog, 302 CGdiObject, 346 CHeaderCtrl, 307 CHotKeyCtrl, 307-308 CInPlaceFrame, 442 CInternetException, 392, 404-405 CInternetFile, 356 CList, 381-382 CListCtrl, 309-310 CListView, 309 CLSIDs, 421 CMainFrame declaration, 245-246 implementation, 246 CMap, 383-384 CMapPtrToPtr, 378 CMapPtrToWord, 379 CMapStringToOb, 377-378 CMapStringToPtr, 378 CMapStringToString, 375-377 CMapWordToOb, 379 CMapWordToPtr, 379 CMCTL declaring, 327-328 implementing, 328-329 CMCTLApp declaration, 516 implementation, 516-517 CMCTLCtrl declaration, 519-520 implementation, 521-523 CMCTLPropPage declaration, 517-518 implementation, 518-519 CMemFile, 356-357 CMemoryException, 401 CMemoryState, 412 CMetaFileDC, 337-338 CMultiLock, 410 CMutex, 409 CMyDialog declaration, 278 member functions, 278-279 CMySheet declaration, 292-293 implementation, 293-294 CNotSupportedException, 403 CObArray, 371-373 CObject, 237-238 CObList, 369-371 adding elements to, 369 creating, 369 finding elements, 370 removing elements from, 370 COCONCntrItem, 460, 479 COleClientItem, 478 COleDialog, 304 COleDispatchException, 405 COleDocument, 444 COleException, 405 COleServerItem, 442-443 COleStreamFile, 357 collection classes, standard properties, 507 COSRVSrvrItem, 442 CPageSetupDialog, 302-303 CPaintDC, 336-337 CPen, 347 CPoint, 410 CPrintDialog, 303-304 CPrintInfo, 412 CProgressCtrl, 311 CPropertyPage, member functions, 291-292 CPtrArray, 374 CPtrList, 373 CRecordset, 603-604 CRecordView, 605 CRect, 410 CRectTracker, 412 CResourceException, 403 CRgn, 349 CRichTextCtrl, 311-312

899

900

classes UNLEASHED CSemaphore, 409 CSingleLock, 410 CSize, 411 CSliderCtrl, 312-313 CSlot declaring, 649-650 implementing, 650-651 CSocketFile, 358 CSpinButtonCtrl, 313 CSrv declaring, 647-648 implementing, 648-649 CStatusBarCtrl, 313-314 CStdioFile, 356 CString, 411 CStringArray, 375 CStringList, 373-374 CSyncObject, 408 CTabCtrl, 314-315 CTime, 411 CToolBarCtrl, 315-316 CToolTipCtrl, 316 CTreeCtrl, 316-317 CTypePtrArray, 385 CTypePtrList, 384 CTypePtrMap, 385 CUIntArray, 374 CUserException, 405 CView, 266 declaration, 267-268 derived classes, 270 member functions, 268-269 message handling, 269 CWaitCursor, 413 CWindowDC, 337 CYAHApp declaring, 240 implementing, 241-243 CYAHDoc declaration, 248 implementation, 249 CYAHView declaration, 250 implementation, 251-252 dialog classes, constructing, 275-277 exception, 228 Internet support architecture, 388-389 CFtpConnection, 390 CGopherConnection, 390-391 CHttpConnection, 391 CInternetException, 392 CInternetFile, 391 CInternetSession, 389 iostream, 188-189 MCIWnd functions, 755-756 macros, 756-758 messages, 756-758 notification messages, 758-759 user interface, 754 MFC, 236 MyPage1 declaration, 289-290 implementation, 290 properties, editing, 26 registering, 421 window global subclassing, 78-79 subclassing, 75-77 superclassing, 80-81 window procedure, 74-75 window classes, 38 Classes subkey (HKEY LOCAL MACHINE), 210 ClassWizard ActiveX Events tab, 26 Automation tab, 25, 499-501 Class Info tab, 26 dialog classes, constructing, 275-277 functions, 27 invoking, 20 Member Variables tab, 22-25 Message Maps tab, 21-22 ClearCommBreak function, 191 ClearCommError function, 191 clearing, 191 client tier, 658 client workstations, configuring for DCOM servers, 572 client-area device contexts, 337 clients named pipe clients, 735-736 RPC client application, 742-743 three-tiered client/server model, 657 clip paths, 130-132 Clipboard (Windows) formats private, 196 registered, 196 standard, 194-196 messages, 198 operations controls, 198 data transfers, 196-197 delayed rendering, 197 pasting data, 197-198 sample program, 199-203 header file, 202 resource file, 202 source code, 199-202 SetClipboardData function, 194 user services, 44 viewers, 198-199 CLIPBRD.EXE program, 198 clipping, 46, 129-132, 345 clip paths, 130-132

color INDEX clipping regions, 129-130 CList class, 381-382 CListCtrl class, 309-310 CListView class, 309 Close All Help command (Test menu), 821 Close function, 354, 509 CLOSE window mangement message, 72 CloseEnhMetaFile function, 112 CloseHandle function, 180 CloseMetaFile function, 112 closing documents, 509 file objects, 180 files, 354 metafiles, 112 CLSID subkey (HKEY CLASSES ROOT), 211 CLSIDs (class IDs), 421 CMainFrame class declaration, 245-246 implementation, 246 CMap class, 383-384 CMapPtrToPtr class, 378 CMapPtrToWord class, 379 CMapStringToOb class, 377-378 CMapStringToPtr class, 378 CMapStringToString class, 375-377 CMapWordToOb class, 379 CMapWordToPtr class, 379 CMC (Common Messaging Calls) API, 667 console application, 674 MAPI, compared, 673 CMCTL class declaring, 327-328 implementing, 328-329 CMCTLApp class declaration, 516 implementation, 516-517 CMCTLCtrl class declaration, 519-520 implementation, 521-523 CMCTLPropPage class declaration, 517-518 implementation, 518-519 CMemFile class, 356-357 CMemoryException class, 401 CMemoryState class, 412 CMetaFileDC class, 337-338 CMultiLock class, 410 CMutex class, 409 CMyDialog class declaration, 278 member functions, 278-279 CMySheet class declaration, 292-293 implementation, 293-294 CNotSupportedException class, 403 CObArray class, 371-373 CObject class, 237-238, 368 CObArray, 371-373 CObList, 369-371 CObList class, 369-371 creating, 369 elements adding, 369 finding, 370 removing, 370 COCONCntrItem class, 460, 479 CoCreateGuid function, 420 code, ClassWizard, 27 code pages character sets, 873 SBCS/MBCS, 873 CoGetMalloc function, 420 COleClientItem class, 478 COleDialog class, 304 COleDispatchException class, 405 COleDocument class, 444 COleException class, 405 COleServerItem class, 442-443 COleStreamFile class, 357 collection classes CObArray, 371-373 CObject, 368 CObList, 369-371 CPtrArray, 374 CPtrList, 373 CStringArray, 375 CStringList, 373-374 CUIntArray, 374 helper functions, 380-381 integral array classes, 374-375 mappings CMapPtrToPtr class, 378 CMapPtrToWord class, 379 CMapStringToOb class, 377-378 CMapStringToPtr, 378 CMapStringToString, 375-377 CMapWordToOb class, 379 CMapWordToPtr class, 379 standard properties, 507 templates, 379-380 CArray, 382-383 CList, 381-382 CMap, 383-384 CTypePtrArray, 385 CTypePtrList, 384 CTypePtrMap, 385 color color selection common dialog, 299 depth, 129

901

902

color UNLEASHED palettes, 125 text, 344 COM (Component Object Model), 46, 418, 572 automation server application, 426-427 accessing, 436-437 functions, 427 Registry entries, 435-436 source code, 428-435 class objects, 421 DirectX interfaces, 785-786 inheritance, 420 inter-object communication, 421-422 interfaces, 418-419 C++ classes, 786 definition, 420-421 IClassFactory, 421 identifiers, 420 IDispatch, 425 IDropSource, 426 IDropTarget, 426 IMalloc, 420 IPersistStorage, 423 IRootStorage, 423 IStorage, 423 IStream, 423 IUnknown, 420 MTS API, 661 memory allocation, 419-420 monikers, 422 MTS, 660 Object Viewer, 11 objects, file viewers, 849 COM+, 577 CombineRgn function, 135 CombineTransform function, 120 combining linear transformations, 119-120 regions, 135 combo boxes, 93 COMBOBOX control statement, 103 Command objects (ADO), 634 command objects, OLE DB, 627 command-line tools, compiler, 4-5 commands bmc, 808 Build menu, Settings, 29 File menu Help Author, 820 New, 13 Insert menu New ATL Object, 542 Object, 447 MCI command strings sets, 762 syntax, 761-762 pict, 809 Spy menu, Find Window, 36 Test menu, 821 ul, 807 v, 807 CommConfigDialog function, 191 CommDlgExtendedError function, 299 comments, preprocessing, 99 Comments property (document objects), 508 commit function, 188 committed pages, 166 common control functions, 46 common controls, 305 animation, 306 header controls, 307 hotkey controls, 307-308 list controls, 309-310 progress bars, 311 RTF control, 311-312 slider controls, 312-313 spin buttons, 313 status bars, 313-314 tab controls, 314-315 toolbars, 315-316 tooltips, 316 tree controls, 316-317 common device contexts, 111 common dialogs, 46 Choose Color, 85 File Open, 84 File Save As, 84-85 Find, 88 Font Selection, 86 MFC classes, 298-299 CColorDialog, 299 CFileDialog, 299-300 CFindReplaceDialog, 300-301 CFontDialog, 302 COleDialog, 304 CPageSetupDialog, 302-303 CPrintDialog, 303-304 Page Setup, 88 Print, 87-88 Replace, 88 sample program listing, 89-91 Common Messaging Calls (CMCs) , 667, 673-674 communication ports, 191 communications programs multiple threads, 182-183 overlapped I/O, 184-186 CompactDatabase function, 622 compacting databases, 622 heaps, 170 CompareElements function, 380

controls INDEX comparing, CRgn objects, 349 Compatibility Arena, 159 compiler, 4 COM+, 577 command-line tools, 4 debugger, 8 EXEs/DLLs, 6-7 resources, 5 startup code, 6 compilers hcrtf.exe, 811 MIDL, 737, 744-745 compiling applications, DirectX, 799 help files, 811 OpenGL applications, 776 resource files, 107 Unicode applications, 874 Windows NT services, 651 Component Gallery, 11 adding components, 28 Component Object Model, see COM components, 11, 28 Components and Controls Gallery dialog box, 28 composite monikers, 422 compound documents, 422 data transfers, 423-424 in-place activation, 425 objects, 424 structured storage, 423 compression ACM, 764 NT support, 179 VCM (Video Compression Manager), 764 Windows 95 support, 179 Config button (Microsoft Help Workshop), 817 Config subkey (HKEY LOCAL MACHINE), 209 connecting to named pipes, 733-734 Connection objects, (ADO), 634 ConnectNamedPipe function, 734 console applications, 43 unicode, 874 console I/O, 189-190 console input objects, 153 constants, Unicode, 874 constrained mapping modes, 114-115 ConstructElements function, 380 constructing dialogs, 274 classes, 275-277 CMyDialogClass, 278-279 member variables, 277-278 modeless, 281-282 templates, 275 objects CArchive, 358 CFile, 354 COleStreamFile, 357 property sheets member functions, 291-292 modal, 291 modeless, 292-294 pages, 287-290 containers, 425, 456, 475 ActiveX controls, 323, 562-567 drag-and-drop application, 478-479 positioning support, 479-481 selection support, 482 embedded objects, editing, 474 skeleton application, 456-457 customizing, 468-474 drawng functions, 471 member functions, 460-466 menus, 466-467 object positions, 469-471 object selection, 472-474 running, 459-460 view class, 464-466 context switching, 141 context-sensitive help, 804-805 AppWizard-generated files, 822 developing, 805-806 DLLs, 812 files, compiling, 811 footnotes, 807 Help Workshop (Microsoft), 820 contents files, 818-820 file testing, 820-821 projects, 813-814, 818 hyperlinks, 807 invoking, 812-813 macros, 811-812 Microsoft Help Workshop, 813 project files, 809-810 RTF files, 806-809 tables of contents, 810-811 topics, 806-809 CONTROL control statement, 103 control data types, Dialog Data Exchange, 285 control developers, 322 control statements, 103 control users, 322 controlfp function, 223 controls, 91 ActiveX, 322-323, 426, 512 adding to projects, 542-543 ATL, 538-539

903

904

controls UNLEASHED bitmaps, 524, 554-555 code overview, 515-520, 523 containers, 323 creating, 512-513 distributing, 534 drawing, 530-531, 546-547 event handling, 550-554, 557 events, 329-330, 528-529 inserting, 323-324 member variables, 326-328 message handling, 329 methods, 528, 545-546 properties, 325-329, 524-528, 543-545 property page interfaces, 547-550 property pages, 531-533 skeleton applications, 540-542 skeleton controls, 513-515 testing, 534, 556 usage tips, 535 button controls, defining, 103 buttons, 92 clipboard support, 198 combo boxes, 93 date time picker, 307 Dialog Data Exchange support, 284-285 edit, 92 IP address, 308 list boxes, 92 MFC classes, 305 CAnimateCtrl, 306 CHeaderCtrl, 307 CHotKeyCtrl, 307-308 CListCtrl, 309-310 CProgressCtrl, 311 CRichEditCtrl, 311-312 CSliderCtrl, 312-313 CSpinButtonCtrl, 313 CStatusBarCtrl, 313-314 CTabCtrl, 314-315 CToolBarCtrl, 315-316 CToolTipCtrl, 316 CTreeCtrl, 316-317 month calendar, 310 scrollbars, 93 static, 92 Windows 95 common controls, 93-94 ControlWizard, ActiveX, 513-515 converting dialog units to pixels, 102 cooperative multitasking, 141-142, 153 lengthy processing, 145-148 message loops, 145 processes, 151-152 synch objects (s/b synchronization) mutexes, 153 sample listing, 153-154 semaphores, 152 variable access, 153 synchronization objects, 152-153 threads secondary, 148-151 thread objects, 151 coordinates logical, 45 constrained mapping, 114-115 unconstrained mapping, 113-114 World Coordinate Tranforms, 115-122 mapping, 339-341 modes, 339 physical, 113 world coordinates, 120 CopyData function, 203 CopyElements function, 381 copying bitmaps, 136 files, 180-181 Core API (ODBC), 584 core grammar (ODBC), 584 COSRVSrvrItem class, 442 CountClipboardFormats function, 198 CPageSetupDialog class, 302-303 CPaintDC class, 336-337 CPoint class, 410 CPrintDialog class, 303-304 CPrintInfo class, 412 CProgressCtrl class, 311 CPropertyPage class, member functions, 291-292 CPtrArray class, 374 CPtrList class, 373 CPUIntArray class, 374 create function, 187 CREATE INDEX statement, 593 Create New Data Source dialog box, 583 CREATE TABLE statements, 593 CREATE VIEW statements, 592 CREATE window mangement message, 72 CreateBitmap function, 129 CreateBrushIndirect function, 123 CreateCompatibleBitmap function, 129 CreateCompatibleDC function, 112, 336 CreateDC function, 111, 335

CYAHView class INDEX CreateDialog function, 82 CreateDIBitmap function, 136 CreateDIBPatternBrushPt function, 124 CreateEnhMetaFile function, 112 CreateEvent function, 152 CreateFile function, 180 CreateFont function, 124 CreateFontIndirect function, 124 CreateHatchBrush function, 124 CreateIC function, 112 CreateMetaFile function, 112 CreateNamedPipe function, 732 CreatePatternBrush function, 124 CreatePen function, 123 CreatePenIndirect function, 123 CreatePipe function, 732 CreateProcess function, 152 CreateRemoteThread function, 79 CreateSemaphore function, 152 CreateSolidBrush function, 124 CreateStockObject function, 347 CreateSurface method, 787 CreateThread function, 151 CreateView method, 561 CreateWindow function, 55, 69-70 CreateWindowEx function, 70 CRecordset class, 603-604 CRecordView class, 605 CRect class, 410 CRectTracker class, 412 CResourceException class, 403 CRgn class, 349 CRichTextCtrl class, 311-312 critical section objects, 153, 409 CSemaphore class, 409 CSingleLock class, 410 CSize class, 411 CSliderCtrl class, 312-313 CSlot class declaring, 649-650 implementing, 650-651 CSocketFile class, 358 CSpinButtonCtrl class, 313 CSrv class declaring, 647-648 implementing, 648-649 CStatusBarCtrl class, 313-314 CStdioFile class, 356 CString class, 411 CStringArray class, 375 CStringList class, 373-374 CSyncObject class, 408 CTabCtrl class, 314-315 CTEXT control statement, 103 CTime class, 411 CToolBarCtrl class, 315-316 CToolTipCtrl class, 316 CTreeCtrl class, 316-317 CTypePtrArray class, 385 CTypePtrList class, 384 CTypePtrMap class, 385 cubes, drawing, 778-780 currency, language-specific formats, 869 CURSOR statement, 100 curves, drawing, 134-135 CUserException class, 405 custom data types, implementing, 285 custom interfaces, 426 custom stream handlers, 763 customizing ActiveX controls bitmaps, 524 events, 528-529 methods, 528 properties, 524-528 DAO applications, 617-619 installation programs, 839-840 interfaces, 426 ODBC applications, 601-602 OLE containers, 468-474 drawing functions, 471 object positions, 469-471 object selection, 472-474 server skeleton application, 447-451 dialogs, 450-451 documnts, 448 drawing code, 448-450 Windows messages, 40 CView class, 266 declaration, 267-268 derived classes, 270 member functions, 268-269 message handling, 269 CWaitCursor class, 413 CWindowDC class, 337 CYAHApp class declaring, 240 implementing, 241-243 CYAHDoc class declaration, 248 implementation, 249 CYAHView class declaration, 250 implementation, 251-252

905

906

DAOs (Database Access Objects) UNLEASHED

D
DAOs (Database Access Objects), 608 classes, 619-620 CDaoDatabase, 621 CDaoException, 622 CDaoFieldExchange, 622 CDaoQueryDef, 622 CDaoRecordset, 620-621 CDaoTableDef, 622 CDaoWorkspace, 621-622 overview, 608 sample application, 608 customizing, 617-619 database, 609-612 skeleton application, 612-616 data pasting from Clipboard, 197-198 transferring between applications, 423-424 pipes, 734 to Clipboard, 196-197 validating, Dialog Data Exchange, 283 data communication program, 723-728 data definition statements (SQL), 593 ALTER TABLE, 593 CREATE INDEX, 593 CREATE TABLE, 593 DROP, 593 GRANT, 593 REVOKE, 593 Data Files button (Microsoft Help Workshop), 818

data manipulation statements (SQL), 591-592 CREATE VIEW, 592 DELETE, 592 INSERT, 592 SELECT, 591-592 UPDATE, 592 data source objects, 627 data sources, 582 application skeletons, 597 specifying, 612 MFC ODBC applications, 594-596 data types, custom, implementing, 285 Database Access Objects, see DAOs Database Options dialog box, 612 databases ADO, 633 Command objects, 634 Connection objects, 634 Error objects, 634 Field objects, 635 Parameter objects, 634 Property objects, 634 Recordset objects, 634 sample application, 635-637 compacting, 622 connections, 621 DAOs, 608-612 building, 609 classes, 619-622 customizing, 617-619 overview, 608 skeleton application, 612-616 tables, 610 data sources, specifying, 612 ODBC, 582 connections, 585

environment allocation, 584 functions, 589-590 memory allocation, 585 MFC applications, 593-601 MFC classes, 602-605 PAIs, 583-586 RFX functions, 604 sample console application, 586-589 setup applet, 582-583 OLE DB, 626 commands, 627 data source objects, 627 enumerators, 628 errors, 628 rowsets, 627 sample application, 628, 632 SDK, 627 sessions, 627 transactions, 628 queries, definitions, 622 recordsets, 620-621 attributes, 604 creating, 620 dynasets, 603 navigating, 604, 620 snapshots, 603 referencing, ADO application, 635-637 repairing, 622 sessions, 621-622 SQL data definition statements, 593 data manipulation statements, 591-592 views, 592 tables, definitions, 622 datagrams (IP), 682 date time picker control, 307

dialog boxes INDEX dates, language-specific formats, 869 DCOM, 572 COM+, 577 DCOM servers client workstations, configuring, 572 coding practices, 575 DCOMCNFG.EXE, 573 DCOMCNFG.EXE, 573 DDP_ functions, 532-533 ddraw.dll, 786 debugger, 8 compiler settings, 29-30 enabling, 29 linker settings, 30 debugging installation programs, 845 preparation, 29-30 declaring classes CAMFCSet, 598-599 CASRVDoc, 502-503 CDAOSet, 613-615 CDocument, 258-259 CMainFrame, 245-246 CMCTL, 327-328 CMCTLApp, 516 CMCTLCtrl, 519-520 CMCTLPropPage, 517-518 CMyDialog, 278 CMySheet, 292-293 CSlot, 649-650 CSrv, 647-648 CView, 267-268 CYAHApp, 240 CYAHDoc, 248 CYAHView, 250 MyPage1, 289-290 default palettes, 125 default routes, 683 Default subkey (HKEY USERS), 211 DefaultFilePath property (application objects), 507 DefDlgProc function, 56, 75 defining button controls, 103 COM interfaces, 420-421 window classes, 38 Windows messages, 40 DefWindowProc function, 56, 75 delayed rendering, clipboard, 197 DELETE statements, 592 DeleteContents function, 262 DeleteDC function, 112, 336 DeleteObject function, 122-125, 347 deleting information contexts, 112 palettes, 125 Registry keys, 214 design mode, 325 desktop window, 64 DESTROY window mangement message, 72 DESTROYCLIPBOARD messages, 198 DestroyWindow function, 52-53 DestructElements function, 380 Detach member function, 357 Developer Network, foreign-language glossaries, 878 developing help files, 805-806 device contexts, 44, 111, 334 attributes, 338 CDC class, 335 client-area, 337 creating, 111, 335-336 handles, retrieving, 111 metafile, 337-338 paint-time, 336-337 scrolling, 344 types, 111-112 window, 337 device control block structure, 191 device drivers, 110, 137 device-independent bitmaps (DIBs), 129 DFX functions, 615-616 dial modifier characters, 715 dialable address (TAPI), 715 dialog box procedure, 83 dialog boxes, 81 Add Member Variable, 23, 277 Add Method, 501 Add Property to Interface, 543 Adding a Class, 275 Advanced Options (MFC AppWizard), 16 common dialogs Choose Color, 85 File Open, 84 File Save As, 84-85 Find, 88 Font Selection, 86 OLE, 91 Page Setup, 88 Print, 87-88 Replace, 88 sample program, 89-91 Components and Controls Gallery, 28 Create New Data Source, 583 Database Options, 612 dialog box procedure, 83 Insert Resource Copy, 881 Media Build Wizard Build Type, 843

907

908

dialog boxes UNLEASHED Disk Type, 843 Media Name, 842 Platforms, 844 Tag Gile, 843 message boxes, 82-83 modal, 81-82 modeless, 82 New, 13 New Class, 27, 275 New Project Information, 19 ODBC Text Setup, 594 Project Settings, 29, 542 Unicode applications, 877 Project Wizard Choose Dialogs, 831 Choose Target Platform, 832 Specify Components, 834 Specify File Groups, 834 Specify Languages, 833 Summary, 834 templates, 83 Dialog Data Exchange (DDE) control data types, 285 custom data types, 285 data validation, 283 simple types, 284 Dialog Data Exchange (DDX) function, 25 DIALOG statements, 102-103 control statements, 103 dialog units, 102 options, 102-103 dialog statements, 102 dialog template resources, 83 dialog templates, inserting ActiveX controls, 323-324 dialog units, 102 dialog-based applications, 15, 270-271 creating, 19 DIALOGEX statement, 103 dialogs (objects), 274 CMyDialog class declaration, 278 member functions, 278-279 common dialogs CColorDialog class, 299 CFileDialog class, 299-300 CFindReplaceDialog class, 300-301 CFontDialog class, 302 COle class, 304 CPageSetupDialog class, 302-303 CPrintDialog class, 303-304 constructing, 274 classes, 275-277 member variables, 277-278 templates, 275 DDE (Dialog Data Exchange) control data types, 285 custom data types, 285 data validation, 283 simple types, 284 invoking, 279-280 message handling, 285-286 modal, 280 modeless, 280-282 constructing, 281 member functions, 282 OLE server applications, 450-451 property sheets, 286 constructing, 291 member functions, 291-292 modeless, 292-294 pages, 287-290 DialogShowSdStartCopy function, 841 DIB clipboard format, 195 DIBs (device-independent bitmaps), 129 creating, 136 DIF clipboard format, 195 Direct3D, 788 Direct3D API, 785 DirectAnimation, 785, 791 DirectDraw, 785-788 DirectDrawCreate function, 787 DirectInput, 791 DirectInput API, 785 directories, FTP, 390 DirectPlay, 785, 790-791 DirectPlayEnumerate function, 790 DirectSetup, 785, 792 DirectShow, 785, 791 DirectSound, 785, 788-790 DirectX, 784 APIs, 785 Direct3D, 788 DirectAnimation, 791 DirectDraw, 786-788 DirectInput, 791 DirectPlay, 790-791 DirectSetup, 792 DirectShow, 791 DirectSound, 788-790 COM, 786 COM interfaces, 785-786 IDirectDraw2, 787 IDirectDrawClipper, 788 IDirectDrawSPalettee, 787 IDirectDrawSurface, 787 IDirectPlay, 790 IDirectSound, 788-790 IUnknown, 785 sample application, 792, 797-799 resource file, 798

DrawDib function INDEX source code, 792 DirectXSetup function, 792 disabling optimization, 29 DISCARDABLE attribute, 100 DisconnectNamedPipe function, 734 Disk Type dialog box, 843 DispatchMessage function, 38, 52 display device contexts, 45 DISPTEST.EXE, 11 distributing ActiveX controls, 534 installation programs distribution sets, 842-844 media types, 829 DllCanUnloadNow function, 862 DllGetClassObject function, 862 DLLs (Dynamic Link Libraries) compiler, 6-7 GDI, 110-111, 122-123 bitmaps, 129, 135-136 brushes, 123-124 fonts, 124-125 palettes, 125-129 paths, 136-137 pens, 123 printing, 137-138 Rectangle function, 111 text output, 137 help files, 812 InstallShield, 838 MFC-based, creating, 20 resource files, 107-108 satellite, 879 shell extension DLLs, 849-850 Registry entries, 864 RESOURCE.H, 853 WMFPRPSH.CPP, 856, 859-861 WMFPRPSH.DEF, 863 WMFPRPSH.H, 855-856 WMFPRPSH.MAK, 863 WMFPRPSH.RC, 854 subclassing, 78-79 TAPI, 716 DNS (Domain Name System), 684 names, DCOM server Registry entries, 573 DOBJVIEW.EXE program, 198 DoCalculate function, 498 document objects, 258 ActiveX documents, 562 CCmdTarget class, 265-266 CDocItem class, 265-266 CDocument class declaring, 258-259 functions, 259-260 data elements, 262-264 message handling, 261 methods, 508-509 modifying, 254 overridable functions, 261-262 paths, 260 properties, 508-509 views, 260 YAH application, 248-249 modifying, 253 Document Template Strings dialog box, 17 document templates creating, 243 strings, substrings, 252-253 document-based applications classes, 18-19 creating, 15-19 database support, 15 OLE support, 15 project files, 15 document-view model, 271 documents ActiveX, 560 compound document technology, 422 data transfer, 423-424 embedded objects, 424 in-place activation, 425 linked objects, 424 structured storage, 423 closing, 509 printing, 509 saving, 509 serialization, 363 documents collection, 508 Documents property (application objects), 507 DoDataExchange function, 497-498 DoDragDrop function, 426, 483 Domain Name System (DNS), 684 domains, 684 DoModal function, 280 DoPreparePrinting function, 268 DoVerb function, 440 DPtoLP function, 120 drag and drop (OLE), 426, 478 container application, 478-479 positioning support, 479-481 selection support, 482 drag sources, 483-484 drop targets, 484-491 supporting, 482-483 DrawDib function, 763

909

910

DrawFocusRect function UNLEASHED DrawFocusRect function, 343 DrawHello function, 60 drawing, 122-123 ActiveX controls, 530-531 ATL, 546-547 CDC class functions, 341 cubes, 778-780 curves, 134-135 filled shapes, 135 functions, 132-133 AngleArc, 135 Arc, 134 ArcTo, 134 LineTo, 134 MoveToEx, 134 OLE server application, 448-450 PolyBezier, 135 PolyBezierTo, 135 Polyline, 134 PolylineTo, 134 PolyPolyline, 134 GDI objects bitmaps, 129, 135-136 brushes, 123-124 fonts, 124-125 palettes, 125-129 pens, 123 lines, 134 OpenGL operations, 771 paths, 136-137 pentagons, 771 polylines, 342 rectangles, 111 shapes, 342-343 drawing functions, 45 DrawText function, 137, 344 DrawTextEx functions, 137 DROP statements, 593 drop targets, implementing, 484-491 DSPBITMAP clipboard format, 195 DSPENHMETAFILE clipboard format, 195 DSPMETAFILEPICT clipboard format, 195 DSPTEXT clipboard format, 195 DumpElements function, 381 DWORD Registry value type, 207 dynamic binding, 621 Dynamic HTML, 654-657 DirectAnimation API, 791 dynasets, 603 ellipses (...) drawing, 135 exception handler, 228-230 embedded objects, 424 Embedded SQL, 584 EmptyClipboard function, 197 ENABLE window mangement message, 73 EnableWindow function, 148 enabling, debugger, 29 end users, 322 EndPath function, 137, 345 ENDSESSION window mangement message, 73 enhanced metafiles, 112 ENHMETAFILE clipboard format, 195 EnterCriticalSection function, 153 Enum subkey (HKEY LOCAL MACHINE), 209 EnumChildWindows function, 66 EnumClipboardFormats function, 198 enumerators, 628 EnumFontFamilies function, 125 EnumThreadWindows function, 66 EnumWindows function, 65 eof function, 188 EqualRgn function, 135 errno global variable, 188 error handling CArchive class, 361 CFile class, 355 Error objects ADO, 634 OLE DB, 628 errors, reporting, 47

E
edit controls, 92 Dialog Data Exchange, 284 editing ActiveX documents, 560 events, 26 property pages, 531 automation properties, 25 class properties, 26 code, ClassWizard, 27 embedded objects, 474 help contents files, 818-820 help project files, 813-815, 818 member variables, 22-25 Registry, manually, 208-209 editors integrated, 8 Registry Editor, 208 EDITTEXT control statement, 103

File menu commands INDEX Event Log command, testing controls, 556 events, 409 ActiveX controls, 329-330, 528-529 event handling, 550-554, 557 states, 152 Excel spreadsheets, ODBC, 586 EXCEPTION_ACCESS_ VIOLATION value, 224 EXCEPTION_CONTINUE_ EXECUTION value, 221 EXCEPTION_CONTINUE_ SEARCH value, 221 EXCEPTION_EXECUTE_ HANDLER value, 221 EXCEPTION_FLT_ OVERFLOW value, 224 EXCEPTION_FLT_ UNDERFLOW value, 224 EXCEPTION_INT_DIVIDE_ BY_ZERO value, 224 EXCEPTION_PRIV_ INSTRUCTION value, 224 EXCEPTION_STACK_ OVERFLOW value, 224 exceptions, 398 C, 220-224 catching, 221 generating, 220 termination handling, 224-226 translating, 230-231 C++ classes, 227 generating, 229-230 termination handling, 226-227 catching, floating point exceptions, 223 DAO classes, 622 handlers, 220 ellipses, 228-230 filter expressions, 221-224 guard blocks, 221 nesting, 222-223 handling, macros, 398-399 MFC CArchiveException class, 402 CDaoException class, 404 CDBException class, 404 CException class, 399-400 CFileException class, 401-402 CInternetException class, 404-405 CMemoryException class, 401 CNotSupportedException class, 403 COleDispatchException class, 405 COleException class, 405 CResourceException class, 403 CUserException class, 405 throwing, 405-406 RPC exception handling, 743-744 executing OpenGL applications, 780 SQL statements, 603 Windows NT services, 651 EXEs, compiler, 6-7 exiting applications, 508 processes, 152 ExitProcess function, 152 EXPAND_SZ Registry value type, 207 Explorer, 848 Appbars, 848 applets, 850 Briefcase reconcilers, 850 extension DLLs, 849-850 Registry entries, 864 RESOURCE.H, 853 WNFPRPSH.CPP, 856, 859-861 WNFPRPSH.DEF, 863 WNFPRPSH.H, 855-856 WNFPRPSH.MAK, 863 WNFPRPSH.RC, 854 file viewers, 849 namespace, 848 property pages, adding, 853-864 taskbar, 848 installing taskbar icons, 851-852 expressions, filter expressions, 221-224 EXSTYLE option (DIALOG statement), 102 ExtCreatePen function, 123 Extended MAPI, 667, 675 Extended TAPI, 720 extended window styles, 70 extension DLLs, 849-850 ExtFloodFill function, 136 ExtTextOut function, 137

911

F
FAT (File Allocation Table), 178 Field objects (ADO), 635 file descriptors, 187 file dialogs, 299-300 file management, kernel services, 42 File menu commands Help Author, 820

912

File menu commands UNLEASHED New, 13 file monikers, 422 file objects, 42 File Open common dialog, 84 file pointers, 181 File Save As common dialog, 84-85 file servers, middle tier, 658 file systems accessing, 178 CD-ROMs, 179 file compression, 179 network volumes, 179 NT support, 178-179 File Transfer Protocol (FTP), 390 file viewers, creating, 849 filename extensions .avi, 753 .rc, 98 .res, 98 .wav, 752 files Application Configuration, 738 closing, 354 copying, 180-181 file objects, 179 asynchronous I/O operations, 181-186 closing, 180 console I/O operations, 189-190 creating, 180 low-level I/O operations, 187-188 opening, 180 simple I/O operations, 180-181 stream I/O operations, 188-189 formats, 752 AVI, 753 MIDI, 753 WAV, 752 Interface Definition Language, 738 locking, 355 management, CFile member functions, 354 memory-mapped files, 161, 171-173 metafiles, 45 opening, 354 reading, 354 resource files compiling, 107 DLLs, 107-108 linking, 107 multiline statements, 101-106 preprocessor directives, 99-100 raw data resources, 106 sample script, 98-99 single-line statements, 100 user-defined resources, 106 version information, 106 swap files, 160-161 transferring FTP, 392-393, 706-707 Gopher, 394, 707-708 HTTP, 394-395, 708-709 writing to, 354 Files button (Microsoft Help Workshop), 813 Files tab (New dialog box), 13 [FILES] section (help project files), 809 filled shapes, drawing, 135 filling bitmaps, 343 regions, 135 FillRect function, 135 filter expressions, 221-224 Find and Replace dialogs, 300-301 Find common dialog, 88 Find function, 370 Find Window command (Spy menu), 36 FindFirstChangeNotification function, 153 finding list elements, 370 FINDREPLACE structure, 88 FindText function, 88 FindWindow function, 66 flattened structures, 718 FlattenPath function, 137 Flip method, 787 floating point exceptions, handling, 223 Flush function, 354 FONT option (DIALOG statement), 102 Font Selection common dialog, 86, 302 FONT statement, 100 fonts CDC class functions, 344-345 installing/removing, 125 logical, 124-125 MFC support, 348 footnotes, help files, 807 FORMATETC structure, 423 formats, 752 AVI, 753 clipboard private formats, 196 registered formats, 196 standard formats, 194-196 MIDI, 753 WAV, 752 foundation classes, MFC library, 237 frame windows, 245-246 OLE server application, 445

functions INDEX FrameRect function, 135 FrameRgn function, 135 framing regions, 135 FreeConsole function, 190 freeing virtual memory, 167 FromHandle function, 336 FTP (File Transfer Protocol), 390, 701-702 file transfers, 392-393, 706-707 FtpFindFirstFile function, 707 FtpOpenFile function, 706 full-servers, 440 FullName property application objects, 507 document objects, 509 function calls, 41 error reporting, 47 GDI services, 44-45 kernal services, 42-43 user services, 44 WriteProfileString, 43 functions, 42 32-bit applications, 163 Abort, 354 AbortDoc, 345 Activate, 509 AddSegmentAtMessagePos, 60 AddView, 260 AfxEnableControlContainer, 323 aggregate functions, 592 AllocConsole, 190 AngleArc, 135 AnimatePalette, 126 Arc, 134 ArcTo, 134 Attach, 357 AVIFile, 763 BeginPaint, 71 BeginPath, 137, 345 bind, 686 BitBlt, 136, 343 Calculate, 501 CallNamedPipe, 734 calls, 41 CancelToClose, 291 catchall, 230 ChooseColor, 85 ChoosePixelFormat, 775 ClearCommBreak, 191 ClearCommError, 191 Close, 354, 509 CloseEnhMetaFile, 112 CloseHandle, 180 CloseMetaFile, 112 CoCreateGuid, 420 CoGetMalloc, 420 CombineRgn, 135 CombineTransform, 120 CommConfigDialog, 191 CommDlgExtendedError, 299 commit, 188 CompactDatabase, 622 CompareElements, 380 ConnectNamedPipe, 734 ConstructElements, 380 controlfp, 223 CopyData, 203 CopyElements, 381 CountClipboardFormats, 198 Create, 187 CreateBitmap, 129 CreateBrushIndirect, 123 CreateCompatibleBitmap, 129 CreateCompatibleDC, 112, 336 CreateDC, 111, 335 CreateDialog, 82 CreateDIBitmap, 136 CreateDIBPatternBrushPt, 124 CreateEnhMetaFile, 112 CreateEvent, 152 CreateFile, 180 CreateFont, 124 CreateFontIndirect, 124 CreateHatchBrush, 124 CreateIC, 112 CreateMetaFile, 112 CreateNamedPipe, 732 CreatePatternBrush, 124 CreatePen, 123 CreatePenIndirect, 123 CreatePipe, 732 CreateProcess, 152 CreateRemoteThread, 79 CreateSemaphore, 152 CreateSolidBrush, 124 CreateStockObject, 347 CreateThread, 151 CreateWindow, 55, 69-70 CreateWindowEx, 70 DDP_ functions, 532-533 DefDlgProc, 56, 75 DefWindowProc, 56, 75 DeleteContents, 262 DeleteDC, 112, 336 DeleteObject, 122, 347 DeletePalette, 125 DestroyWindow, 52-53 DestructElements, 380 Detach, 357 DFX functions, 615-616 DialogShowSdStartCopy, 841 DirectDrawCreate, 787 DirectPlayEnumerate, 790 DirectXSetup, 792 DisconnectNamedPipe, 734 DispatchMessage, 38, 52 DllCanUnloadNow, 862 DllGetClassObject, 862 DoCalculate, 498 DoDataExchange, 497-498 DoDragDrop, 426, 483 DoModal, 280

913

914

functions UNLEASHED DoPreparePrinting, 268 DoVerb, 440 DPtoLP, 120 DrawDib, 763 DrawFocusRect, 343 DrawHello, 60 DrawText, 137, 344 DrawTextEx, 137 DumpElements, 381 EmptyClipboard, 197 EnableWindow, 148 EndPath, 137, 345 EnterCriticalSection, 153 EnumChildWindows, 66 EnumClipboardFormats, 198 EnumFontFamilies, 125 EnumThreadWindows, 66 EnumWindows, 65 eof, 188 EqualRgn, 135 error reporting, 47 ExitProcess, 152 ExtCreatePen, 123 ExtFloodFill, 136 ExtTextOut, 137 FillRect, 135 Find, 370 FindFirstChangeNotification, 153 FindText, 88 FindWindow, 66 FlattenPath, 137 Flush, 354 FrameRect, 135 FrameRgn, 135 FreeConsole, 190 FromHandle, 336 FtpFindFirstFile, 707 FtpOpenFile, 706 GDI services, 46 GetCharABCWidths, 125 GetClassInfo, 38 GetClientRect, 775 GetClipboardFormatName, 196 GetCommState, 191 GetDC, 111 GetDesktopWindow, 65 GetDeviceCaps, 125 GetDialogBaseUnits, 102 GetDocument, 268 GetDriveType, 178 GetEnhMetaFile, 112 GetExceptionCode, 229 GetFirstViewPosition, 259 GetHeadPosition, 369 gethostbyname, 687 GetIDispatch, 506 GetLastError, 47 GetLogFont, 348 GetLogPen, 347 GetMessage, 52, 148 GetMetaFile, 112 GetNamedPipeHandleState, 733 GetNamedPipeInfo, 733 GetNumberFormat, 869 GetObjectContext, 661 GetOpenFileName, 84 GetOpenFileNamePreview, 755 GetParent, 66 GetPixel, 136 GetPrivateProfileString, 43 GetProcessHeap, 170 GetRgnBox, 135 GetROP2, 338 GetSafeHdc, 336 GetSaveFileName, 84 GetSize, 372 GetStatus, 354 GetStdHandle, 190 GetStockObject, 123 GetSystemPaletteEntries, 125 GetTabbedExtent, 125 GetTailPosition, 369 GetTextExtent, 344 GetVolumeInformation, 178 GetWindow, 66 glBegin, 771 GopherFindFirstFile, 708 GopherOpenFile, 708 GrayString, 344 HashKey, 381 HeapAlloc, 170 HeapCompact, 170 HeapCreate, 170 HeapDestroy, 170 HeapFree, 170 HeapRealloc, 170 HeapSize, 170 Help, 508 HttpOpenRequest, 709 HttpSendRequest, 709 InsertAt, 372 InterlockedDecrement, 174 InterlockedIncrement, 174 InternetOpen, 705 InternetReadFile, 706 InternetSetStatusCallback, 709 Invalidate, 479 InvalidateRect, 71-72 InvalidateRng, 71-72 InvertRect, 135 InvertRgn, 135 IsClipboardFormatAvailable, 198 IsSelected, 268, 466 joyGetPosEx, 791 kernal services, 43 LeaveCriticalSection, 153 lineGetID, 721 lineGetNewCalls, 722 lineHandoff, 722 lineInitialize, 720 lineMakeCall, 721 lineNegotiateAPIVersion, 721 lineSetAppSpecific, 723

functions INDEX lineShutdown, 721 LineTo, 134 LoadBitmap, 129 locale-related, 870 LockRange, 355 LongPathToQuote, 840 lseek, 187 MAPIAddress, 673 MAPISendDocuments, 670-671 MAPISendMail, 672 MaskBlt, 136 mciGetCreatorTask, 762 mciGetDeviceID, 762 mciSendCommand, 762 mciSendString, 762 MCIWndCreate, 750-752, 755 MCIWndRegisterClass, 755 MessageBox, 51, 82 midl_user_allocate, 742 midl_user_free, 742 MoveToEx, 134 NewWindow, 509 obsolete, 171 OffsetRgn, 135 OnApply, 292 OnCancel, 292 OnCancelEditCntr, 466 OnChangeItemPosition, 461-462, 469 OnCloseDocument, 262 OnCreate, 486 OnDragEnter, 485-487 OnDragLeave, 485, 488 OnDragOver, 485-488 OnDraw, 463-464, 471-473, 778 OnDrop, 485-489 OnFileSendMail, 262, 677 OnGetItemPosition, 470 OnInsertObject, 466 OnLButtonDown, 269 OnNewDocument, 253, 261 OnOpenDocument, 262 OnPrepareDC, 269, 341 OnSaveDocument, 262 OnSetFocus, 466 OnSize, 466 OnViewDialog, 280 OnViewPropertySheet, 291 OnViewPropertysheet, 291-294 Open, 354 open, 187 OpenClipboard, 197 OpenFile, 180 OpenSCManager, 642 PageSetupDlg, 88 PaintRgn, 343 PatBlt, 136, 343 PathToRegion, 137 PeekMessage, 148 Playback, 752 PlayEnhMetaFile, 112 PlayMetaFile, 112 PlaySound, 764 PlgBlt, 136 PolyBezier, 135 PolyBezierTo, 135 Polyline, 134 PolylineTo, 134 PolyPolygon, 135 PolyPolyline, 134 PolyTextOut, 137 PostNcDestroy, 292 PostQuitMessage, 52-55 PreCreateWindow, 777 Print, 509 PrintDlg, 87 PrintOut, 509 PrintPreview, 509 PtInRegion, 135 PurgeComm, 191 PX_ functions, 527-528 QueryInterface, 421 Quit, 508 RaiseException, 224 read, 187 ReadConsole, 190 ReadFile, 190 RealizePalette, 125 Rectangle, 111 RectInRegion, 135 RegCloseKey, 213 RegCreateKeyEx, 213 RegDeleteKey, 214 RegEnumKeyEx, 214 RegisterClass, 67-69 RegisterClipboardFormat, 196 RegisterWindowMessage, 40 RegOpenKeyEx, 213 RegQueryValueEx, 213 RegSetKeySecurity, 214 RegSetValueEx, 213 ReleaseMutex, 153 ReleaseSemaphore, 152 RemoveAt, 372 RemoveView, 260 RepairDatabase, 622 Repeat, 508 ReplaceText, 88 ResetEvent, 152 ResumeThread, 151 RevertToSaved, 509 RFX functions, 604 RoundRect, 135 RpcServerUseProtseqEp, 740 RpcStringBindingCompose, 743 Save, 509 SaveAs, 509 ScrollDC, 344 Seek, 354 select, 689 SelectClipPath, 137, 346 SelectClipRgn, 130 SelectObject, 122

915

916

functions UNLEASHED SelectPalette, 125 SelectTAPIDevice, 727 Serialize, 360, 470 SerializeElements, 264, 381 set_se_translator, 230-231 SetAbortDoc, 345 SetBkColor, 124, 338 SetBkMode, 124 SetBrushOrgEx, 124 SetClassLong, 38, 77 SetClipboardData, 194 SetCommBreak, 191 SetCommMask, 186 SetConsoleMode, 190 SetDIBitsToDevice, 136 SetFilePointer, 181 SetLastError, 47 setlocale, 870 SetMenus, 203 SetModified, 292 SetModifiedFlag, 260 SetNamedPipeHandleState, 733 SetPixel, 136, 343 SetPixelFormat, 770 SetPolyFillMode, 338 SetROP2, 338 SetSize, 372 SetStatus, 354 SetTextColor, 344 SetTextJustification, 344 SetThreadLocale, 872 SetTitle, 260 SetupComm, 191 SetupFolders, 840 SetViewportExtEx, 113 SetViewportOrgEx, 113 SetWindowExtEx, 113 SetWindowLong, 38, 75 SetWindowOrgEx, 113 SetWindowsHookEx, 79 SetWizardMode, 292 SetWorldTransform, 120 SHAppBarMessage, 848 Shell_NotifyIcon, 848 Shutdown, 741 socket, 686 SQLAllocConnect, 585 SQLAllocEnv, 584 SQLAllocStmt, 585 SQLBindParameter, 590 SQLBrowseConnect, 589 SQLConnect, 585 SQLDisconnect, 586 SQLExecDirect, 589 SQLExecute, 585 SQLExtendedFetch, 590 SQLNativeSql, 590 SQLParamOptions, 590 SQLPrepare, 585 SQLRowCount, 590 SQLSetConnectOption, 589 SQLSetPos, 590 SQLSetStmtOption, 589-590 SQLTransact, 590 standard C/C++ library, 47 StartServiceCtrlDispatcher, 642 StretchBlt, 136, 343 StretchDIBits, 136 stub functions, 736 TabbedTextOut, 137, 344 tapiGetLocationInfo, 713 tapiRequestMakeCall, 713 tell, 187 TextOut, 137, 344 TlsAlloc, 174 TlsFree, 174 TransactNamedPipe, 734 TransmitCommChar, 191 Undo, 508 UnlockRange, 355 UnrealizeObject, 347 UpdateAllViews, 260 user services, 44 UuidCreate, 420 virtual, 419 VirtualAlloc, 167 VirtualFree, 167 VirtualLock, 167 VirtualProtect, 167 VirtualQuery, 167 WaitCommEvent, 191 WaitForMultipleObjects, 186 WaitNamedPipe, 734 wcreat, 187 wglCreateContext, 770 WidenPath, 137 WinHelp, 812-813 WinMain, 55 wopen, 187 WriteConsole, 190 WriteConsoleOutputAttribute, 190 WSAAsyncSelect, 690 Yield, 142

G
game development, 784 GDI (Graphics Device Interface), 334, 110-111 clipping operations, 345 coordinates, mapping, 339-341 device contexts attributes, 338 client-area, 337 creating, 335-336 metafile, 337-338 paint-time, 336-337 window, 337 drawing operations, CDC class functions, 341-345 functions BeginPaint, 71 Rectangle, 111 IDirectDrawSurface interface, 787

GRANT statements INDEX MFC support, 346-347 bitmaps, 348 brushes, 347-348 fonts, 348 palettes, 348-349 pens, 347 regions, 349 objects bitmaps, 129, 135-136 brushes, 123-124 creating, 122 deleting, 122 fonts, 124-125 palettes, 125-129 pens, 123 selecting, 341-342 paths, 136-137 printing, 137-138 regions, combining, 135 text output, 137 gdi.dll, 110-111 GDIOBJFIRST clipboard format, 195 GDIOBJLAST clipboard format, 195 generating exceptions MFC applications, 405-406 sample program listing, 220 generic mappings, Unicode applications, 875 Generic Windows Application, 56-57 get_all method, 656 GetCharABCWidths function, 125 GetClassInfo function, 38 GetClientRect function, 775 GetClipboardFormatName function, 196 GetCommState function, 191 GetDC function, 111 GetDesktopWindow function, 65 GetDeviceCaps function, 125 GetDialogBaseUnits function, 102 GetDocObjectServer member function, ActiveX document server, 568 GetDocument function, 268 GetDriveType function, 178 GetEnhMetaFile function, 112 GetExceptionCode function, 229 GetFirstViewPosition function, 259 GetHeadPosition function, 369 gethostbyname function, 687 GetIDispatch function, 506 GetLastError function, 47 GetLogFont function, 348 GetLogPen function, 347 GetMessage function, 52, 148 GetMetaFile function, 112 GetNamedPipeHandleState function, 733 GetNamedPipeInfo function, 733 GetNumberFormat function, 869 GetObjectContext function, 661 GetOpenFileName function, 84 GetOpenFileNamePreview function, 755 GetParent function, 66 GetPixel function, 136 GetPrivateProfileString function, 43 GetProcessHeap function, 170 GetRgnBox function, 135 GetROP2 function, 338 GetSafeHdc function, 336 GetSaveFileName function, 84 GetSize function, 372 GetStatus member function, 354 GetStdHandle function, 190 GetStockObject function, 123 GetSystemPaletteEntries function, 125 GetTabbedExtent function, 125 GetTailPosition function, 369 GETTEXT window mangement message, 73 GetTextExtent function, 344 gettime application, 691 MFC, 692-694 WinSock, 690-692 GetVolumeInformation function, 178 GetWindow function, 66 GLAUX (OpenGL Progrmmng Guide Auxiliary Library), 772 glBegin function, 771 global subclassing, 78-79 GlobalAlloc, 165 globally unique identifiers (GUIDs), 739 GLU (OpenGL Utility Library), 772 Gopher, 390-391, 702-704 file transfers, 394, 707-708 GopherFindFirstFile function, 708 GopherOpenFile function, 708 GRANT statements, 593

917

918

graphics UNLEASHED graphics bitmaps, CDC class functions, 343-344 coordinates, mapping, 339-341 DirectX, 784-785 COM interfaces, 785-786 Direct3D, 788 DirectDraw, 786-788 DirectPlay, 790-791 sample application, 792, 797-799 drawing functions, CDC class, 341-345 fonts, CDC class functions, 344-345 GDI, 334 object selection, 341-342 lines, drawing, 342 OpenGL library, 768 C applications, 772-776 drawing operations, 771 GLAUX, 772 GLU, 772 initialization, 770-771 MFC applications, 776-780 overview, 768-769 shapes, drawing, 342-343 Graphics Device Interface, see GDI GrayString function, 344 guard blocks, 221 guard pages, 167 guidgen.exe, 420, 739 GUIDs (globally unique identifiers), 420, 739 DirectPlay interface, 790

H
HAL, 786 handler functions, 22 handlers ellipses (...), 228-230 exception handlers, 220 nesting, 222-223 termination handlers, 225 handles, 43 hardware, device contexts, 111-112 HashKey function, 381 hatch brushes, creating, 124 hcrtf.exe compiler, 811 HDROP clipboard format, 195 header controls, 93, 307 headers (IP), 682 HeapAlloc function, 170 HeapCompact function, 170 HeapCreate function, 170 HeapDestroy function, 170 HeapFree function, 170 HeapRealloc function, 170 heaps, 170 HeapSize function, 170 Height property (application objects), 507 HEL, 786 Hello, World program, 50-51, 238 application objects, 239-243 AppWizard options, 239 compiling, 50 customizing, 253-255 executing, 50 document objects, 248-249 declaring, 248 implementing, 249 modifying, 253-254

graphics version, 58-60 message loops, 51-53 message map, 243-245 resources, 252-253 source code, 50 window procedures, 53-56 windows frame window, 245-246 view window, 250-252 HELLO.C listing, 851-852 Help Author command (File menu), 820 help files, 804-805 AppWizard-generated files, 822 compiling, 811 developing, 805-806 DLLs, 812 footnotes, 807 Help Workshop (Microsoft) contents files, 818-820 file testing, 820-821 project editing, 813-814, 818 hyperlinks, 807 invoking, 812-813 macros, 811-812 Microsoft Help Workshop, 813 multilanguage applications, 885 project files, 809-810 editing, 813, 816-818 RTF, 806-809 tables of contents, 810-811 editing, 818-820 topics, 806-809 Help function, 508 Help Index property sheet, 819 Help Workshop (Microsoft), 12, 813 buttons Alias, 817 Bitmaps, 815

implementing INDEX Config, 817 Data Files, 818 Edit Files, 819 Files, 813 Link Files, 819 Map, 816 Options, 813 Save and Compile, 818 Tabs, 819 Windows, 814 projects contents files, 818-820 editing, 813-814, 818 testing, 820-821 helper functions (collection classes), 380-381 HIENGLISH mapping mode, 339 HIMETRIC mapping mode, 115, 339 HKEY CLASSES ROOT key, 207, 210-211 HKEY CURRENT CONFIG key, 208 HKEY CURRENT USER key, 211-212 HKEY DYN DATA key, 208 HKEY LOCAL MACHINE key, 207-210 HKEY USERS key, 208, 211 host addresses, 683 hostnames, 684 hotkey controls, 93, 307-308 HTML (Hypertext Markup Language) dynamic HTML, 654 get_all method, 656 HTTP (HyperText Transfer Protocol), 391, 704-705 file transfers, 394-395, 708-709 HttpOpenRequest function, 709 HttpSendRequest function, 709

919

I
I/O operations (file objects), 180-181 asynchronous, 181-186 communication ports, 191 console, 189-190 low-level file descriptors, 187 functions, 187-188 streams, 188-189 IBM PC BIOS functions, 47 IClassFactory interface, 421 ICMP, 681 ICON control statement, 103 ICON statement, 100 icons, taskbar icons, installing, 851-852 IContextMenu interface, 849 ID_VIEW_DIALOG identifier, 280 IDataObject viewer, 198 IDE (Integrated Development Environment) compiler command-line tools, 4 EXEs/DLLs, 6-7 resources, 5 components, 11 debugger, 8 integrated editor, 8 message spy, 11 profilers, 12 project templates, 10 project workspaces, 10 PVIEW.EXE, 11 resources, editing, 9 version control systems, 12 identifiers (interfaces) CLSIDs, 421 GUIDs, 420 Identity tab (DCOMCNFG.EXE), 574

IDirectDraw2 interface, 787 IDirectDrawClipper interface, 788 IDirectDrawPalette interface, 787 IDirectDrawSurface interface, 787 IDirectPlay interface, 790 IDirectSound interface, 788-790 IDispatch interface, 425, 656 IDropSource interface, 426 IDropTarget interfaces, 426 IExtractIcon interface, 849 IFileViewer interface, 849 IHTMLElement interface, 656 image lists, 309 IMalloc interface, 420 implementing classes CAMFCSet, 599-600 CDAOSet, 615 CMainFrame, 246 CMCTL, 328-329 CMCTLApp, 516-517 CMCTLCtrl, 521-523 CMCTLPropPage, 518-519 CMySheet, 293-294 CSlot, 650-651 CSrv, 648-649 CYAHApp, 241-243 CYAHDoc, 249 CYAHView, 251-252 MyPage1, 290 custom data types, 285 drag and drop drag sources, 483-484 drop targets, 484-491 functions, DoCalculate, 498 property sheets modeless, 293-294 pages, 290

920

implicit_handle keyword UNLEASHED implicit_handle keyword, 739 in keyword, 739 in-place activation, 425 in-place editing, 440 in-process servers, 425, 440 include directives, 810 Index Files button (Microsoft Help Workshop), 819 information contexts, 112 inheritance, COM, 420 INI files, Registry, compared, 212 initializing CRgn objects, 349 member variables, 491 MFC classes CFile, 354 OpenGL, 770-771 C, 775 MFC, 777-778 WinSock, 686 InitInstance function, ActiveX document server, 568 Insert menu commands New ATL Object, 542 Object, 447 Insert Resource Copy dialog box, 881 INSERT statements, 592 InsertAt function, 372 inserting ActiveX controls, 323-324 components into projects, 28 installation programs, 828 creating, 831-835 components, 834 dialogs, 831 languages, 833 setup types, 834 target platforms, 832 customizing, 839-840 debugging, 845 distribution sets, 842-844 file groups, 835-838 InstallShield, 830 requirements, 828-829 authentication, 829 conditional copying, 830 configuration updates, 830 efficient storage, 829 media types, 829 registration, 829 shared components, 830 uninstallation, 830 user options, 829 setup scripts, 840-841 testing, 844 installing fonts, 125 multilanguage applications, 885-886 taskbar icons, 851-852 Windows NT services, 651 InstallShield, 828-830 install programs creating, 831-835 customizing, 839-840 debugging, 845 distribution sets, 842-844 file groups, 835-838 setup scripts, 840-841 testing, 844 localized, 886 Project Wizard, 831-835 Integer division value (filter expressions), 224 integers, 162 integral array classes, 374-375 integrated editor, 8 inter-object communication, 421-422 Interactive property (application objects), 507 Interface Definition Language (IDL) files, 738 interfaces ActiveX documents, 561 ActiveX property pages adding, 531 ATL, 547-550 control properties, 531-533 editing, 531 multiple pages, 533 COM, 418-419 defining, 420-421 IClassFactory, 421 identifiers, 420 IDispatch, 425 IDropSource, 426 IDropTarget, 426 IMalloc, 420 IPersistStorage, 423 IRootStorage, 423 IStorage, 423 IStream, 423 IUnknown, 420 customizing, 426 DHTML, 656 IOleCommandTarget, 561 IOleDocumentSite, 562 IOleDocumentView, 561 MCI, 759-761 command sets, 762 command syntax, 761-762 functions, 762-763 macros, 762-763 multimedia devices, 759 notification messages, 763 video playback, 760-761 multimedia, 753 ACM, 764 audio mixers, 765 AVIFile functions, 763 custom handlers, 763 DrawDib function, 763-764

IWebBrowser2 interface INDEX VCM, 764 video capture, 764 multimediaPlaySound function, 764 interlocked variable access, 153, 174 InterlockedDecrement function, 174 InterlockedIncrement function, 174 Intermediate Mode (Direct3D), 788 international applications multilanguage resources, 877 help files, 885 installation, 885-886 satellite DLLs, 879 translating user interface, 878 versions, 879 nationalization grammar, 868 language-specific info, 869 non-European languages, 870 text size, 867 user input, 869 tools character sets, 872-873 locales, 870-872 unicode applications console, 874 MFC, 876 Windows applications, 875-876 Internet connections establishing, 389-390 FTP, 390 Gopher, 390-391 HTTP, 391 domains, 684 file transfers, 391 protocols, 700 ARP, 681 DNS, 684 FTP, 701-702 Gopher, 702-704 HTTP, 704-705 ICMP, 681 IP, 681 RARP, 681 TCP, 680 TCP/IP, 680-685 UDP, 681 services, 685 sessions, establishing, 389 support classes architecture, 388-389 CFtpConnection, 390 CGopherConnection, 390-391 CHttpConnection, 391 CInternetException, 392 CInternetFile, 391 CInternetSession, 389 Internet Control Message Protocol (ICMP), 681 Internet Explorer, DHTML, Automation, 655 Internet Protocol, see IP InternetOpen function, 705 InternetReadFile function, 706 InternetSetStatusCallback function, 709 Invalidate function, 479 InvalidateRect function, 71-72 InvalidateRng function, 71-72 inverting regions, 135 World Coordinate Transforms, 121-122 InvertRect function, 135 InvertRgn function, 135 invoking ClassWizard, 20 context-sensitive help, 812-813 dialogs, 279-280 hcrtf utility, 811 property sheets, modeless, 294 IOleCommandTarget interface, 561 IOleDocumentSite interface, 562 methods, 565 IOleDocumentView interface, 561 iostream classes, 188-189 IP (Internet Protocol), 681 address control, 308 datagrams, 682 headers, 682 host addresses, 683 hostnames, 684 multicasting, 683 numbers, 308 routing, 683 IPersistFile interface, 849-850 IPersistStorage interface, 423 IRootStorage interface, 423 IsClipboardFormatAvailable function, 198 IShellExtInit interface, 849 IShellFolder interface, 848 ISOTROPIC mapping mode, 115, 340 IsSelected function, 268, 466 IStorage interface, 423 IStream interface, 423 istrstream class, 188 item monikers, 422 IUnknown interface, 420, 785-786 IWebBrowser2 interface, 656

921

922

joyGetPosEx function UNLEASHED

J-K
joyGetPosEx function, 791 justifying text, 344 kernal services (Windows), 42-43 keys (Registry) creating, 213-214 deleting, 214 editing, 208-209 HKEY CLASSES ROOT, 207, 210-211 HKEY CURRENT CONFIG, 208 HKEY CURRENT USER, 211-212 HKEY DYN DATA, 208 HKEY LOCAL MACHINE, 207-210 HKEY USERS, 208, 211 names, 206 opening, 212-213 predefined, 208 keywords DIALOG, 102-103 implicit_handle, 739 in, 739 MENUITEM, 104 POPUP, 104 SEPARATOR, 104 STRINGTABLE, 105 virtual, 419 volatile, 151 see also statements Keywords property (document objects), 509 KILLFOCUS window mangement message, 73

L
LANGUAGE statement, 100, 880 LeaveCriticalSection function, 153 Left property (application objects, 507 Level 1 API (ODBC), 584 Level 2 API (ODBC), 584 libraries ATL (ActiveX Template Library) advantages, 538-539 bitmaps, 554-555 building controls, 539 control methods, 545-546 control properties, 543-545 drawing functions, 546-547 event handling, 550-554, 557 Object Wizard, 542-543 property pages, 547-550 Proxy Generator, 552 skeleton project, 540-542 testing controls, 556 compiling, 5 MFC applications, 236 foundation classes, 237 MAPI support, 677-678 OpenGL C applications, 772-776 drawing operations, 771 GLAUX, 772 GLU, 772 initialization, 770-771 MFC applications, 776-780 overview, 768-769 standard C/C++, 47

WinInet, 700, 705-706 asynchronous mode, 709 cache management functions, 709 FTP sessions, 706-707 Gopher sessions, 707-708 HTTP sessions, 708-709 library functions, Unicode, 874 line devices, 714 linear transformations combining, 119-120 matric forms, 118-119 reflection, 118 rotation, 116 scaling, 116 shearing, 117 translation, 115 World Coordinate Transforms, 120 lineGetID function, 721 lineGetNewCalls function, 722 lineHandoff function, 722 lineInitialize function, 720 lineMakeCall function, 721 lineNegotiateAPIVersion function, 721 lines, drawing, 134, 342 lineSetAppSpecific function, 723 lineShutdown function, 721 LineTo function, 134 Link Files button (Microsoft Help Workshop), 819 LINK Registry value type, 207 linked objects, 424 linker, 4 linking help files, 807 resource files, 107 list boxes, 92 list controls, 93, 309-310 LISTBOX control statement, 103

listings INDEX listings accessing a controls properties, 329 ADO example, 635-637 Application Configuration File, 739 ASRV.odl type library, 504 automation client, 436 automation server, 428-434 C++ Exception handling, 226 CACTL::OnDraw member function, 546-547 CACTL::OnLButtonDown member function, 554 CACTLProp::Apply member function, 550 CADAOSet::GetDefaultSQL, 618 CADCCntrItem class declaration, 563 implementation, 564 CAMFCSet class declaration, 598-599 implementation, 599-600 CASRVDoc class declaration, 502-503 CASRVDoc::Calculate, 503 CASRVDoc::DoCalculate implementation, 498 CASRVDoc::DoDataExchange, 498 CASRVDoc::GetResult, 503 CASRVView::OnCalculate implementation, 499 CColorDialog in CDLG, 299 CCubeView::PreCreateWindow, 777 CDAOSet class declaration, 613-615 implementation, 615 CFile in a console application, 355 CFileDialog in CDLG, 300 CFindReplaceDialog in CDLG, 301 CFontDialog in CDLG, 302 client communication using shared memory, 173 clip paths, 131-132 CMainFrame class declaration, 245-246 implementation, 246 CMainFrame::OnViewDialog function, 280 CMainFrame::OnViewPropertysheet, 291 CMapStringToString console application, 376-377 CMC in a console application, 674 CMCTL class declaration, 327-328 implementation, 328-329 CMCTLApp class declaration, 516 implementation, 516-517 CMCTLCtrl class declaration, 519-520 implementation, 521-523 CMCTLCtrl property get/ set methods, 526 CMCTLCtrl::OnDraw member function, 530 CMCTLPropPage class declaration, 517-518 implementation, 518-519 CMyDialog class declaration, 278 CMyDialog member functions, 278-279 CMyPage1 class declaration, 289 implementation, 290 CMySheet class declaration, 292 implementation, 293 COCONCntrItem::Invalidate, 481 COCONCntrItem::OnChangeItemPosition, 469-470, 480 COCONCntrItem::OnDraw, 481 COCONCntrItem::OnGetItemPosition, 480 COCONCntrItem::Serialize, 470, 480-481 COCONView::OnCreate member, 486 COCONView::OnDragEnter function, 486-487 COCONView::OnDraw, 471 COCONView::OnDrop member function, 488-489 COCONView::OnLButtonDown, 483-484, 490 COM+, 577 common dialogs, 89-91 communication program w/ multiple threads, 182-183 w/ overlapped I/O, 184-186 compiling DLL function, 6 DLLs, 7 constructing a modeless dialog, 281 copying file using Win32 file functions, 180-181

923

924

listings UNLEASHED CPageSetupDialog in CDLG, 303 CPrintDialog in CDLG, 304 CSlot class declaration, 649-650 implementation, 650-651 CSrv class declaration, 647-648 implementation, 648-649 CStdioFile, 356 CYAHApp class declaration, 240 implementation, 241-243 CYAHDoc class declaration, 248 implementation, 249 CYAHView class declaration, 250 implementation, 251-252 DCOM automation client, 576 Registry entries, 573 determining integer size, 162 DHTML, controlling from C++, 655 DialogShowSdStartCopy function (InstallShield), 841 DirectX application, 792 DirectX application resource file, 798 dynamic HTML, 654 example for the use of a mutex object, 154 exception handlers and GetExceptionCode, 229 failed attempt to catch C exceptions, 228 final version of COCONView::OnDraw, 473 FTP application, 706 generating an exception, 220 get and put functions in ACTL, 545-546 GetOpenFileNamePreview, 755-756 gettime command-line utility, 691-692 Gopher application, 707 graphics version of Hello, World, 58-60 handler for WM_LBUTTONDOWN messages, 472, 482 handling a mouse event in MCTL, 529 handling an ActiveX control event, 330 handling of TCN_SELCHANGE messages, 315 handling of TVN_SELCHANGED messages, 318 handling sparse matrices, 167-169 handling the divide by zero exception, 221 handling WM_HSCROLL messages, 313 header file for clipboardaware program, 202 Hello program hello.c, 50 multilingual, 880 new window class, 53-54 simple message loop, 52 HELLO.C, a taskbar icon example, 851-852 help project file, 809 help table of contents file, 810 help topic file, 806-807 HTTP application, 708-709 COCONCntrItem functions, 461 of CCubeView::OnCreate, 778 CCubeView::OnDraw, 779-780 COCONView::OnDraw, 463 Serialize function, 443 OnDraw function, 444 COCONCntrItem functions, 461 COSRVView::OnLButtonDown, 451 tree control, 317 m_rect, 469, 479 progress bar, 311 Interface Definition Language File, 738 world transformation, 121-122 modeless property sheet, 294 list control in CTRL, 310 loading a list from a CArchive, 362 locale, changing at initialization, 884 main service module, MAIN.CPP, 644-646 MAPISendDocuments function, 670-671 MCI playback using command messages, 760 MCI playback using command strings, 760 member functions in modeless dialog, 282 member variable initialization, 491

LOCALE clipboard format INDEX message handler functions in CACTLProp, 549 MFC console application, 236 MFC dialog class, 238 MFC gettime: The CGTSocket class, 694 MFC gettime:CGTDlg::OnConnect, 693 modeless version of CMyDialog, 281 MSG structure, 39 multimedia playback application, 750-751 named pipe client, 735 named pipe server, 734-735 nesting exception handlers, 222-223 new view class member functions, 464-466 ODBC console application, 587 OLE DB example, 629 OLE messaging example in Visual Basic, 676-677 OpenTabDialog implementation file, 820 palette animation, 126-129 processing a hotkey in CTRL, 308 processing in a secondary thread, 149-150 processing loop example source file, 146-147 raising C++ exceptions, 229-230 Registry entries for automation server, 435 resource allocation problem, 224 resource file for clipboardaware prgrm, 202 resource script example, 98 retrieving files from FTP server, 392 Gopher server, 393 HTTP server, 394 RPC client, 742 RPC server, 740 saving a list using a CArchive, 361 schema.ini file created by ODBC setup, 596 server communication using shared memory, 172 set_se_translator function, 231 setlocale function, 871 SetThreadLocale function, 872 SetupFolders function (InstallShield), 841 shell extensions, DLL RESOURCE.H, 853 RESOURCE.H, 853 WMFPRPSH.CPP, 856-861 WMFPRPSH.DEF, 863 WMFPRPSH.H, 855-856 WMFPRPSH.MAK, 863 WMFPRPSH.RC, 854 Registry entries, 864 simple clipboard-aware application, 199, 202 Simple MAPI in a console application, 672 simple message loop, 145 simple OpenGL application, 772-774 simple Registry reader, 214-217 simplest TAPI application, 712 subclassing in a DLL, 78-79 subclassing the BUTTON class, 76-77 superclassing the BUTTON class, 80-81 tab control initialization in CTRL, 314 TAPI data communication program, 723, 726 termination handlers, 225 termination handling with C, 227 Unicode generic mappings, 875 simple program, 874 Windows application, 876 variable initialization in the CASRVDoc, 497 Visual Basic code used for testing ASRV, 505 window procedure skeleton, 68 lists collection classes CObList, 369-371 CPtrList, 373 CStringList, 373-374 loading, CArchive, 362 saving, CArchive, 361-362 little-endian byte ordering, 687 LoadBitmap function, 129 loading bitmaps, 129 lists, CArchive, 362 LoadLibrary function, compiling DLLs, 7 local loops, 713 local servers, 425 LOCALE clipboard format, 195

925

926

locales UNLEASHED locales, 871-872 international applications, 870 Location tab (DCOMCNFG.EXE), 574 locking files, 355 LockRange member function, 355 LOENGLISH mapping mode, 115, 339 LOGBRUSH structure, 348 logical addresses, 158 logical coordinates, 45, 113 constrained mapping, 114-115 mapping, 339-341 unconstrained mapping, 113-114 World Coordinate Transforms, 115-122 invertng, 121-122 setting, 120 logical fonts, 124-125 logical palettes, 125 LOGPEN structure, 347 LOMETRIC mapping mode, 115, 339 LongPathToQuote function, 840 loops message loops, 51-53 multiple, 57, 60 multitasking, 145 threads, 41 low-level I/O operations, 187-188 file descriptors, 187 fuctions, 187-188 lseek function, 187 LTEXT control statement, 103

M
m_pSelection member variable, 463 macros exception handling, 398-399 help files, 811-812 MCIWnd class, 756-758 serialization, 363 strings, 812 main function, compiling EXEs/DLLs, 6 MAIN.CPP listing, 644-646 make utility, 4 makefiles, 4 malloc, 164 managing files, CFile member functions, 354 memory address spaces, 158-159 malloc, 164 memory-mapped files, 161, 171-173 runtime memory, 171 shared memory, 159, 165, 171-173 stray pointers, 165 swap files, 160-161 threads, 174 virtual memory, 166-171 processes, 151-152 windows extended styles, 70 registration, 67-69 window creation, 69-70 manual-reset events, 409 Map button (Microsoft Help Workshop), 816 MAPI (Messaging API), 46, 666 applications, 668 architecture, 666-667

CMC, 673-674 Extended MAPI, 675 MFC support, 677-678 OLE messaging, 675-677 profiles, 669 service providers, 668-669 Simple MAPI, 670-673 MAPIAddress function, 673 MAPISendDocuments function, 670-671 MAPISendMail function, 672 mapping, 375 logical coordinates constrained mapping, 114-115 unconstrained mappng, 113-114 CMapPtrToPtr class, 378 CMapPtrToWord class, 379 CMapStringToOb class, 377-378 CMapStringToPtr class, 378 CMapStringToString class, 375-377 CMapWordToOb class, 379 CMapWordToPtr class, 379 marshaling, 421, 736 MaskBlt function, 136 matrices (transformations), 118-119 MBCS (multiple-byte character sets), 873 MCI (Media Control Interface), 753, 759-761 commands sets, 762 syntax, 761-762 functions, 762-763 macros, 762-763 multimedia devices, 759

metafiles INDEX notification messages, 763 video playback, 760-761 mciGetCreatorTask function, 762 mciGetDeviceID function, 762 mciSendCommand function, 762 mciSendString function, 762 MCIWnd window class, 753-754 functions, 755-756 macros, 756-758 messages, 756-758 notification messages, 758-759 user interface, 754 MCIWndCreate function, 750-752, 755 MCIWndRegisterClass function, 755 MDI (multiple-document interface) applications classes, 18-19 database support, 15 MFC AppWizard, 17-19 OLE support, 15 project files, 15 user services, 44 Media Build Wizard, 842-844 Media Control Interface, see MCI media modes, 722 Media Name dialog box, 842 media streams, 712 member functions, see functions member variables ActiveX controls, 326-328 adding to dialogs, 277-278 editing, ClassWizard, 22-25 memory 32-bit applications, 163 allocating COM, 419-420 malloc, 164 ODBC, 585 shared memory, 165 stray pointers, 165 device contexts, 112 heaps, 170 management address spaces, 158-159 memory-mapped files, 161, 171-173 swap files, 160-161 managing runtime memory, 171 threads, 174 physical, accessing, 175 shared, 159 based pointers, 173 memory-mapped files, 171-173 virtual committed pages, 166 functions, 167-171 guard pages, 167 managing, 166-167 reserved pages, 166 memory-mapped files, 161, 171-173 MENU option (DIALOG statement), 102 menu statements, 104 MENUITEM keyword, 104 menus, OLE container applications, 466-467 message boxes, 82-83 message maps, 21-22, 243-245 message spy, 11 message stores, MAPI, 668 MessageBox function, 51, 82 Unicode applications, 875 messages (Windows), 36 ActiveX controls, 329 clipboard, 198 defining, 40 handling, dialogs, 285-286 lengthy processing, 145-148 loops, 51-53 multiple, 57, 60 multitasking, 145 MCIWnd class, 756-758 multitasking, 40-41 notification MCI, 763 MCIWnd class, 758-759 palette-related, 126 processes, 40 queues, 40 routing, 244 structure, 39 threads, 40 types, 38 window management, 72-74 PAINT, 71 WM messages, 39 MESSAGETABLE statement, 100 Messaging API, see MAPI messaging-aware applications, 668 messaging-based applications, 668 messaging-enabled applications, 668 metafile, device contexts, 112 metafile clipboard formats, 195 metafile device contexts, 45, 337-338 METAFILEPICT clipboard format, 195 metafiles, 45 Aldus placeable, 853 property pages, 853-864

927

928

METHODDATA structure UNLEASHED METHODDATA structure, 418 methods ActiveX controls, 528 CreateView, 561 get_all (HTML), 656 IOleDocumentSite interface, 565 IUnknown, 786 methods, see functions MFC (Microsoft Foundation Classes) applications, 236 AppWizard, 14 Activex document server, 567 dialog boxes, Advanced Options, 16 dialog-based projects, creating, 19 DLLs, creating, 20 SDI/MDI projects, creating, 15-19 CArchive class creating objects, 358-359 error handling, 361 overloaded operators, 359-360 reading objects, 359-360 sample application, 361-362 writing to objects, 359-360 CAsynchSocket class, 692-695 sample application, 692-694 synchronous operations, 694 CCmdTarget class, 265-266 CCriticalSection class, 409 CDialog class, 274 CDocItem class, 265-266 CDocument class data elements, 262-264 declaring, 258-259 member functions, 259-260 message handling, 261 overridable functions, 261-262 CEvent class, 409 CFile class CInternetFile subclass, 356 CMemFile subclass, 356-357 COleStreamFile subclass, 357 CSocketFile subclass, 358 error handling, 355 file locking, 355 file management, 354 hierarchy, 352 initialization, 354 object associations, 356 reading from/writing to objects, 354 sample console application, 355-356 CMultiLockclass, 410 CMutex class, 409 CObject class, 368 COleDocument class, 444 collection classes CObArray, 371-373 CObList, 369-371 CPtrArray, 374 CPtrList, 373 CStringArray, 375 CStringList, 373-374 CUIntArray, 374 helper functions, 380-381 integral array classes, 374-375 common controls CAnimateCtrl class, 306 CHeaderCtrl class, 307 CHotKeyCtrl class, 307-308 CListCtrl class, 309-310 CProgressCtrl class, 311 CRichTextCtrl class, 311-312 CSliderCtrl class, 312-313 CSpinButtonCtrl class, 313 CStatusBarCtrl class, 313-314 CTabCtrl class, 314-315 CToolBarCtrl class, 315-316 CToolTipCtrl class, 316 CTreeCtrl class, 316-317 common dialogs, 298-299 CColorDialog, 299 CFileDialog, 299-300 CFindReplaceDialog, 300-301 CFontDialog, 302 COleDialog, 304 CPageSetupDialog, 302-303 CPrintDialog, 303-304 container applications, ActiveX, 566 CPropertyPage class, member functions, 291-292 CSemaphore class, 409 CSingleLockclass, 410 CSyncObject class, 408 CView class, 266 declaration, 267-268 derived classes, 270 member functions, 268-269 message handling, 269 device contexts, 334 attributes, 338

MFC (Microsoft Foundation Classes) INDEX CDC class, 335 client-area, 337 creating, 335-336 metafile, 337-338 paint-time, 336-337 window, 337 dialogs class construction, 275-277 CMyDialog class, 278-279 constructing, 274 Dialog Data Exchange, 283-285 invoking, 279-280 member variables, 277-278 message handling, 285-286 modal, 280 modeless, 280-282 property sheets, 286-294 templates, 275 exception handling, 398 CArchiveException class, 402 CDaoException class, 404 CDBException class, 404 CException class, 399-400 CFileException class, 401-402 CInternetException class, 404-405 CMemoryException class, 401 CNotSupportedException, 403 COleDispatchException, 405 COleException class, 405 CResourceException class, 403 CUserException class, 405 macros, 398-399 throwing exceptions, 405-406 foundation classes, 237 GDI support, 346-347 bitmaps, 348 brushes, 347-348 fonts, 348 palettes, 348-349 pens, 347 regions, 349 Hello World application, 238 MAPI support, 677-678 mappings CMapPtrToPtr class, 378 CMapPtrToWord class, 379 CMapStringToOb class, 377-378 CMapStringToPtr class, 378 CMapStringToString class, 375-377 CMapWordToOb class, 379 CMapWordToPtr class, 379 multiple-language resources, 880 ODBC applications, 593-594 customizing, 601-602 data sources, 594-596 RFX functions, 604 skeleton applications, 596-601 ODBC classes, 602 CDatabase, 603 CFieldExchange, 604 CRecordset, 603-604 CRecordView, 605 OLE server application, 440 AppWizard options, 441-442 customizing, 447-451 dialogs, 450-451 document items, 444, 448 drawing code, 448-450 frame windows, 445 modes of operation, 445-446 registering, 452-453 running, 447 serialization, 452 server items, 442-444 OpenGL applications, 776-780 serialization, 352, 362 documents, 363 helper macros, 363 OLE operations, 364 simple data type classes, 410-411 skeleton application application objects, 239-243 AppWizard options, 239 customizing, 253-255 documents, 248-249 frame window, 245-246 message maps, 243-245 resources, 252-253 string resources, 253 view window, 250-252 structure/support classes, 411-413 templates, 379-380 CArray, 382-383 CList, 381-382 CMap, 383-384 CTypedPtrArray, 385 CTypedPtrList, 384 CTypedPtrMap, 385

929

930

MFC (Microsoft Foundation Classes) UNLEASHED threads, 408-410 creating, 407-408 multithreading, 406 thread safety, 406-407 unicode, 876 Microsoft Help Workshop contents files, 818-820 file testing, 820-821 project editing, 813-814, 818 Microsoft Foundation Classes, see MFC Microsoft Interface Developmnt Language compiler, 737, 744-745 Microsoft Telephony API, see TAPI middle tier, 658 MIDI (Musical Instrument Digital Interface), 753 playback, 764 MIDL compiler (Microsoft Interface Dev. Language), 737, 744-745 midl_user_allocate function, 742 midl_user_free function, 742 mini-servers, 440 mixer devices (audio), 765 MM_ANISOTROPIC mapping mode, 115 MM_HIENGLISH mapping mode, 115 MM_HIMETRIC mapping mode, 115 MM_ISOTROPIC mapping mode, 115 MM_LOENGLISH mapping mode, 115 MM_LOMETRIC mapping mode, 115 MM_TEXT mapping mode, 115 MM_TWIPS mapping mode, 115 modal dialog boxes, 81-82, 280 modal property sheets, constructing, 291 modeless dialog boxes, 82, 280-282 constructing, 281 member functions, 282 modeless property sheets constructing, 292 implementing, 293-294 invoking, 294 modes (coordinate mapping), 339-340 monikers, 422 month calendar control, 310 MOVE window mangement message, 73 MoveToEx function, 134 MS-DOS Interrupt 21 functions, 47 MSG structure, 39 MTS application components, 660 programming, 661 scalability, 660 MULTI_SZ Registry value type, 207 multicasting (IP), 683 multilanguage resources, 877 help files, 885 installation, 885-886 satellite DLLs, 879 translating user interface, 878 versions, 879 multiline text, outputting, 344 multimedia, 750 audio, playback, 752 DirectX, 784-785 COM interfaces, 785-786 Direct3D, 788 DirectAnimation, 791 DirectDraw, 786-788 DirectInput, 791 DirectPlay, 790-791 DirectSetup, 792 DirectShow, 791 DirectSound, 788-790 sample application, 792, 797-799 formats, 752-753 interfaces, 753 ACM, 764 audio mixers, 765 AVIFile functions, 763 custom handlers, 763 DrawDib function, 763-764 PlaySound function, 764 VCM, 764 video capture, 764 video, playback, 750-752 multiple-document-based (MDI) applications, see MDI multiple ActiveX property pages, 533 multiple byte character sets (MBCS), 873 multiple message loops, 57, 60 multiple window procedures, 57, 60 multitasking, 140 context switching, 141 cooperative, 141-142 lengthy processing, 145-148 message loops, 145 message queues, 40 operating system support, 141

objects INDEX preemptive, 141-143 process scheduling, 141 processes, 40-41 creating, 151-152 managing, 151-152 terminating, 152 synchronization objects, 152-153 threads, 40-41, 142 exit code, 151 memory management, 174 secondary, 148-151 suspended states, 151 thread objects, 151 thread-local storage, 174 user interface, 143 worker, 143 Windows 95, 140, 143-144 multithreading, MFC thread creation, 407-408 thread safety, 406-407 thread synchronization, 408-410 Musical Instrument Digital Interface (MIDI), 753 mutex objects, 409 mutexes (mutual exclusion objects), 153 MyPage1 class, 289-290 creating, 732 data transfers, 734 sample client application, 735-736 sample server application, 734-735 names, hostnames, 684 namespace, 848 naming, Registry keys, 206 nationalization grammar, 868 language-specific info, 869 non-European languages, 870 text size, 867 user input, 869 navigating recordsets, 604, 620 window hierarchy, 65-66 ncacn RPC protocols, 741 NCACTIVATE window mangement message, 74 ncadg RPC protocols, 741 ncalrpc RPC protocol, 741 NCCREATE window mangement message, 74 NCDESTROY window mangement message, 74 NCPAINT window mangement message, 74 nesting exception handlers, 222-223 networks addresses, 683 protocols ARP, 681 DNS, 684 ICMP, 681 IP, 681 RARP, 681 TCP, 680 TCP/IP, 680-685 UDP, 681 New ATL Object command (Insert menu), 542 New Class dialog box, 27, 275 New command (File menu), 13 New dialog box, 13 New Project Information dialog box, 19 New Technology File System (NTFS), 178 NewWindow function, 509 NFC, OpenGL applications initializing, 777-778 running, 780 nmake utility, 4 NONE Registry value type, 207 notification messages MCI, 763 MCIWnd class, 758-759 NOTIFYERROR message (MCIWnd), 759 NOTIFYMEDIA message (MCIWnd), 759 NOTIFYMODE message (MCIWnd), 759 NTFS (New Technology File System), 178

931

O
Object command (Insert menu), 447 object file, 4 object linking and embedding, see OLE objects ADO, 633 Commands, 634 Connection, 634 Error, 634 Fields, 635 Parameter, 634 Property, 634 Recordset, 634

N
Name property (application objects), 508 Name property (document objects), 509 named pipes connections breaking, 734 establishing, 733 multiple, 734

932

objects UNLEASHED sample application, 635-637 application, 507-508 CArchive, 238 constructing, 358 creating, 358-359 reading, 359-360 writing to, 359-360 CClientDC, 236 CFile, 352-354 COleStreamFile, constructing, 357 coordinates logical, 45, 113-115 mapping, 113-115 physical, 113 World Coordinate Transforms, 115-122 critical section, 409 current selection, 267 DAOs, 608, 619 CDaoDatabase class, 621 CDaoException class, 622 CDaoFieldExchange class, 622 CDaoQueryDef class, 622 CDaoRecordset class, 620-621 CDaoTableDef class, 622 CDaoWorkspace class, 621-622 overview, 608 sample application, 608-619 dialogs classes, 275-277 CMyDialog class, 278-279 constructing, 274 Dialog Data Exchange, 283-285 invoking, 279-280 member variables, 277-278 message handling, 285-286 modal, 280 modeless, 280-282 property sheets, 286-294 templates, 275 documents data elements, 262-264 declaring, 258-259 message handling, 261 methods, 508-509 modifying, 254 overridable functions, 261-262 paths, 260 properties, 508-509 views, 260 drawing, 122-123 embedded, 424 events, 409 files, 179 asynchronous I/O operations, 181-186 closing, 180 console I/O operations, 189-190 creating, 180 low-level I/O operations, 187-188 opening, 180 simple I/O operations, 180-181 stream I/O operations, 188-189 GDI bitmaps, 129, 135-136 brushes, 123-124 creating, 122 deleting, 122 fonts, 124-125 MFC support, 347-349 palettes, 125-129 pens, 123 selecting, 341-342 handles, 43 inter-object communication, 421-422 linked, 424 mutexes, 409 OLE DB commands, 627 data source objects, 627 enumerators, 628 errors, 628 rowsets, 627 sample console application, 628, 632 sessions, 627 transactions, 628 property pages, constructing, 287-290 proxy objects, 421 QueryDef, 620 reusing, COM, 420 semaphores, 409 serialization, 238 signaled, 152 synchronization, 152-153 TableDef, 620 threads, 151 objects collection, 509 obsolete functions, 171 OCON application, 479 containers, 456, 475 customizing, 468 drawing functions, 471 menus, 466-467 object positions, 469-471 object selection, 472-474 skeleton application, 456-466 drag sources, 483-484 drop target, 484-491 positioning support, 479-481 selection support, 482 serialization, 480

OLE servers INDEX ODBC (Open Database Connectivity), 582 APIs, 583-586 connections establishing, 585 terminating, 586 core grammar, 584 data sources, 582 database classes, Unicode, 876 environment allocation, 584 functions, 589-590 memory allocation, 585 MFC applications, 593-594 customizing, 601-602 data sources, 594-596 skeleton applications, 596-601 MFC classes, 602-605 CDatabase, 603 CFieldExchange, 604 CRecordset, 603-604 CRecordView, 605 RFX functions, 604 sample console application, 586-589 setup applet, 582-583 ODBC Text Setup dialog box, 594 OEMTEXT clipboard format, 194 OffsetRgn function, 135 OLE (object linking and embedding) ActiveX, 426 bitmaps, 524 code overview, 515-520, 523 creating, 512-513 distributing, 534 drawing, 530-531 events, 528-529 methods, 528 properties, 524-528 property page interfaces, 531-533 skeleton controls, 513-515 testing, 534 usage tips, 535 automation, 425-426, 494 servers, 494-499 standard objects, 506-509 supporting, 499-503 testing, 504-505 type library, 503-504 COM automation server application, 426-437 class objects, 421 inheritance, 420 inter-object communication, 421-422 interface definition, 420-421 interface identifiers, 420 interfaces, 418-419 memory allocation, 419-420 monikers, 422 common dialogs, 91, 304 compared to ActiveX documents, 560 compound documents, 422 data transfers, 423-424 embedded objects, 424 in-place activation, 425 linked objects, 424 structured storage, 423 containers, 425, 456, 475 customizing, 468-474 drawing functions, 471 embedded objects, 474 member functions, 460-466 menus, 466-467 object positions, 469-471 object selection, 472-474 skeleton application, 456-460 view class, 464-466 DB, 626-627 commands, 627 data source objects, 627 enumerators, 628 errors, 628 rowsets, 627 sample console application, 628, 632 SDK, 627 sessions, 627 transactions, 628 drag and drop, 426, 478 container application, 478-482 drag sources, 483-484 drop targets, 484-491 supporting, 482-483 interfaces, customizing, 426 messaging, 675-677 serialization, 364 servers, 425, 440 threads, 422 OLE servers converting to support ActiveX documents, 569 dialogs, 450-451 documents, modifying, 448 drawing code, 448-450 skeleton applications, 440 skeleton servers AppWizard options, 441-442 customizing, 447-451 document items, 444 frame windows, 445 modes of operation, 445 operation modes, 446 registering, 452-453

933

934

OLE servers UNLEASHED running, 447 serialization, 452 server items, 442-444 OnApply function, 292 OnCancel function, 292 OnCancelEditCntr function, 466 OnChangeItemPosition function, 461-462, 469 OnCloseDocument function, 262 OnCreate function, 486 OnDragEnter function, 485-487 OnDragLeave function, 485, 488 OnDragOver function, 485-488 OnDraw function, 463-464, 471-473 OnDraw member function, 778 OnDrop function, 485, 488-489 OnFileSendMail function, 262, 677 OnGetItemPosition function, 470 OnInsertObject function, 466 OnLButtonDown function, 269 OnNewDocument function, 253, 261 OnOpenDocument function, 262 OnPrepareDC function, 269, 341 OnSaveDocument function, 262 OnSetFocus function, 466 OnSize function, 466 OnViewDialog function, 280 OnViewPropertySheet function, 291 OnViewPropertysheet function, 291, 294 Open Database Connectivity, see ODBC Open function, 354 open function, 187 Open Systems Interconnect (OSI) standard, 680 OpenClipboard function, 197 OpenFile function, 180 OpenGL, 768 applications C, 772-776 compiling, 776 MFC, 776-780 drawing operations, 771 GLAUX, 772 GLU, 772 initialization, 770-771 C applications, 775 MFC applications, 777-778 MFC applications, running, 780 overview, 768-769 opening file objects, 180 files, 354 Registry keys, 212-213 OpenSCManager function, 642 OpenTabDialog implementation file, 820 operating systems, 48 Windows 95, 36 optimization, disabling, 29 Options button (Microsoft Help Workshop), 813 [OPTIONS] section (help project files), 809 OSI (Open Systems Interconnect) standard, 680 overlapped I/O, communications program, 184-186 OWNERDISPLAY clipboard format, 195 ownership call ownership (TAPI), 722 windows, 65

P
p dial modifier character, 715 packets, TCP, 684 Page Setup common dialog, 88 Page Setup common dialogs, 302-303 page tables, 160 PageSetupDlg function, 88 paint-time device contexts, 336-337 painting, 70-71 PaintRgn function, 343 PALETTE clipboard format, 195 palettes animation, 126-129 color, 125 default, 125 deleting, 125 logical, 125 MFC support, 348-349 palette-realted messages, 126 realizing, 125-126 selecting, 125 system, 125 Parameter objects (ADO), 634 pasting data from Clipboard, 197-198

Project Settings dialog box INDEX PatBlt function, 136, 343 path brackets, 137 path functions, 345-346 Path property (application objects), 508 Path property (document objects), 509 paths, 136-137 PathToRegion function, 137 pattern brushes, creating, 124 PeekMessage function, 148 PENDATA clipboard format, 195 pens, 123 MFC support, 347 pentagons, drawing, 771 persistence ActiveX control properties, 325 properties, 527-528 phone devices, 714 physical addresses, 158 physical coordinates, 113 world coordinates, 120 physical memory, accessing, 175 pict command, 809 pipes anonymous, 732 creating, 732-733 data transfers, 734 named, 732 connecting to, 733-734 sample client application, 735-736 sample server application, 734-735 pixels, converting dialog units to, 102 placeable metafiles, 853 platforms differences, 48 Windows 95, 36 Platforms dialog box, 844 Playback function, 752 PlayEnhMetaFile function, 112 playing audio, 752 video, 750-752 PlayMetaFile function, 112 PlaySound function, 764 PlgBlt function, 136 pointers based, 173 file pointers, 181 strays, 165 PolyBezier function, 135 PolyBezierTo function, 135 polygons, drawing, 135 Polyline function, 134 polylines, drawing, 134, 342 PolylineTo function, 134 PolyPolygon function, 135 PolyPolyline function, 134 PolyTextOut function, 137 POPUP keyword, 104 port monitors, 137 port numbers, 685 ports, 191 POSITION type, 369-371 PostNcDestroy function, 292 PostQuitMessage function, 52-55 PreCreateWindow member function, 777 predefined control classes, 69 predefined Registry keys HKEY CLASSES ROOT, 207, 210-211 HKEY CURRENT CONFIG, 208 HKEY CURRENT USERS, 211-212 HKEY DYN DATA, 208 HKEY LOCAL MACHINE, 207-210 HKEY USERS, 208, 211 opening, 212-213 preemptive multitasking, 141-143 preprocessor, 4 directives, 99-100 primitives, drawing, 771 Print common dialog, 87-88 Print function, 509 print spoolers, 137 PrintDlg function, 87 printer escapes, 138 printing CDC class functions, 345 common dialogs, 303-304 documents, 509 GDI functions, 137-138 print devices, 137 PrintOut function, 509 PrintPreview function, 509 private clipboard formats, 196 private device contexts, 111 PRIVATEFIRST clipboard format, 195 PRIVATELAST clipboard format, 195 probing, 722 procedures, window procedures, 53-56 multiple, 57, 60 Process Exploder, 11 processes creating, 151-152 managing, 151-152 multitasking, 40-41 terminating, 152 profilers, 12 profiles (MAPI), 669 progress bars, 93, 311 Project Settings dialog box, 29, 542 Unicode applications, 877

935

936

Project Wizard (InstallShield) UNLEASHED Project Wizard (InstallShield) Choose Dialogs dialog box, 831 Choose Target Platform dialog box, 832 Specify Components dialog box, 834 Specify File Groups dialog box, 834 Specify Languages dialog box, 833 Summary dialog box, 834 Project Wizard (InstallShield), 831-835 projects, 10 components, inserting, 28 creating with AppWizards, 13 dialog-based, 15 creating, 19 DLLs, creating, 20 foreign translations, 878 help files, 809-810 editing, 813-815, 818 makefiles, 5 MDI classes, 18-19 creating, 15-19 database support, 15 OLE support, 15 project files, 15 SDI classes, 18-19 creating, 15-19 database support, 15 OLE support, 15 project files, 15 templates, 10 workspaces, 10 Projects tab (New dialog box), 13 properties ActiveX controls, 325-326, 524 accessing, 329 adding, 526-527 ATL, 543-545 persistence, 325, 527-528 collection objects, 507 standard objects, 507-508 Property objects (ADO), 634 property pages ActiveX ATL, 547-550 adding, 531 connecting with properties, 531-533 editing, 531 multiple, 533 adding to shell, 853-864 property sheets, 286 constructing modal, 291 pages, 287-290 member functions, 291-292 modeless, 292-294 protected mode, 164 Protected Mode FAT (VFAT) file system, 178 protocols, 700 ARP, 681 DNS, 684 FTP, 390, 701-702 file transfers, 706-707 Gopher, 390-391, 702-704 file transfers, 707-708 HTTP, 391, 704-705 file transfers, 708-709 ICMP, 681 IP, 681 datagrams, 682 headers, 682 host addresses, 683 hostnames, 684 multicasting, 683 routing, 683 RARP, 681 RPC, 741 TCP, 680 packets, 684 TCP/IP OSI model, 680 resources, 695 UDP, 681 Proxy Generator (ATL), 552 proxy objects, 421 PtInRegion function, 135 PurgeComm function, 191 pushbuttons, 92 PVIEW.EXE, 11 PX_ functions, 527-528

Q
queries, definitions, 622 QueryDef (query definition) objects, 620 QUERYENDSESSION window mangement message, 72 querying, Registry values, 213 QueryInterface function, 421, 786 queues, 40 Quit function, 508 QUIT window mangement message, 72

R
radio buttons, 92 RaiseException function, 224 RARP (Reverse Address Resolution Protocol), 681 raw-data strings, 106 rc filename extension, 98 RC_INVOKED symbol, 99

reporting errors (functions) INDEX RCDATA resources, 106 read function, 187 ReadConsole function, 190 ReadFile function, 190 reading CArchive objects, 359-360 files, 354 ReadOnly property (document objects), 509 RealizePalette function, 125 realizing, palettes, 125-126 reconcilers (Briefcase), 850 Recordset objects (ADO), 634 recordsets, 620-621 attributes, 604 creating, 620 dynasets, 603 navigating, 604, 620 snapshots, 603 Rectangle function, 111 rectangles, drawing, 111 RectInRegion function, 135 reentrancy, 141 referencing databases, ADO application, 635-637 reflection, 118 RegCloseKey function, 213 RegCreateKeyEx function, 213 RegDeleteKey function, 214 regedit.exe program, 208 regedit32.exe program, 208 RegEnumKeyEx function, 214 regions, 135 clipping regions, 129-130 MFC support, 349 RegisterClass function, 67-69 RegisterClipboardFormat function, 196 registered clipboard formats, 196 registering classes, 421 installation programs, 829 OLE server application, 452-453 windows, 67-69 RegisterWindowMessage function, 40 Registry, 206 ActiveX document server, 569 automation server entries, 435-436 capacity, 207 DCOM servers, client workstations, 572 editing, manually, 208-209 Editor, 208 functions, 213-214 INI files, compared, 212 keys, 206-207 creating, 213-214 deleting, 214 HKEY CLASSES ROOT, 207, 210-211 HKEY CURRENT CONFIG, 208 HKEY CURRENT USER, 211-212 HKEY DYN DATA, 208 HKEY LOCAL MACHINE, 207-210 HKEY USERS, 208, 211 names, 206 opening, 212-213 predefined, 208 MAPI profiles, 669 OLE server entries, 452-453 reader program, 214-218 shell extension DLL entries, 864 structure, 206 values names, 206 querying, 213 setting, 213 types, 207 Windows 95, 206 Windows NT, 206 RegOpenKeyEx function, 213 RegQueryValueEx function, 213 RegSetKeySecurity function, 214 RegSetValueEx function, 213 ReleaseMutex function, 153 ReleaseSemaphore function, 152 Remote Procedure Calls, see RPCs RemoveAt function, 372 RemoveView function, 260 removing fonts, 125 list elements, 370 views, 260 Windows NT services, 651 RENDERALLFORMATS messages, 197 repainting window contents, 71-72 RepairDatabase function, 622 repairing databases, 622 Repeat function, 508 Replace common dialog, 88 ReplaceText function, 88 reporting errors (functions), 47

937

938

Requests For Comments UNLEASHED Requests For Comments, 695-696 res filename extension, 98 reserved pages, 166 reserved system arenas, 159 ResetEvent function, 152 resource allocation problem, sample listing, 224 resource dispensers, 661 resource files compiling, 107 DLLs, 107-108 linking, 107 preprocessor directives, 99-100 raw data resources, 106 sample script, 98-99 statements accelerators, 101 BITMAP, 100 CURSOR, 100 dialogs, 102-103 FONT, 100 ICON, 100 LANGUAGE, 100 menus, 104 MESSAGETABLE, 100 string tables, 104-105 toolbars, 105-106 version information, 106 user-defined resources, 106 resource localization, NT/ 95, 879-885 resource managers, 660 RESOURCE.H listing, 853 RESOURCE_LIST Registry value type, 207 resources editing, 9 language, changing, 885 ResumeThread function, 151 resuming suspended threads, 151 Retained Mode (Direct3D), 788 reusing objects, COM, 420 Reverse Address Resolution Protocol (RARP), 681 RevertToSaved function, 509 revision control systems, 12 REVOKE statements, 593 RFCs, 695-696 RFX functions, 604 RGNDATA structure, 349 Rich Text Format (RTF), RTF control, 311-312 rich-text edit controls, 93 RIFF clipboard format, 195 root domains, 684 rotation, 116 RoundRect function, 135 routing (IP), 244, 683 rowsets, OLE DB, 627 RPCs (Remote Procedure Calls), 736-737 example of, 737 client application, 742-743 interface, 738-740 server application, 740-742 exception handling, 743-744 MIDL compiler, 737, 744-745 protocols, 741 stub functions, 736 RpcServerUseProtseqEp function, 740 RpcStringBindingCompose function, 743 RTEXT control statement, 103 RTF (Rich Text Format) help files, 806-809 RTF control, 311-312 running container skeleton application, 459-460 server skeleton application, 447 runtime memory, managing, 171

S
satellite DLLs, multilanguage applications, 879 Save function, 509 SaveAs function, 509 Saved property (document objects), 509 saving documents, 509 lists, CArchive, 361-362 SBCS (single-byte character sets, 873 scalability, MTS, 660 scaling, 116 schema numbers, 359 SCROLLBAR control statement, 103 scrollbars, 93 ScrollDC function, 344 SDI applications (singledocument interface) classes, 18-19 database support, 15 MFC AppWizard, 17-19 OLE support, 15 project files, 15 SDKs OLE DB, 627 Service example, 642-643 searched updates, 592 secondary threads, 148-151

SetFilePointer function INDEX Security tab (DCOMCNFG.EXE), 574 Seek function, 354 select function, 689 SELECT statement, 589, 591-592 SelectClipPath function, 137, 346 SelectClipRgn function, 130 selecting GDI objects, 341-342 SelectObject function, 122 selector functions, 32-bit applications, 163 SelectPalette function, 125 SelectTAPIDevice functions, 727 semaphores, 152, 409 Send a Macro command (Test menu), 821 SEPARATOR keyword, 104 serialization, 238, 352, 362, 695 documents, 363 helper macros, 363 MFC support, 264 OCON application, 480 OLE operations, 364 server skeleton application, 452 Serialize member function, 360, 470 SerializeElements function, 264, 381 server tier, 658 servers, 425 activation, 440 ActiveX documents, 561, 567-569 applications, resource managers, 660 automation servers, 426-427, 494 accessing, 436-437 application skeleton, 495 calculators, 495-499 functional description, 427 Registry entries, 435-436 source code, 428-435 standard objects, 506-509 supporting, 499-503 testing, 504-505 type libraries, 503-504 user interfaces, 494-495 dialogs, adding, 450-451 document items, modifying, 448 drawing code, adding, 448-450 FTP transfers, 392-393, 706-707 full-servers, 440 Gopher transfers, 394, 707-708 HTTP transfers, 394-395, 708-709 in-place editing, 440 in-process, 425, 440 local, 425 mini-servers, 440 named pipe servers, 734-735 RPC server application, 740-742 skeleton applications, 440 AppWizard options, 441-442 customizing, 447-451 document items, 444 frame windows, 445 modes of operation, 445-446 registering, 452-453 running, 447 serialization, 452 server items, 442-444 Service Control Manager, 640 service providers (MAPI), 668-669 services (NT), 640 compiling, 651 creating architecture, 643-644 CSlot class, 649-651 CSrv class, 647-649 main service module, 644-647 SDK Service example, 642-643 installing, 651 removing, 651 running, 651 Service Control Manager, 640 starting, 641 stopping, 641 Win32 API, 641-642 Services applet, 640 session objects database sessions, 621-622 OLE DB, 627 set_se_translator function, 230-231 SetAbortDoc function, 345 SetBkColor function, 124, 338 SetBkMode function, 124 SetBrushOrgEx function, 124 SetClassLong function, 38, 77 SetClipboardData function, 194 SetCommBreak function, 191 SetCommMask function, 186 SetConsoleMode function, 190 SetDIBitsToDevice function, 136 SetFilePointer function, 181

939

940

SETFOCUS window mangement message UNLEASHED SETFOCUS window mangement message, 73 SetLastError function, 47 SetLocale function, 870 SetMenus function, 203 SetModified function, 292 SetModifiedFlag function, 260 SetNamedPipeHandleState function, 733 SetPixel function, 136, 343 SetPixelFormat function, 770 SetPolyFillMode function, 338 SetROP2 function, 338 SetSize function, 372 SetStatus member function, 354 SETTEXT window mangement message, 73 SetTextColor function, 344 SetTextJustification function, 344 SetThreadLocale function, 872 setting Registry values, 213 Settings command (Build menu), 29 SetTitle function, 260 Setup programs, ODBC, 582-583 setup scripts, installation programs, 840-841 SetupComm function, 191 SetupFolders function, 840-841 SetViewportExtEx function, 113 SetViewportOrgEx function, 113 SetWindowExtEx function, 113 SetWindowLong function, 38, 75 SetWindowOrgEx function, 113 SetWindowsHookEx function, 79 SetWizardMode function, 292 SetWorldTransform function, 120 shapes cubes, drawing, 778-780 drawing, 135, 342-343 filling, 135 pentagons, drawing, 771 rectangles, drawing, 111 SHAppBarMessage function, 848 shared arenas, 159 shared memory, 159, 165 based pointers, 173 memory-mapped files, 171-173 shearing, 117 shell Appbars, 848 applets, 850 Briefcase reconcilers, 850 extension DLLs, 849-850 Regsitry entries, 864 RESOURCE.H, 853 WMFPRPSH.CPP, 856-861 WMFPRPSH.DEF, 863 WMFPRPSH.H, 855-856 WMFPRPSH.MAK, 863 WMFPRPSH.RC, 854 file viewers, 849 namespace, 848 property pages, 853-864 taskbar, 848 icons installation, 851-852 Shell Extension DLL::RESOURCE.H (listing), 853 Shell_NotifyIcon function, 848 SHOWWINDOW window mangement message, 73 Shutdown function, 741 simple data typesMFC classes, 410-411 Simple MAPI, 667, 670-673 console application, 672-673 functions MAPIAddress, 673 MAPIFindNext, 673 MAPIReadMail, 673 MAPISendDocuments, 670-672 MAPISendMail, 672 simple types, Dialog Data Exchange, 284 single-byte character sets (SBCS), 873 single-document-based (SDI) applications, 15 SIZE window mangement message, 73 sizing arrays, 372 bitmaps, 348 skeleton server application (OLE) AppWizard options, 441-442 customizing, 447-451 document items, 444 frame windows, 445 modes of operation, 445 operation modes, 446 registering, 452-453 running, 447 serialization, 452 server items, 442-444 slider controls, 93, 312-313 snapshots (recordsets), 603 sockaddr_in structure, 686 socket function, 686

statements INDEX sockets, 685 CAsynchSocket class sample application, 692-694 synchronous operatns, 694 WinSock, 685-686 associating, 686 asynchronous socket functions, 690 blocking, 689 byte ordering, 687 communication, 687-689 creating, 686 initialization, 686 name service, 687 RFCs, 696 sample application, 690-692 software InstallShield, 828-830 raising exceptions, 224 Software subkeys HKEY CURRENT USER, 212 HKEY LOCAL MACHINE, 210 solid brushes, 124 sound, see audio source code, visible text, 869 sparce matrices, virtual memory management, 167-169 Specify Components dialog box (Project Wizard), 834 Specify File Groups dialog box (Project Wizard), 834 Specify Languages dialog box (Project Wizard), 833 spin buttons, 93, 313 Spy menu commands, Find Window, 36 Spy++ application, 36 SQL (Structured Query Language), 591 DAO functions, 608 data definition statements, 593 data manipulation statements, 591-592 Embedded SQL, 584 SELECT statement, 589 statements, executing, 603 views, 592 SQLAllocConnect function, 585 SQLAllocEnv function, 584 SQLAllocStmt function, 585 SQLBindParameter function, 590 SQLBrowseConnect function, 589 SQLConnect function, 585 SQLDisconnect function, 586 SQLDriverConnect, 588 SQLExecDirect function, 589 SQLExecute function, 585 SQLExtendedFetch function, 590 SQLNativeSql function, 590 SQLParamOptions function, 590 SQLPrepare function, 585 SQLRowCount function, 590 SQLSetConnectOption function, 589 SQLSetPos function, 590 SQLSetStmtOption function, 589-590 SQLTransact function, 590 Stack overflow value (filter expressions), 224 standalone mode, 445 standard automation objects, 506 application objects, 507-508 document objects, 508-509 properties, 507 standard C/C++ library functions, 47 standard clipboard formats, 194-196 standard metafiles, 112 standard objects application, 507-508 documents, 508-509 StartServiceCtrlDispatcher function, 642 startup code, 6 statements BITMAP, 100 CURSOR, 100 FONT, 100 guard blocks, 221 ICON, 100 LANGUAGE, 100, 880 MESSAGETABLE, 100 multiline accelerators, 101 DIALOG, 102-103 MENU, 104 STRINGTABLE, 104-105 TOOLBAR, 105-106 version information, 106 SQL ALTER TABLE, 593 CREATE index, 593 CREATE TABLE, 593 CREATE VIEW, 592 DELETE, 592 DROP, 593 executing, 603 GRANT, 593 INSERT, 592

941

942

statements UNLEASHED REVOKE, 593 SELECT, 591-592 UPDATE, 592 see also keywords static controls, 92 status bars, 93, 313-314 StatusBar property (application objects), 508 STGMEDIUM structure, 424 stray pointers, 165 stream I/O, 188-189 StretchBlt function, 136, 343 StretchDIBits function, 136 string binding, 743 string keyword, 739 string resources, adding to applications, 253 strings tables, 104-105 Unicode, 874 STRINGTABLE keyword, 105 structure/support classes, 411-413 Structured Query Language, see SQL structured storage (compound documents), 423 stub functions, 736 STYLE option (DIALOG statement), 102 subclassing, 38 global, 78-79 window classes, 75-77 Subject property (document objects), 509 Summary dialog box (Project Wizard), 834 superclassing, 38, 80-81 Supplementary Telephony service, 720 suspended states (threads), 151 swap files, 160-161 SYLK clipboard format, 195 synchronous operations, 42, 694 objects, 152-154 TAPI, 716 threads, MFC, 408-410 system functions, Unicode versions, 874 system global classes, 75 system modals, 82 system palettes, 125 System subkey (HKEY LOCAL MACHINE), 210 SZ Registry value type, 207 architecture asynchronous operations, 716-717 synchronous operations, 716 variable-length structures, 718-719 devices, 714 media modes, 722 programming model, 720-721 services Assisted TAPI, 712-713 Basic Telephony, 719 Extended TAPI, 720 Supplementary Telephony, 720 tapiexe.exe program, 716 tapiGetLocationInfo function, 713 tapiRequestMakeCall function, 713 task modals, 82 taskbar, 848 icons, installing, 851-852 tasks, 40 TCP (Transfer Control Protocol), 680 packets, 684 TCP/IP (Transfer Control Protocol/Internet Protocol) host addresses, 683 hostnames, 684 Internet services, 685 IP datagrams, 682 IP headers, 682 multicasting, 683 OSI model, 680 resources, 695 routing, 683 TCP packets, 684 Telephony API, see TAPI telephony services, 713 tell function, 187

T
t, dial modifier character, 715 tab controls, 94, 314-315 TabbedTextOut function, 137, 344 TableDef (table definition) objects, 620 tables DAO sample database, 610 definitions, 622 strings, 104-105 tables of contents, help files, 810-811 editing, 818-820 tabs, setting, 344 Tabs button (Microsoft Help Workshop), 819 Tag File dialog box, 843 TAPI (Telephony API), 46, 712 addresses, 714-715 applications data communication example, 723-728 multiple, 722-723

transferring INDEX templates, 10 ATL, 538-539 advantages, 538-539 bitmaps, 554-555 control methods, 545-546 control properties, 543-545 drawing functions, 546-547 event handling, 550-554, 557 Object wizard, 542-543 property pages, 547-550 Proxy Generator, 552 skeleton project, 540-542 testing controls, 556 collection classes, 379-380 CArray, 382-383 CList, 381-382 CMap, 383-384 CTypedPtrArray, 385 CTypedPtrList, 384 CTypedPtrMap, 385 helper functions, 380-381 dialog templates, 83, 275 document templates creating, 243 strings, 252-253 terminating database connections, ODBC, 586 processes, 152 termination handling C, 224-226 C++, 226-227 Test Container, 556 Test menu commands, 821 testing ActiveX controls, 534, 556 automation servers, 504-505 help files, Microsoft Help Workshop, 820-821 installation programs, 844 text CDC class functions, 344-345 colors, 344 GDI output functions, 137 justifying, 344 multiline, outputting, 344 nationalization, 867 tabs, setting, 344 TEXT clipboard format, 194 text clipboard formats, 194 TEXT mapping mode, 115 TextOut function, 137, 344 threads, 36, 142 apartments, 422 exit code, 151 interlocked variable access, 174 memory management, 174 messages, 41 MFC creating, 407-408 multithreading, 406 synchronization, 408-410 thread safety, 406-407 multiple, communications program, 182-183 OLE, 422 processes, 40 secondary, 148-151 suspended states, 151 synchronization, 42 thread objects, 151 thread-local storage, 174 user interface, 143 user-interface, 407 variables, incrementing/ decrementing, 153 worker, 143 worker threads, 41, 407-408 three-state check boxes, 92 three-tiered client/server model, 657-659 tiers, 658 TIFF clipboard format, 195 Title property (document objects), 509 TLS Indexes, 174 TlsAlloc function, 174 TlsFree function, 174 toolbar controls, 94 TOOLBAR statement, 105-106 toolbars ActiveX controls, bitmaps, 554 Appbars, 848 CToolBarCtrl class, 315-316 CToolTipCtrl class, 316 tooltip controls, 94 Top property (application objects), 508 top-level domains, 684 topic aliases (help files), 817 trackbars, 312-313 tracking rectangles, 473 Transaction Server application components, 660 programming, 661 scalability, 660 transactions, OLE DB, 628 TransactNamedPipe function, 734 transferring data between applications, 423-424 pipes, 734 files FTP, 392-393, 706-707 Gopher, 394, 707-708 HTTP, 394-395, 708-709

943

944

transformations UNLEASHED transformations linear, combining, 119-120 matrix forms, 118-119 reflection, 118 rotation, 116 scaling, 116 shearing, 117 translation, 115 World Coordinate Transforms, 120 translating C exceptions, 230-231 translation, 115 Transmission Control Protocol./Internet Protocol, see TCP/IP Transmission Control Protocol, see TCP TransmitCommChar function, 191 transport providers, MAPI, 668 tree controls, 94, 316-317 TWIPS mapping mode, 115, 339 type library (ASRV server), 503-504 type modifiers, 32-bit applications, 162 Windows applications, 875-876 writing, 874 UNICODETEXT clipboard format, 194 UNIX, 40 UnlockRange member function, 355 unmarshal function, 736 unmarshaling, 421 UnrealizeObject function, 347 up-down controls, 313 update regions, 71 UPDATE statements, 592 UpdateAllViews function, 260 User Datagram Protocol (UDP), 681 user input, nationalization, 869 user interfaces multilanguage applications, translating, 878 resources, 43, 106 threads, 143, 407 user mode, 325 users, User module (Windows), 44 utilities, clipboard viewers, 198-199 UuidCreate function, 420 uuidgen.exe utility, 420 variable-length structures (TAPI), 718-719 variables adding to dialogs, 277-278 member variables ActiveX controls, 326-328 editing, 22-25 VCM (Video Compression Manager), 764 verbs, 440 version control systems, 12 version information resource files, 106 versions, multilanguage applications, 879 VFAT (Protected Mode FAT) file system, 178 video DirectX, 790-791 playback, 750-752 VCM, 764 video capture, 764 Video Compression Manager (VCM), 764 view windows, 250-252 CView class, 266 declaration, 267-268 derived classes, 270 member functions, 268-269 message handling, 269 modifying, 254-255 zoom factor, 267 viewers, Clipboard viewers, 198-199 viewports, 113 views, SQL, 592 virtual functions, 419 virtual keyword, 419 virtual memory allocating, 167 committed pages, 166 freeing, 167 functions, 167-171

U
UDP (User Datagram Protocol), 681 ul command, 807 unconstrained mapping mode, 113-114 Undo function, 508 Unicode, international applications, 873 console, 874 MFC, 876

V
v command, 807 validating data, Dialog Data Exchange, 283 values, Registry names, 206 querying, 213 setting, 213 types, 207

Windows 95 INDEX guard pages, 167 managing, 166-167 memory-mapped files, 161, 171-173 reserved pages, 166 swap files, 160-161 VirtualAlloc function, 167 VirtualFree function, 167 VirtualLock function, 167 VirtualProtect function, 167 VirtualQuery function, 167 Visible property (application objects), 508 volatile keyword, 151 volumes, 179 window classes, 38 window device contexts, 337 window management (WM) messages, 39, 72-74 Window Styles dialog box, 17 Window Support classes, 237 windows, 36, 64 controls, 91-94 creating, 69-70 CreateWindow, 55 dialog boxes common dialogs, 84-91 dialog box procedure, 83 dialog templates, 83 message boxes, 82-83 modal, 81-82 modeless, 82 extended styles, 70 frame windows, 245-246 OLE server application, 445 handles, 64 hierarchy, 64-66 management, 67 messages, 72-74 methods, 64 painting, 70-71 procedures, 53-56, 74-75 multiple, 57, 60 properties, 64 registering, 67-69 repainting, 71-72 threads, 36 view windows, 250-252 CView class, 266 declaring, 267-268 functions, 268-269 message handling, 269 modifying, 254-255 zoom factor, 267 window class global subclassing, 78-79 subclassing, 75-77 superclassing, 80-81 window procedure, 74-75 WNDCLASS structure, 67-69 Windows 3.1, multitasking, 142 Windows 95 APIs, 46 applications, 36 unicode, 875-876 Clipboard, 194-196 controls, 198 data transfers, 196-197 delayed rendering, 197 formats, 194-196 messages, 198 pasting from, 197-198 sample program, 199-203 SetClipboardData function, 194 viewers, 198-199 common controls, 93-94 Explorer Appbars, 848 applets, 850 Briefcase reconcilers, 850 extension DLLs, 849-850 file viewers, 849 namespace, 848 property pages, 853-864 taskbar, 848 taskbar icons, 851-852 file compression, 179 memory, address spaces, 159 function calls, 41-42 GDI services, 44-46 kernal services, 42-43 user services, 44

945

W
w, dial modifier character, 715 WaitCommEvent function, 191 WaitForMultipleObjects function, 186 WaitNamedPipe function, 734 WAV (waveform audio) format, 752 WAVE clipboard format, 195 waveform audio, 752 wcreat function, 187 Web pages DHTML, 654 three-tiered model, 659 wglCreateContext function, 770 WidenPath function, 137 Width property (application objects), 508 Win32 API, Windows NT services, 641-642 Window button (Microsoft Help Workshop), 814

946

Windows 95 UNLEASHED functions, error reporting, 47 messages, 36 defining, 40 loops, 51-53 multitasking, 40 processes, 40-41 queues, 40 routing, 244 structure, 39 threads, 40-41 types, 38 WM, 39 multitasking, 140, 143-144 Registry, 206 automation server entries, 435-436 capacity, 207 editing, 208-209 INI files, 212 keys, 206-212 MAPI profiles, 669 OLE server entries, 452-453 operations, 212-214 reader program, 214-218 resource localization, 879-885 Unicode, 874 values, 206-207 Windows NT file compression, 179 file systems, 178-179 PVIEW.EXE, 11 resource localization, 879-885 services, 640 architecture, 643-644 compiling, 651 creating, 642-651 CSlot class, 649-651 CSrv class, 647-649 installing, 651 main service module, 644-647 removing, 651 running, 651 SDK Service example, 642-643 Service Control Manager, 640 starting, 641 stopping, 641 Win32 API, 641-642 Unicode, 874 Windows Open Services Architecture (WOSA), 667 [WINDOWS] section (help project files), 809 WinHelp API command (Test menu), 821 WinHelp function, 812-813 WinInet library, 700, 705, 709 sessions establishing, 705-706 FTP, 706-707 Gopher, 707-708 HTTP, 708-709 WinMain function, 55 Unicode applications, 875 WinSock, 685-686 asynchronous socket functions, 690 blocking, 689 byte ordering, 687 initialization, 686 name service, 687 RFCs, 696 sample application, 690-692 sockets associating, 686 communication, 687-689 creating, 686 wizards AppWizards, MAPI support, 677 ClassWizard, 20-21 ActiveX Events tab, 26 Add Class function, 27 Automation tab, 25 Class Info tab, 26 Edit Code function, 27 invoking, 20 Member Variables tab, 22-25 Message Maps tab, 21-22 Media Build Wizard, 842-844 MFC AppWizard, 14 WM messages, see window management messages WM_DESTROYCLIPBOARD messages, 198 WM_NOTIFY messages, 863 WM_PAINT message, 71 WM_RENDERALLFORMATS messages, 197 WM_RENDERFORMAT messages, 197 WMFPRPSH.CPP listing, 856-861 WMFPRPSH.DEF listing, 863 WMFPRPSH.H listing, 855-856 WMFPRPSH.MAK listing, 863 WMFPRPSH.RC listing, 854 WNDCLASS structure, 67-69 wopen function, 187 worker threads, 41, 143, 407-408 workspaces, 10

zoom factor (views) INDEX World Coordinate Transforms, 115-122 inverting, 121-122 setting, 120 WOSA (Windows Open Services Architecture), 667 WriteConsole function, 190 WriteConsoleOutputAttribute function, 190 WriteProfileString function, 43 writing to CArchive objects, 359-360 files, 354 WSAAsyncSelect function, 690

947

Y-Z
YAH project application objects, 239-243 AppWizard options, 239 customizing, 253-255 document objects, 248-249 declaring, 248 implementing, 249 modifying, 253-254 message map, 243-245 resources, 252-253 windows frame window, 245-246 view window, 250-252 Yield function, 142 Z-order (windows), 64 zoom factor (views), 267

948

Dictionary Level 1 UNLEASHED

Resource Centers Books & Software Personal Bookshelf WWW Yellow Pages Online Learning Special Offers
Choose the online ebooks that you can view from your personal workspace on our site.
w

Get the best


information and learn about latest developments in:
s Design s Graphics and Multimedia s Enterprise Computing and DBMS

Site Search Industry News

About MCP

Site Map

Product Support

Turn to the Authoritative Encyclopedia of Computing


You'll find over 150 full text books online, hundreds of shareware/freeware applications, online computing classes and 10 computing resource centers full of expert advice from the editors and publishers of: Adobe Press BradyGAMES Cisco Press Hayden Books Lycos Press New Riders Que Que Education & Training Sams Publishing Waite Group Press Ziff-Davis Press

s General Internet Information s Operating Systems s Networking and Hardware s PC and Video Gaming s Productivity Applications s Programming s Web Programming and Administration

The Authoritative Encyclopedia of Computing

s Web Publishing

When youre looking for computing information, consult the authority. The Authoritative Encyclopedia of Computing at mcp.com.

You might also like