How to customize the common print dialog box in an MFC application (132909)



The information in this article applies to:

  • The Microsoft Foundation Classes (MFC), when used with:
    • Microsoft Visual C++ for Windows, 16-bit edition 1.0
    • Microsoft Visual C++ for Windows, 16-bit edition 1.5
    • Microsoft Visual C++ for Windows, 16-bit edition 1.51
    • Microsoft Visual C++ for Windows, 16-bit edition 1.52
    • Microsoft Visual C++, 32-bit Editions 2.0
    • Microsoft Visual C++, 32-bit Editions 2.1
    • Microsoft Visual C++, 32-bit Editions 2.2
    • Microsoft Visual C++, 32-bit Editions 4.0
    • Microsoft Visual C++, 32-bit Editions 4.1
    • Microsoft Visual C++, 32-bit Enterprise Edition 4.2
    • Microsoft Visual C++, 32-bit Enterprise Edition 5.0
    • Microsoft Visual C++, 32-bit Enterprise Edition 6.0
    • Microsoft Visual C++, 32-bit Professional Edition 4.2
    • Microsoft Visual C++, 32-bit Professional Edition 5.0
    • Microsoft Visual C++, 32-bit Professional Edition 6.0
    • Microsoft Visual C++, 32-bit Learning Edition 6.0
    • Microsoft Visual Studio .NET (2003), Professional Edition
    • Microsoft Visual Studio .NET (2003), Enterprise Developer Edition
    • Microsoft Visual Studio .NET (2002), Professional Edition
    • Microsoft Visual Studio .NET (2002), Enterprise Developer Edition

This article was previously published under Q132909

SUMMARY

This article explains how to modify and use the standard print dialog box in a typical MFC application. Customizing the print dialog box involves modifying the Microsoft Windows default common print dialog box template. When you customize the common print dialog box, you can add new controls and remove existing controls.

MORE INFORMATION

In some situations, you may find it necessary to customize the standard print dialog box. This involves modifying the existing print dialog box template and modifying the view implementation of printing to use this customized template.

Step-by-Step Procedure

To implement a customized print dialog box, follow these steps:
  1. To copy the PRINTDLGORD dialog box template (the ID is 1538) from Commdlg.rc to your application .rc file, follow these steps.

    Note In Visual C++ 4.x and 5.0, this dialog box template resides in the file Include\Prnsetup.dlg. In Visual C++ 6.0, this file is located in the \Samples\Vc98\Mfc\General\Clipart on the MSDN CD. In Microsoft Visual C++ .NET, this file is located in the \Samples\Vc\Mfc\General\Clipart folder on the MSDN CD.
    1. Open Commdlg.rc.
    2. Enable cut-and-pasting of resources in your project resource file:
      1. If you are using Visual C++ .NET, open your project .rc file in Resource Editor mode (instead of the default Resource View mode). To do this, close the Resource View window of your project. Find your project .rc file in the Solution Explorer window. Right-click the .rc file in the Solution Explorer window, and then click Open With. In the Open With dialog box, click to select Resource Editor as the program with which to open the .rc file, and then click Open.
      2. If you are using Visual C++ 4.x or 5.0, add the line #include "windows.h" to the top of the file Include\prnsetup.dlg. Save and close this file. Reopen it as "Resources" by using the Open As box in the File Open dialog box.
    3. In the Resource Editor window of the Commdlg.rc file, click the PRINTDLGORD dialog resource, click Edit menu, and then click Copy. The PRINTDLGORD ID is 1538. This PRINTDLGORD symbol is defined in the header file Dlgs.h.
    4. In the Resource Editor window of your project .rc file, right-click Dialog and then click Paste. Now, save and close the project .rc file in the Resource Editor window, and then reopen the default Resource View window of your project. If you are using Visual C++ 6.0, there are no context menus for copying resources. On the File menu, click Open to open the Commdlg.rc file, and then open the Resource View menu for your project. Select the 1538 resource ID in Commdlg.rc, hold down the CTRL key, and then drag the dialog box resource from the Commdlg.rc window to your Resource View window.
    1. If you are using Visual C++ .NET, follow these steps:
      1. Click Add Class on the Project menu.
      2. The Add Class menu is context-sensitive. The Add Class menu does not appear when the Resource View is active. Switch to Solution Explorer before you click Add Class on the Project menu.
      3. Expand the MFC category, click the MFC class template, and then click Open.
      4. Use the MFC Class Wizard to add a Visual C++ class for this dialog box template, such as CMyPrintDialog. Derive this new class from CPrintDialog.
    2. If you are using Visual C++ 6.0, click Add Class on the Project menu.

      The Add Class menu is context-sensitive. The Add Class menu does not appear when the Resource View is active. Switch to ClassView before you click Add Class on the Project menu. Type the name for the class, set the Base Class as CPrintDialog, and then click OK.
  2. Change all references from CDialog to CPrintDialog in both the header and implementation file of the newly-created class. Note that this step is not necessary if you have derived your class directly from CPrintDialog.
  3. Because the constructor of CPrintDialog differs from CDialog, modify the constructor of CMyPrintDialog by using the following code: NOTE: This step is not necessary if you have derived your class directly from CPrintDialog.
          // Header file of CMyPrintDialog.
          class CMyPrintDialog : public CPrintDialog
          {
          // Construction.
          public:
    
          // The arguments to the following constructor closely match
          // CPrintDialog. Note the difference in the second argument.
          CMyPrintDialog(BOOL bPrintSetupOnly,
                         // TRUE for Print Setup, FALSE for Print Dialog.
                           DWORD dwFlags = PD_ALLPAGES | PD_USEDEVMODECOPIES |
                                        PD_HIDEPRINTTOFILE,
                         // Combination of flags. Refer to the Windows SDK
                         // documentation for PRINTDLG structure for a
                         // description of all the flags that can be used.
                            CWnd* pParentWnd = NULL);
    
          // Rest of the class declaration.
          ...
    
          DECLARE_MESSAGE_MAP()
          };
    
          // Implementation file of CMyPrintDialog.
          CMyPrintDialog::CMyPrintDialog(BOOL bPrintSetupOnly,
            DWORD dwFlags /* = PD_ALLPAGES | PD_USEDEVMODECOPIES |
                          PD_HIDEPRINTTOFILE */,
            CWnd* pParentWnd /* = NULL */)
            : CPrintDialog(bPrintSetupOnly, dwFlags, pParentWnd)
          {
            //{{AFX_DATA_INIT(CMyPrintDialog)
            // NOTE: the ClassWizard will add member initialization here.
            //}}AFX_DATA_INIT
          }
     
    					
  4. Make any desired changes to the copied dialog box template and associated source code. NOTE: Do not delete any of the controls that are present in the original dialog box template. Deleting the controls causes problems in the DoDataExchange function of CPrintDialog. Instead, either disable the unwanted controls or hide them in an overridden OnInitDialog member function of your CPrintDialog-derived class.
  5. Modify the CView-derived class (for example, CMyView) to use the customized print dialog box by using the following code:
          //  Implementation file of the view (for example, in myview.cpp).
          ...
          #include "MyPrintDialog.h"  // Include the CMyPrintDialog header file.
          ...
    
          // Override OnPreparePrinting of the CView-derived class as below:
          BOOL CMyView::OnPreparePrinting(CPrintInfo* pInfo)
          {
             // Delete the CPrintDialog object created in the CPrintInfo
             // constructor, and substitute with customized print dialog.
             delete pInfo->m_pPD;
    
             // Construct and substitute with customized print dialog.
             pInfo->m_pPD = new CMyPrintDialog(FALSE);
    
             // Set the page range.
             pInfo->m_pPD->m_pd.nMinPage = 1;      // one based page numbers.
             pInfo->m_pPD->m_pd.nMaxPage = 0xffff; // how many pages is unknown.
    
             // Change the PRINTDLG struct so that the custom print dialog box will
             // be used.
             pInfo->m_pPD->m_pd.hInstance = AfxGetInstanceHandle();
             pInfo->m_pPD->m_pd.lpPrintTemplateName =
                                                  MAKEINTRESOURCE(1538);
    
             // Set the Flags of the PRINTDLG structure as shown, else the
             // changes will have no effect.
             pInfo->m_pPD->m_pd.Flags |= PD_ENABLEPRINTTEMPLATE;
    
             // For details about these flags, refer to the SDK documentation
             // on the PRINTDLG structure.
    
             return DoPreparePrinting(pInfo);
          }

REFERENCES

For more information, see the PRINTDLG structure in the Windows SDK documentation and the "OnPreparePrinting" and "CPrintDialog" topics in the Foundation Class documentation.

Modification Type:MajorLast Reviewed:9/1/2005
Keywords:kbcode kbFAQ kbhowto kbprint kbProgramming KB132909 kbAudDeveloper