Introduction
Thanks to Chris Maunder for introducing the printing method in Printing without the Document/View framework. Many people have been asking for print preview without the doc/view framework as well. However, so far, no body has proposed a way to do that yet. I have studied on how Microsoft implements print preview in doc/view framework using the class CPreviewView
, which is derived from CSrollView
. The print preview is called from CView::OnFilePrintPreview( )
and CView::DoPrintPreview
. They are undocumented, the source code can be found from VC\mfc\src\Viewprev.cpp. The implementation relies on CView
and CFrameWnd
framework extensively.
In order to use the default print preview functionality in non doc/view framework, like dialog based applications, I played a trick by creating temporary CFrameWnd
and CView
objects and then calling the default OnFilePrintPreview( )
from the temporary view class. I borrowed Chris Maunder's CGridCtrl
control from MFC Grid control (derived from CWnd) article as an example to illustrate the implementation. I included another demo project, which was modified from Iuri Apollonio's article: Generic printing class (and how to print a list control content). The code was developed under VC5 environment and has been tested in Windows 98 and NT platforms without any problem. :)
How to use
Add a function PrintPreview( )
in your control class (CGridCtrl
in this case) or the class where you put your OnBeginPrinting
, OnPrint
, etc. Also, add two member variables to the class and initialize them:
private:
CSingleDocTemplate* m_pTemplate;
public:
BOOL m_bPrintPreview;
void PrintPreview();
#include "ViewPrintPreview.h"
CGridCtrl::CGridCtrl(...)
{
m_pTemplate=NULL;
m_bPrintPreview=FALSE;
}
void CGridCtrl::PrintPreview()
{
if (m_bPrintPreview)
{
AfxGetApp()->m_pMainWnd->SetFocus();
return;
}
CFrameWnd* pOldFrame=(CFrameWnd*)AfxGetThread()->m_pMainWnd;
pOldFrame->ShowWindow(SW_HIDE);
if (!m_pTemplate)
{
m_pTemplate = new CSingleDocTemplate(
IDR_MENU,
NULL,
RUNTIME_CLASS(CFrameWnd),
RUNTIME_CLASS(CViewPrintPreview));
AfxGetApp()->AddDocTemplate(m_pTemplate);
}
CFrameWnd * pFrameWnd = m_pTemplate->CreateNewFrame( NULL, NULL );
m_bPrintPreview=TRUE;
m_pTemplate->InitialUpdateFrame( pFrameWnd, NULL, FALSE);
CViewPrintPreview* pView=(CViewPrintPreview*)pFrameWnd->GetActiveView();
pView->m_pCtrl=this;
pView->m_pOldFrame=pOldFrame;
AfxGetApp()->m_pMainWnd=pFrameWnd;
pFrameWnd->SetWindowText(_T("Koay Kah Hoe Print Preview"));
pFrameWnd->ShowWindow(SW_SHOWMAXIMIZED);
pView->OnFilePrintPreview();
}
A CSingleDocTemplate
object is used to create a frame and a view window. The view class CViewPrintPreview
has no chance to show itself, as it is suppressed by preview view immediately. The m_pMainWnd
pointer is changed to the new CFrameWnd
so that the preview class can use it as parent frame. (Has any one done this before?) The original m_pMainWnd
is saved in the view class, when preview is ended, it will restore the pointer back to the original value. Then, OnFilePrintPreview
is called.
Add the view class CViewPrintPreview
and its header file to the project. The main job of the view class is to pass the printing functions OnBeginPrinting
, OnPrint
and OnEndPrinting
to the control class or wherever you place them. This must be done through the view class as the print preview calls these functions from view class. When the preview window is closed, the program must restore the m_pMainWnd
pointer back to its original value. Then, the frame and view window must be destroyed. This is done in the function OnEndPrintPreview
.
Now, you can view the preview window already. However, this is not the end of the story yet, the toolbar buttons cannot update themselves as normally they should be. After some investigation, I found that the window normally calls the message WM_IDLEUPDATECMDUI
to update the toolbar's state from the OnIdle
function in SDI or MDI application. For dialog based applications, the OnIdle
function is not called. Thus, the message WM_IDLEUPDATECMDUI
is not sent to update toolbar. We have to send the message ourselves. I overrode the virtual function ContinueModal( )
in the main dialog class, to do the job. You must include the header file AfxPriv.h in order to recognize the WM_IDLEUPDATECMDUI
message.
BOOL CGridCtrlDemoDlg::ContinueModal()
{
if (m_Grid.m_bPrintPreview)
AfxGetApp()->m_pMainWnd->SendMessageToDescendants(WM_IDLEUPDATECMDUI,
(WPARAM)TRUE, 0, TRUE, TRUE);
return CDialog::ContinueModal();
}
Special precaution has to be taken when you use the print preview. As the m_pMainWnd
pointer is pointed to the new frame window when doing preview, elsewhere in your application that uses this pointer could cause your program to crash. You can use the m_bPrintPreview
as an indicator to determine which window the m_pMainWnd
is pointed to.
FAQ on compilation using static library
In dialog based applications, the resource file for the print preview is not included. This problem can be solved by simply including the resource file afxprint.rc to your project file. Open your resource file as text explicitly, add the following line in TEXTINCLUDE 3
section below #include afxres.rc
:
"#include ""afxprint.rc"" // printing/print preview resources\r\n"
At the end of the resource file, do the actual including again as follows, below #include afxres.rc
:
#include afxprint.rc // printing/print preview resources
FAQ on print button in preview toolbar
When pressing the print button in preview toolbar, the ID_FILE_PRINT
command message is sent. Thus, you must create a message handler to the ID_FILE_PRINT
message at your main dialog class, or wherever the message could be caught.
Add a line like below in the message map of your dialog class
BEGIN_MESSAGE_MAP(CMyDlg, CDialog)
...
ON_COMMAND(ID_FILE_PRINT, OnFilePrint)
END_MESSAGE_MAP()
Then of course, you must have the handler function OnFilePrint()
. See the given example on how it is done.
In cases where there are more than one printing function in the application, but there is only one general ID_FILE_PRINT
message, special processing is needed to ensure that the correct printing function is called. There are two possible solutions. The first way is to use a variable identifying which printing should take place. The message handler should refer to this variable to call the correct printing function. The other way is to reroute the message mapping by overriding OnCmdMsg()
. Each printable dialog or control has its own message handler for ID_FILE_PRINT
. When a particular dialog or control is active, the message should be routed to the active dialog or control first.
Finally, I wish you happy previewing! :)
ViewPrintPreview header file
#if !defined(AFX_VIEWPRINTPREVIEW_H__137FC880_1607_11D3_9317_8F51A5F9742F__INCLUDED_)
#define AFX_VIEWPRINTPREVIEW_H__137FC880_1607_11D3_9317_8F51A5F9742F__INCLUDED_
#if _MSC_VER >= 1000
#pragma once
#endif
#include "GridCtrl.h"
class CViewPrintPreview : public CView
{
protected:
CViewPrintPreview();
DECLARE_DYNCREATE(CViewPrintPreview)
public:
CGridCtrl *m_pCtrl;
CFrameWnd *m_pOldFrame;
public:
virtual void OnFilePrintPreview();
protected:
virtual void OnDraw(CDC* pDC);
virtual void OnBeginPrinting(CDC* pDC, CPrintInfo* pInfo);
virtual void OnPrint(CDC* pDC, CPrintInfo* pInfo);
virtual void OnEndPrinting(CDC* pDC, CPrintInfo* pInfo);
virtual BOOL OnPreparePrinting(CPrintInfo* pInfo);
virtual void OnEndPrintPreview(CDC* pDC,
CPrintInfo* pInfo, POINT point, CPreviewView* pView);
protected:
virtual ~CViewPrintPreview();
#ifdef _DEBUG
virtual void AssertValid() const;
virtual void Dump(CDumpContext& dc) const;
#endif
protected:
DECLARE_MESSAGE_MAP()
};
#endif
ViewPrintPreview implementation file
#include "stdafx.h"
#include "ViewPrintPreview.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
IMPLEMENT_DYNCREATE(CViewPrintPreview, CView)
CViewPrintPreview::CViewPrintPreview()
{
m_pOldFrame=NULL;
}
CViewPrintPreview::~CViewPrintPreview()
{
}
BEGIN_MESSAGE_MAP(CViewPrintPreview, CView)
END_MESSAGE_MAP()
void CViewPrintPreview::OnDraw(CDC* pDC)
{
CDocument* pDoc = GetDocument();
}
#ifdef _DEBUG
void CViewPrintPreview::AssertValid() const
{
CView::AssertValid();
}
void CViewPrintPreview::Dump(CDumpContext& dc) const
{
CView::Dump(dc);
}
#endif
void CViewPrintPreview::OnFilePrintPreview()
{
CView::OnFilePrintPreview();
}
BOOL CViewPrintPreview::OnPreparePrinting(CPrintInfo* pInfo)
{
return DoPreparePrinting(pInfo);
}
void CViewPrintPreview::OnBeginPrinting(CDC* pDC, CPrintInfo* pInfo)
{
m_pCtrl->OnBeginPrinting(pDC, pInfo);
}
void CViewPrintPreview::OnPrint(CDC* pDC, CPrintInfo* pInfo)
{
m_pCtrl->OnPrint(pDC, pInfo);
}
void CViewPrintPreview::OnEndPrinting(CDC* pDC, CPrintInfo* pInfo)
{
m_pCtrl->OnEndPrinting(pDC, pInfo);
}
void CViewPrintPreview::OnEndPrintPreview(CDC* pDC,
CPrintInfo* pInfo, POINT point, CPreviewView* pView)
{
CView::OnEndPrintPreview(pDC, pInfo, point, pView);
m_pOldFrame->ShowWindow(SW_SHOW);
AfxGetApp()->m_pMainWnd=m_pOldFrame;
m_pCtrl->m_bPrintPreview=FALSE;
GetParentFrame()->DestroyWindow();
}