mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1251 lines
35 KiB
1251 lines
35 KiB
/*
|
|
* npprint.c -- Code for printing from sxspad.
|
|
* Copyright (C) 1984-2000 Microsoft Inc.
|
|
*/
|
|
|
|
#define NOMINMAX
|
|
#include "precomp.h"
|
|
|
|
//#define DBGPRINT
|
|
|
|
/* indices into chBuff */
|
|
#define LEFT 0
|
|
#define CENTER 1
|
|
#define RIGHT 2
|
|
|
|
INT tabSize; /* Size of a tab for print device in device units*/
|
|
HWND hAbortDlgWnd;
|
|
INT fAbort; /* true if abort in progress */
|
|
INT yPrintChar; /* height of a character */
|
|
|
|
|
|
RECT rtMargin;
|
|
|
|
/* left,center and right string for header or trailer */
|
|
#define MAXTITLE MAX_PATH
|
|
TCHAR chBuff[RIGHT+1][MAXTITLE];
|
|
|
|
/* date and time stuff for headers */
|
|
#define MAXDATE MAX_PATH
|
|
#define MAXTIME MAX_PATH
|
|
TCHAR szFormattedDate[MAXDATE]=TEXT("Y"); // formatted date (may be internationalized)
|
|
TCHAR szFormattedTime[MAXTIME]=TEXT("Y"); // formatted time (may be internaltionalized)
|
|
SYSTEMTIME PrintTime; // time we started printing
|
|
|
|
|
|
INT xPrintRes; // printer resolution in x direction
|
|
INT yPrintRes; // printer resolution in y direction
|
|
INT yPixInch; // pixels/inch
|
|
INT xPhysRes; // physical resolution x of paper
|
|
INT yPhysRes; // physical resolution y of paper
|
|
|
|
INT xPhysOff; // physical offset x
|
|
INT yPhysOff; // physical offset y
|
|
|
|
INT dyTop; // width of top border (pixels)
|
|
INT dyBottom; // width of bottom border
|
|
INT dxLeft; // width of left border
|
|
INT dxRight; // width of right border
|
|
|
|
INT iPageNum; // global page number currently being printed
|
|
|
|
/* define a type for NUM and the base */
|
|
typedef long NUM;
|
|
#define BASE 100L
|
|
|
|
/* converting in/out of fixed point */
|
|
#define NumToShort(x,s) (LOWORD(((x) + (s)) / BASE))
|
|
#define NumRemToShort(x) (LOWORD((x) % BASE))
|
|
|
|
/* rounding options for NumToShort */
|
|
#define NUMFLOOR 0
|
|
#define NUMROUND (BASE/2)
|
|
#define NUMCEILING (BASE-1)
|
|
|
|
#define ROUND(x) NumToShort(x,NUMROUND)
|
|
#define FLOOR(x) NumToShort(x,NUMFLOOR)
|
|
|
|
/* Unit conversion */
|
|
#define InchesToCM(x) (((x) * 254L + 50) / 100)
|
|
#define CMToInches(x) (((x) * 100L + 127) / 254)
|
|
|
|
void DestroyAbortWnd(void) ;
|
|
VOID TranslateString(TCHAR *);
|
|
|
|
BOOL CALLBACK AbortProc(HDC hPrintDC, INT reserved)
|
|
{
|
|
MSG msg;
|
|
|
|
while( !fAbort && PeekMessage((LPMSG)&msg, NULL, 0, 0, TRUE) )
|
|
{
|
|
if( !hAbortDlgWnd || !IsDialogMessage( hAbortDlgWnd, (LPMSG)&msg ) )
|
|
{
|
|
TranslateMessage( (LPMSG)&msg );
|
|
DispatchMessage( (LPMSG)&msg );
|
|
}
|
|
}
|
|
return( !fAbort );
|
|
|
|
UNREFERENCED_PARAMETER(hPrintDC);
|
|
UNREFERENCED_PARAMETER(reserved);
|
|
}
|
|
|
|
|
|
INT_PTR CALLBACK AbortDlgProc(
|
|
HWND hwnd,
|
|
UINT msg,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
static HMENU hSysMenu;
|
|
|
|
switch( msg )
|
|
{
|
|
case WM_COMMAND:
|
|
fAbort= TRUE;
|
|
DestroyAbortWnd();
|
|
return( TRUE );
|
|
|
|
case WM_INITDIALOG:
|
|
hSysMenu= GetSystemMenu( hwnd, FALSE );
|
|
SetDlgItemText( hwnd, ID_FILENAME,
|
|
fUntitled ? szUntitled : PFileInPath(szFileName) );
|
|
SetFocus( hwnd );
|
|
return( TRUE );
|
|
|
|
case WM_INITMENU:
|
|
EnableMenuItem( hSysMenu, (WORD)SC_CLOSE, (DWORD)MF_GRAYED );
|
|
return( TRUE );
|
|
}
|
|
return( FALSE );
|
|
|
|
UNREFERENCED_PARAMETER(wParam);
|
|
UNREFERENCED_PARAMETER(lParam);
|
|
}
|
|
|
|
|
|
/*
|
|
* print out the translated header/footer string in proper position.
|
|
* uses globals xPrintWidth, ...
|
|
*
|
|
* returns 1 if line was printed, otherwise 0.
|
|
*/
|
|
|
|
INT PrintHeaderFooter (HDC hDC, INT nHF)
|
|
{
|
|
SIZE Size; // to compute the width of each string
|
|
INT yPos; // y position to print
|
|
INT xPos; // x position to print
|
|
|
|
if( *chPageText[nHF] == 0 ) // see if anything to do
|
|
return 0; // we didn't print
|
|
|
|
TranslateString( chPageText[nHF] );
|
|
|
|
// figure out the y position we are printing
|
|
|
|
if( nHF == HEADER )
|
|
yPos= dyTop;
|
|
else
|
|
yPos= yPrintRes - dyBottom - yPrintChar;
|
|
|
|
// print out the various strings
|
|
// N.B. could overprint which seems ok for now
|
|
|
|
if( *chBuff[LEFT] ) // left string
|
|
{
|
|
TextOut( hDC, dxLeft, yPos, chBuff[LEFT], lstrlen(chBuff[LEFT]) );
|
|
}
|
|
|
|
if( *chBuff[CENTER] ) // center string
|
|
{
|
|
GetTextExtentPoint32( hDC, chBuff[CENTER], lstrlen(chBuff[CENTER]), &Size );
|
|
xPos= (xPrintRes-dxRight+dxLeft)/2 - Size.cx/2;
|
|
TextOut( hDC, xPos, yPos, chBuff[CENTER], lstrlen(chBuff[CENTER]) );
|
|
}
|
|
|
|
if( *chBuff[RIGHT] ) // right string
|
|
{
|
|
GetTextExtentPoint32( hDC, chBuff[RIGHT], lstrlen(chBuff[RIGHT]), &Size );
|
|
xPos= xPrintRes - dxRight - Size.cx;
|
|
TextOut( hDC, xPos, yPos, chBuff[RIGHT], lstrlen(chBuff[RIGHT]) );
|
|
}
|
|
return 1; // we did print something
|
|
}
|
|
/*
|
|
* GetResolutions
|
|
*
|
|
* Gets printer resolutions.
|
|
* sets globals: xPrintRes, yPrintRes, yPixInch
|
|
*
|
|
*/
|
|
|
|
VOID GetResolutions(HDC hPrintDC)
|
|
{
|
|
xPrintRes = GetDeviceCaps( hPrintDC, HORZRES );
|
|
yPrintRes = GetDeviceCaps( hPrintDC, VERTRES );
|
|
yPixInch = GetDeviceCaps( hPrintDC, LOGPIXELSY );
|
|
|
|
xPhysRes = GetDeviceCaps( hPrintDC, PHYSICALWIDTH );
|
|
yPhysRes = GetDeviceCaps( hPrintDC, PHYSICALHEIGHT );
|
|
|
|
xPhysOff = GetDeviceCaps( hPrintDC, PHYSICALOFFSETX );
|
|
yPhysOff = GetDeviceCaps( hPrintDC, PHYSICALOFFSETY );
|
|
}
|
|
|
|
/* GetMoreText
|
|
*
|
|
* Gets the next line of text from the MLE, returning a pointer
|
|
* to the beginning and just past the end.
|
|
*
|
|
* linenum - index into MLE (IN)
|
|
* pStartText - start of MLE (IN)
|
|
* ppsStr - pointer to where to put pointer to start of text (OUT)
|
|
* ppEOL - pointer to where to put pointer to just past EOL (OUT)
|
|
*
|
|
*/
|
|
|
|
VOID GetMoreText( INT linenum, PTCHAR pStartText, PTCHAR* ppsStr, PTCHAR* ppEOL )
|
|
{
|
|
INT Offset; // offset in 'chars' into edit buffer
|
|
INT nChars; // number of chars in line
|
|
|
|
Offset= (INT)SendMessage( hwndEdit, EM_LINEINDEX, linenum, 0 );
|
|
|
|
nChars= (INT)SendMessage( hwndEdit, EM_LINELENGTH, Offset, 0 );
|
|
|
|
*ppsStr= pStartText + Offset;
|
|
|
|
*ppEOL= (pStartText+Offset) + nChars;
|
|
}
|
|
|
|
#ifdef DBGPRINT
|
|
TCHAR dbuf[100];
|
|
VOID ShowMargins( HDC hPrintDC )
|
|
{
|
|
INT xPrintRes, yPrintRes;
|
|
RECT rct;
|
|
HBRUSH hBrush;
|
|
|
|
xPrintRes= GetDeviceCaps( hPrintDC, HORZRES );
|
|
yPrintRes= GetDeviceCaps( hPrintDC, VERTRES );
|
|
hBrush= GetStockObject( BLACK_BRUSH );
|
|
SetRect( &rct, 0,0,xPrintRes-1, yPrintRes-1 );
|
|
FrameRect( hPrintDC, &rct, hBrush );
|
|
SetRect( &rct, dxLeft, dyTop, xPrintRes-dxRight, yPrintRes-dyBottom );
|
|
FrameRect( hPrintDC, &rct, hBrush );
|
|
}
|
|
|
|
VOID PrintLogFont( LOGFONT lf )
|
|
{
|
|
wsprintf(dbuf,TEXT("lfHeight %d\n"), lf.lfHeight ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfWidth %d\n"), lf.lfWidth ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfEscapement %d\n"), lf. lfEscapement ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfOrientation %d\n"), lf.lfOrientation ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfWeight %d\n"), lf.lfWeight ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfItalic %d\n"), lf.lfItalic ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfUnderline %d\n"), lf.lfUnderline ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfStrikeOut %d\n"), lf.lfStrikeOut ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfCharSet %d\n"), lf.lfCharSet ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfOutPrecision %d\n"), lf.lfOutPrecision ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfClipPrecison %d\n"), lf.lfClipPrecision ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfQuality %d\n"), lf.lfQuality ); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfPitchAndFamily %d\n"), lf.lfPitchAndFamily); ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("lfFaceName %s\n"), lf.lfFaceName ); ODS(dbuf);
|
|
}
|
|
#endif
|
|
|
|
// GetPrinterDCviaDialog
|
|
//
|
|
// Use the common dialog PrintDlgEx() function to get a printer DC to print to.
|
|
//
|
|
// Returns: valid HDC or INVALID_HANDLE_VALUE if error.
|
|
//
|
|
|
|
HDC GetPrinterDCviaDialog( VOID )
|
|
{
|
|
PRINTDLGEX pdTemp;
|
|
HDC hDC;
|
|
HRESULT hRes;
|
|
|
|
//
|
|
// Get the page setup information
|
|
//
|
|
|
|
if( !g_PageSetupDlg.hDevNames ) /* Retrieve default printer if none selected. */
|
|
{
|
|
g_PageSetupDlg.Flags |= (PSD_RETURNDEFAULT|PSD_NOWARNING );
|
|
PageSetupDlg(&g_PageSetupDlg);
|
|
g_PageSetupDlg.Flags &= ~(PSD_RETURNDEFAULT|PSD_NOWARNING);
|
|
}
|
|
|
|
//
|
|
// Initialize the dialog structure
|
|
//
|
|
|
|
ZeroMemory( &pdTemp, sizeof(pdTemp) );
|
|
|
|
pdTemp.lStructSize= sizeof(pdTemp);
|
|
|
|
pdTemp.hwndOwner= hwndSP;
|
|
pdTemp.nStartPage= START_PAGE_GENERAL;
|
|
pdTemp.Flags= PD_NOPAGENUMS | PD_RETURNDC | PD_NOCURRENTPAGE |
|
|
PD_NOSELECTION | 0;
|
|
|
|
// if use set printer in PageSetup, use it here too.
|
|
|
|
if( g_PageSetupDlg.hDevMode )
|
|
{
|
|
pdTemp.hDevMode= g_PageSetupDlg.hDevMode;
|
|
}
|
|
|
|
if( g_PageSetupDlg.hDevNames )
|
|
{
|
|
pdTemp.hDevNames= g_PageSetupDlg.hDevNames;
|
|
}
|
|
|
|
|
|
//
|
|
// let user select printer
|
|
//
|
|
|
|
hRes= PrintDlgEx( &pdTemp );
|
|
|
|
//
|
|
// get DC if valid return
|
|
//
|
|
|
|
hDC= INVALID_HANDLE_VALUE;
|
|
|
|
if( hRes == S_OK )
|
|
{
|
|
if( (pdTemp.dwResultAction == PD_RESULT_PRINT) || (pdTemp.dwResultAction == PD_RESULT_APPLY) )
|
|
{
|
|
if( pdTemp.dwResultAction == PD_RESULT_PRINT )
|
|
{
|
|
hDC= pdTemp.hDC;
|
|
}
|
|
|
|
//
|
|
// Get the page setup information for the printer selected in case it was
|
|
// the first printer added by the user through sxspad.
|
|
//
|
|
if( !g_PageSetupDlg.hDevMode )
|
|
{
|
|
g_PageSetupDlg.Flags |= (PSD_RETURNDEFAULT|PSD_NOWARNING );
|
|
PageSetupDlg(&g_PageSetupDlg);
|
|
g_PageSetupDlg.Flags &= ~(PSD_RETURNDEFAULT|PSD_NOWARNING);
|
|
}
|
|
|
|
// change devmode if user pressed print or apply
|
|
g_PageSetupDlg.hDevMode= pdTemp.hDevMode;
|
|
g_PageSetupDlg.hDevNames= pdTemp.hDevNames;
|
|
}
|
|
}
|
|
|
|
// FEATURE: free hDevNames
|
|
|
|
return( hDC );
|
|
}
|
|
|
|
INT NpPrint( PRINT_DIALOG_TYPE type)
|
|
{
|
|
HDC hPrintDC;
|
|
|
|
SetCursor( hWaitCursor );
|
|
|
|
switch( type )
|
|
{
|
|
case UseDialog:
|
|
hPrintDC= GetPrinterDCviaDialog();
|
|
break;
|
|
case NoDialogNonDefault:
|
|
hPrintDC= GetNonDefPrinterDC();
|
|
break;
|
|
case DoNotUseDialog:
|
|
default:
|
|
hPrintDC= GetPrinterDC();
|
|
break;
|
|
}
|
|
|
|
if( hPrintDC == INVALID_HANDLE_VALUE )
|
|
{
|
|
SetCursor( hStdCursor );
|
|
return( 0 ); // message already given
|
|
}
|
|
|
|
return( NpPrintGivenDC( hPrintDC ) );
|
|
|
|
}
|
|
|
|
INT NpPrintGivenDC( HDC hPrintDC )
|
|
{
|
|
HANDLE hText= NULL; // handle to MLE text
|
|
HFONT hPrintFont= NULL; // font to print with
|
|
HANDLE hPrevFont= NULL; // previous font in hPrintDC
|
|
|
|
BOOL fPageStarted= FALSE; // true if StartPage called for this page
|
|
BOOL fDocStarted= FALSE; // true if StartDoc called
|
|
PTCHAR pStartText= NULL; // start of edit text (locked hText)
|
|
TEXTMETRIC Metrics;
|
|
TCHAR msgbuf[MAX_PATH]; // Document name for tracking print job
|
|
INT nLinesPerPage; // not inc. header and footer
|
|
// iErr will contain the first error discovered ie it is sticky
|
|
// This will be the value returned by this function.
|
|
// It does not need to translate SP_* errors except for SP_ERROR which should be
|
|
// GetLastError() right after it is first detected.
|
|
INT iErr=0; // error return
|
|
DOCINFO DocInfo;
|
|
LOGFONT lfPrintFont; // local version of FontStruct
|
|
LCID lcid; // locale id
|
|
|
|
fAbort = FALSE;
|
|
hAbortDlgWnd= NULL;
|
|
|
|
SetCursor( hWaitCursor );
|
|
|
|
GetResolutions( hPrintDC );
|
|
|
|
// Get the time and date for use in the header or trailer.
|
|
// We use the GetDateFormat and GetTimeFormat to get the
|
|
// internationalized versions.
|
|
|
|
GetLocalTime( &PrintTime ); // use local, not gmt
|
|
|
|
lcid= GetUserDefaultLCID();
|
|
|
|
GetDateFormat( lcid, DATE_LONGDATE, &PrintTime, NULL, szFormattedDate, MAXDATE );
|
|
|
|
GetTimeFormat( lcid, 0, &PrintTime, NULL, szFormattedTime, MAXTIME );
|
|
|
|
|
|
/*
|
|
* This part is to select the current font to the printer device.
|
|
* We have to change the height because FontStruct was created
|
|
* assuming the display. Using the remembered pointsize, calculate
|
|
* the new height.
|
|
*/
|
|
|
|
lfPrintFont= FontStruct; // make local copy
|
|
lfPrintFont.lfHeight= -(iPointSize*yPixInch)/(72*10);
|
|
lfPrintFont.lfWidth= 0;
|
|
|
|
//
|
|
// convert margins to pixels
|
|
// ptPaperSize is the physical paper size, not the printable area.
|
|
// do the mapping in physical units
|
|
//
|
|
|
|
SetMapMode( hPrintDC, MM_ANISOTROPIC );
|
|
|
|
SetViewportExtEx( hPrintDC,
|
|
xPhysRes,
|
|
yPhysRes,
|
|
NULL );
|
|
|
|
SetWindowExtEx( hPrintDC,
|
|
g_PageSetupDlg.ptPaperSize.x,
|
|
g_PageSetupDlg.ptPaperSize.y,
|
|
NULL );
|
|
|
|
rtMargin = g_PageSetupDlg.rtMargin;
|
|
|
|
LPtoDP( hPrintDC, (LPPOINT) &rtMargin, 2 );
|
|
|
|
SetMapMode( hPrintDC,MM_TEXT ); // restore to mm_text mode
|
|
|
|
hPrintFont= CreateFontIndirect(&lfPrintFont);
|
|
|
|
if( !hPrintFont )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
hPrevFont= SelectObject( hPrintDC, hPrintFont );
|
|
if( !hPrevFont )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
SetBkMode( hPrintDC, TRANSPARENT );
|
|
if( !GetTextMetrics( hPrintDC, (LPTEXTMETRIC) &Metrics ) )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
// The font may not a scalable (say on a bubblejet printer)
|
|
// In this case, just pick some font
|
|
// For example, FixedSys 9 pt would be non-scalable
|
|
|
|
if( !(Metrics.tmPitchAndFamily & (TMPF_VECTOR | TMPF_TRUETYPE )) )
|
|
{
|
|
// remove just created font
|
|
|
|
hPrintFont= SelectObject( hPrintDC, hPrevFont ); // get old font
|
|
DeleteObject( hPrintFont );
|
|
|
|
memset( lfPrintFont.lfFaceName, 0, LF_FACESIZE*sizeof(TCHAR) );
|
|
|
|
hPrintFont= CreateFontIndirect( &lfPrintFont );
|
|
if( !hPrintFont )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
hPrevFont= SelectObject( hPrintDC, hPrintFont );
|
|
if( !hPrevFont )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
if( !GetTextMetrics( hPrintDC, (LPTEXTMETRIC) &Metrics ) )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
yPrintChar= Metrics.tmHeight+Metrics.tmExternalLeading; /* the height */
|
|
|
|
tabSize = Metrics.tmAveCharWidth * 8; /* 8 ave char width pixels for tabs */
|
|
|
|
// compute margins in pixels
|
|
|
|
dxLeft= max(rtMargin.left - xPhysOff,0);
|
|
dxRight= max(rtMargin.right - (xPhysRes - xPrintRes - xPhysOff), 0 );
|
|
dyTop= max(rtMargin.top - yPhysOff,0);
|
|
dyBottom= max(rtMargin.bottom - (yPhysRes - yPrintRes - yPhysOff), 0 );
|
|
|
|
#ifdef DBGPRINT
|
|
{
|
|
TCHAR dbuf[100];
|
|
RECT rt= g_PageSetupDlg.rtMargin;
|
|
POINT pt;
|
|
|
|
wsprintf(dbuf,TEXT("Print pOffx %d pOffy %d\n"),
|
|
GetDeviceCaps(hPrintDC, PHYSICALOFFSETX),
|
|
GetDeviceCaps(hPrintDC, PHYSICALOFFSETY));
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("PHYSICALWIDTH: %d\n"), xPhysRes);
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("HORZRES: %d\n"),xPrintRes);
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("PHYSICALOFFSETX: %d\n"),xPhysOff);
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("LOGPIXELSX: %d\n"),
|
|
GetDeviceCaps(hPrintDC,LOGPIXELSX));
|
|
ODS(dbuf);
|
|
|
|
GetViewportOrgEx( hPrintDC, (LPPOINT) &pt );
|
|
wsprintf(dbuf,TEXT("Viewport org: %d %d\n"), pt.x, pt.y );
|
|
ODS(dbuf);
|
|
GetWindowOrgEx( hPrintDC, (LPPOINT) &pt );
|
|
wsprintf(dbuf,TEXT("Window org: %d %d\n"), pt.x, pt.y );
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("PrintRes x: %d y: %d\n"),xPrintRes, yPrintRes);
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("PaperSize x: %d y: %d\n"),
|
|
g_PageSetupDlg.ptPaperSize.x,
|
|
g_PageSetupDlg.ptPaperSize.y );
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("unit margins: l: %d r: %d t: %d b: %d\n"),
|
|
rt.left, rt.right, rt.top, rt.bottom);
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("pixel margins: l: %d r: %d t: %d b: %d\n"),
|
|
rtMargin.left, rtMargin.right, rtMargin.top, rtMargin.bottom);
|
|
ODS(dbuf);
|
|
|
|
wsprintf(dbuf,TEXT("dxLeft %d dxRight %d\n"),dxLeft,dxRight);
|
|
ODS(dbuf);
|
|
wsprintf(dbuf,TEXT("dyTop %d dyBot %d\n"),dyTop,dyBottom);
|
|
ODS(dbuf);
|
|
}
|
|
#endif
|
|
|
|
|
|
/* Number of lines on a page with margins */
|
|
/* two lines are used by header and footer */
|
|
nLinesPerPage = ((yPrintRes - dyTop - dyBottom) / yPrintChar);
|
|
|
|
if( *chPageText[HEADER] )
|
|
nLinesPerPage--;
|
|
if( *chPageText[FOOTER] )
|
|
nLinesPerPage--;
|
|
|
|
|
|
/*
|
|
** There was a bug in NT once where a printer driver would
|
|
** return a font that was larger than the page size which
|
|
** would then cause sxspad to constantly print blank pages
|
|
** To keep from doing this we check to see if we can fit ANYTHING
|
|
** on a page, if not then there is a problem so quit. MarkRi 8/92
|
|
*/
|
|
if( nLinesPerPage <= 0 )
|
|
{
|
|
FontTooBig:
|
|
MessageBox( hwndSP, szFontTooBig, szNN, MB_APPLMODAL | MB_OK | MB_ICONEXCLAMATION );
|
|
|
|
SetLastError(0); // no error
|
|
|
|
ErrorExit:
|
|
iErr= GetLastError(); // remember the first error
|
|
|
|
ExitWithThisError: // preserve iErr (return SP_* errors)
|
|
|
|
if( hPrevFont )
|
|
{
|
|
SelectObject( hPrintDC, hPrevFont );
|
|
DeleteObject( hPrintFont );
|
|
}
|
|
|
|
if( pStartText ) // were able to lock hText
|
|
LocalUnlock( hText );
|
|
|
|
if( fPageStarted )
|
|
{
|
|
if( EndPage( hPrintDC ) <= 0 )
|
|
{
|
|
// if iErr not already set then set it to the new error code.
|
|
if( iErr == 0 )
|
|
{
|
|
iErr= GetLastError();
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
if( fDocStarted )
|
|
{
|
|
if( fAbort ) {
|
|
AbortDoc( hPrintDC );
|
|
}
|
|
else {
|
|
if( EndDoc( hPrintDC ) <= 0 )
|
|
{
|
|
// if iErr not already set then set it to the new error code.
|
|
if (iErr == 0)
|
|
{
|
|
iErr= GetLastError();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
DeleteDC( hPrintDC );
|
|
|
|
DestroyAbortWnd();
|
|
|
|
SetCursor( hStdCursor );
|
|
|
|
if (!fAbort)
|
|
{
|
|
return( iErr );
|
|
}
|
|
else
|
|
{
|
|
return( SP_USERABORT );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
if( (iErr= SetAbortProc (hPrintDC, AbortProc)) < 0 )
|
|
{
|
|
goto ExitWithThisError;
|
|
}
|
|
|
|
// get printer to MLE text
|
|
hText= (HANDLE) SendMessage( hwndEdit, EM_GETHANDLE, 0, 0 );
|
|
if( !hText )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
pStartText= LocalLock( hText );
|
|
if( !pStartText )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
GetWindowText( hwndSP, msgbuf, CharSizeOf(msgbuf) );
|
|
|
|
EnableWindow( hwndSP, FALSE ); // Disable window to prevent reentrancy
|
|
|
|
hAbortDlgWnd= CreateDialog( hInstanceSP,
|
|
(LPTSTR) MAKEINTRESOURCE(IDD_ABORTPRINT),
|
|
hwndSP,
|
|
AbortDlgProc);
|
|
|
|
if( !hAbortDlgWnd )
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
DocInfo.cbSize= sizeof(DOCINFO);
|
|
DocInfo.lpszDocName= msgbuf;
|
|
DocInfo.lpszOutput= NULL;
|
|
DocInfo.lpszDatatype= NULL; // Type of data used to record print job
|
|
DocInfo.fwType= 0; // not DI_APPBANDING
|
|
|
|
SetLastError(0); // clear error so it reflects errors in the future
|
|
|
|
if( StartDoc( hPrintDC, &DocInfo ) <= 0 )
|
|
{
|
|
iErr = GetLastError();
|
|
goto ExitWithThisError;
|
|
}
|
|
fDocStarted= TRUE;
|
|
|
|
|
|
// Basicly, this is just a loop surrounding the DrawTextEx API.
|
|
// We have to calculate the printable area which will not include
|
|
// the header and footer area.
|
|
{
|
|
INT iTextLeft; // amount of text left to print
|
|
INT iSta; // status
|
|
UINT dwDTFormat; // drawtext flags
|
|
DRAWTEXTPARAMS dtParm; // drawtext control
|
|
RECT rect; // rectangle to draw in
|
|
UINT dwDTRigh = 0; // drawtext flags (RTL)
|
|
|
|
iPageNum= 1;
|
|
fPageStarted= FALSE;
|
|
|
|
// calculate the size of the printable area for the text
|
|
// not including the header and footer
|
|
|
|
ZeroMemory( &rect, sizeof(rect) );
|
|
|
|
rect.left= dxLeft; rect.right= xPrintRes-dxRight;
|
|
rect.top= dyTop; rect.bottom= yPrintRes-dyBottom;
|
|
|
|
if( *chPageText[HEADER] != 0 )
|
|
{
|
|
rect.top += yPrintChar;
|
|
}
|
|
|
|
if( *chPageText[FOOTER] != 0 )
|
|
{
|
|
rect.bottom -= yPrintChar;
|
|
}
|
|
|
|
iTextLeft= lstrlen(pStartText);
|
|
|
|
//Get the edit control direction.
|
|
if (GetWindowLong(hwndEdit, GWL_EXSTYLE) & WS_EX_RTLREADING)
|
|
dwDTRigh = DT_RIGHT | DT_RTLREADING;
|
|
|
|
|
|
while( !fAbort && (iTextLeft>0) )
|
|
{
|
|
#define MAXSTATUS 100
|
|
TCHAR szPagePrinting[MAXSTATUS+1];
|
|
|
|
// update abort dialog box to inform user where we are in the printing
|
|
_sntprintf( szPagePrinting, MAXSTATUS, szCurrentPage, iPageNum );
|
|
SetDlgItemText( hAbortDlgWnd, ID_PAGENUMBER, szPagePrinting );
|
|
|
|
PrintHeaderFooter( hPrintDC, HEADER );
|
|
|
|
ZeroMemory( &dtParm, sizeof(dtParm) );
|
|
|
|
dtParm.cbSize= sizeof(dtParm);
|
|
dtParm.iTabLength= tabSize;
|
|
|
|
dwDTFormat= DT_EDITCONTROL | DT_LEFT | DT_EXPANDTABS | DT_NOPREFIX |
|
|
DT_WORDBREAK | dwDTRigh | 0;
|
|
|
|
if( StartPage( hPrintDC ) <= 0 )
|
|
{
|
|
iErr= GetLastError();
|
|
goto ExitWithThisError;
|
|
}
|
|
fPageStarted= TRUE;
|
|
|
|
#ifdef DBGPRINT
|
|
ShowMargins(hPrintDC);
|
|
#endif
|
|
|
|
/* Ignore errors in printing. EndPage or StartPage will find them */
|
|
iSta= DrawTextEx( hPrintDC,
|
|
pStartText,
|
|
iTextLeft,
|
|
&rect,
|
|
dwDTFormat,
|
|
&dtParm);
|
|
|
|
PrintHeaderFooter( hPrintDC, FOOTER );
|
|
|
|
if( EndPage( hPrintDC ) <= 0 )
|
|
{
|
|
iErr= GetLastError();
|
|
goto ExitWithThisError;
|
|
}
|
|
fPageStarted= FALSE;
|
|
|
|
iPageNum++;
|
|
|
|
// if we can't print a single character (too big perhaps)
|
|
// just bail now.
|
|
if( dtParm.uiLengthDrawn == 0 )
|
|
{
|
|
goto FontTooBig;
|
|
}
|
|
|
|
pStartText += dtParm.uiLengthDrawn;
|
|
iTextLeft -= dtParm.uiLengthDrawn;
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
iErr=0; // no errors
|
|
goto ExitWithThisError;
|
|
|
|
}
|
|
|
|
|
|
VOID DestroyAbortWnd (void)
|
|
{
|
|
EnableWindow(hwndSP, TRUE);
|
|
DestroyWindow(hAbortDlgWnd);
|
|
hAbortDlgWnd = NULL;
|
|
}
|
|
|
|
|
|
|
|
const DWORD s_PageSetupHelpIDs[] = {
|
|
ID_HEADER_LABEL, IDH_PAGE_HEADER,
|
|
ID_HEADER, IDH_PAGE_HEADER,
|
|
ID_FOOTER_LABEL, IDH_PAGE_FOOTER,
|
|
ID_FOOTER, IDH_PAGE_FOOTER,
|
|
0, 0
|
|
};
|
|
|
|
/*******************************************************************************
|
|
*
|
|
* PageSetupHookProc
|
|
*
|
|
* DESCRIPTION:
|
|
* Callback procedure for the PageSetup common dialog box.
|
|
*
|
|
* PARAMETERS:
|
|
* hWnd, handle of PageSetup window.
|
|
* Message,
|
|
* wParam,
|
|
* lParam,
|
|
* (returns),
|
|
*
|
|
*******************************************************************************/
|
|
|
|
UINT_PTR CALLBACK PageSetupHookProc(
|
|
HWND hWnd,
|
|
UINT Message,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
{
|
|
|
|
INT id; /* ID of dialog edit controls */
|
|
POINT pt;
|
|
|
|
switch (Message)
|
|
{
|
|
|
|
case WM_INITDIALOG:
|
|
for (id = ID_HEADER; id <= ID_FOOTER; id++)
|
|
{
|
|
SendDlgItemMessage(hWnd, id, EM_LIMITTEXT, PT_LEN-1, 0L);
|
|
SetDlgItemText(hWnd, id, chPageText[id - ID_HEADER]);
|
|
}
|
|
|
|
SendDlgItemMessage(hWnd, ID_HEADER, EM_SETSEL, 0,
|
|
MAKELONG(0, PT_LEN-1));
|
|
return TRUE;
|
|
|
|
case WM_DESTROY:
|
|
// We don't know if the user hit OK or Cancel, so we don't
|
|
// want to replace our real copies until we know! We _should_ get
|
|
// a notification from the common dialog code!
|
|
for( id = ID_HEADER; id <= ID_FOOTER; id++ )
|
|
{
|
|
GetDlgItemText(hWnd, id, chPageTextTemp[id - ID_HEADER],PT_LEN);
|
|
}
|
|
break;
|
|
|
|
case WM_HELP:
|
|
//
|
|
// We only want to intercept help messages for controls that we are
|
|
// responsible for.
|
|
//
|
|
|
|
id = GetDlgCtrlID(((LPHELPINFO) lParam)-> hItemHandle);
|
|
|
|
if (id < ID_HEADER || id > ID_FOOTER_LABEL)
|
|
break;
|
|
|
|
WinHelp(((LPHELPINFO) lParam)-> hItemHandle, szHelpFile,
|
|
HELP_WM_HELP, (UINT_PTR) (LPVOID) s_PageSetupHelpIDs);
|
|
return TRUE;
|
|
|
|
case WM_CONTEXTMENU:
|
|
//
|
|
// If the user clicks on any of our labels, then the wParam will
|
|
// be the hwnd of the dialog, not the static control. WinHelp()
|
|
// handles this, but because we hook the dialog, we must catch it
|
|
// first.
|
|
//
|
|
if( hWnd == (HWND) wParam )
|
|
{
|
|
|
|
GetCursorPos(&pt);
|
|
ScreenToClient(hWnd, &pt);
|
|
wParam = (WPARAM) ChildWindowFromPoint(hWnd, pt);
|
|
|
|
}
|
|
|
|
//
|
|
// We only want to intercept help messages for controls that we are
|
|
// responsible for.
|
|
//
|
|
|
|
id = GetDlgCtrlID((HWND) wParam);
|
|
|
|
if (id < ID_HEADER || id > ID_FOOTER_LABEL)
|
|
break;
|
|
|
|
WinHelp((HWND) wParam, szHelpFile, HELP_CONTEXTMENU,
|
|
(UINT_PTR) (LPVOID) s_PageSetupHelpIDs);
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
/***************************************************************************
|
|
* VOID TranslateString(TCHAR *src)
|
|
*
|
|
* purpose:
|
|
* translate a header/footer strings
|
|
*
|
|
* supports the following:
|
|
*
|
|
* && insert a & char
|
|
* &f current file name or (untitled)
|
|
* &d date in Day Month Year
|
|
* &t time
|
|
* &p page number
|
|
* &p+num set first page number to num
|
|
*
|
|
* Alignment:
|
|
* &l, &c, &r for left, center, right
|
|
*
|
|
* params:
|
|
* IN/OUT src this is the string to translate
|
|
*
|
|
*
|
|
* used by:
|
|
* Header Footer stuff
|
|
*
|
|
* uses:
|
|
* lots of c lib stuff
|
|
*
|
|
***************************************************************************/
|
|
|
|
|
|
VOID TranslateString (TCHAR * src)
|
|
{
|
|
// File, Page, Time, Date, Center, Right, Left
|
|
// these *never* change so don't put into resources for localizers
|
|
TCHAR letters[15]=TEXT("fFpPtTdDcCrRlL");
|
|
TCHAR buf[MAX_PATH];
|
|
TCHAR *ptr;
|
|
INT page;
|
|
INT nAlign=CENTER; // current string to add chars to
|
|
INT foo;
|
|
INT nIndex[RIGHT+1]; // current lengths of (left,center,right)
|
|
struct tm *newtime;
|
|
time_t long_time;
|
|
INT iLen; // length of strings
|
|
|
|
nIndex[LEFT] = 0;
|
|
nIndex[CENTER] = 0;
|
|
nIndex[RIGHT] = 0;
|
|
|
|
/* Get the time we need in case we use &t. */
|
|
time (&long_time);
|
|
newtime = localtime (&long_time);
|
|
|
|
|
|
while (*src) /* look at all of source */
|
|
{
|
|
while (*src && *src != TEXT('&'))
|
|
{
|
|
chBuff[nAlign][nIndex[nAlign]] = *src++;
|
|
nIndex[nAlign] += 1;
|
|
}
|
|
|
|
if (*src == TEXT('&')) /* is it the escape char? */
|
|
{
|
|
src++;
|
|
|
|
if (*src == letters[0] || *src == letters[1])
|
|
{ /* &f file name (no path) */
|
|
if (!fUntitled)
|
|
{
|
|
GetFileTitle(szFileName, buf, CharSizeOf(buf));
|
|
}
|
|
else
|
|
{
|
|
lstrcpy(buf, szUntitled);
|
|
}
|
|
|
|
/* Copy to the currently aligned string. */
|
|
if( nIndex[nAlign] + lstrlen(buf) < MAXTITLE )
|
|
{
|
|
lstrcpy( chBuff[nAlign] + nIndex[nAlign], buf );
|
|
|
|
/* Update insertion position. */
|
|
nIndex[nAlign] += lstrlen (buf);
|
|
}
|
|
|
|
}
|
|
else if (*src == letters[2] || *src == letters[3]) /* &P or &P+num page */
|
|
{
|
|
src++;
|
|
page = 0;
|
|
if (*src == TEXT('+')) /* &p+num case */
|
|
{
|
|
src++;
|
|
while (_istdigit(*src))
|
|
{
|
|
/* Convert to int on-the-fly*/
|
|
page = (10*page) + (*src) - TEXT('0');
|
|
src++;
|
|
}
|
|
}
|
|
|
|
wsprintf( buf, TEXT("%d"), iPageNum+page ); // convert to chars
|
|
|
|
if( nIndex[nAlign] + lstrlen(buf) < MAXTITLE )
|
|
{
|
|
lstrcpy( chBuff[nAlign] + nIndex[nAlign], buf );
|
|
nIndex[nAlign] += lstrlen (buf);
|
|
}
|
|
src--;
|
|
}
|
|
else if (*src == letters[4] || *src == letters[5]) /* &t time */
|
|
{
|
|
iLen= lstrlen( szFormattedTime );
|
|
|
|
/* extract time */
|
|
if( nIndex[nAlign] + iLen < MAXTITLE )
|
|
{
|
|
_tcsncpy (chBuff[nAlign] + nIndex[nAlign], szFormattedTime, iLen);
|
|
nIndex[nAlign] += iLen;
|
|
}
|
|
}
|
|
else if (*src == letters[6] || *src == letters[7]) /* &d date */
|
|
{
|
|
iLen= lstrlen( szFormattedDate );
|
|
|
|
/* extract day month day */
|
|
if( nIndex[nAlign] + iLen < MAXTITLE )
|
|
{
|
|
_tcsncpy (chBuff[nAlign] + nIndex[nAlign], szFormattedDate, iLen);
|
|
nIndex[nAlign] += iLen;
|
|
}
|
|
}
|
|
else if (*src == TEXT('&')) /* quote a single & */
|
|
{
|
|
if( nIndex[nAlign] + 1 < MAXTITLE )
|
|
{
|
|
chBuff[nAlign][nIndex[nAlign]] = TEXT('&');
|
|
nIndex[nAlign] += 1;
|
|
}
|
|
}
|
|
/* Set the alignment for whichever has last occured. */
|
|
else if (*src == letters[8] || *src == letters[9]) /* &c center */
|
|
nAlign=CENTER;
|
|
else if (*src == letters[10] || *src == letters[11]) /* &r right */
|
|
nAlign=RIGHT;
|
|
else if (*src == letters[12] || *src == letters[13]) /* &d date */
|
|
nAlign=LEFT;
|
|
|
|
src++;
|
|
}
|
|
}
|
|
/* Make sure all strings are null-terminated. */
|
|
for (nAlign= LEFT; nAlign <= RIGHT ; nAlign++)
|
|
chBuff[nAlign][nIndex[nAlign]] = (TCHAR) 0;
|
|
|
|
}
|
|
|
|
/* GetPrinterDC() - returns printer DC or INVALID_HANDLE_VALUE if none. */
|
|
|
|
HANDLE GetPrinterDC (VOID)
|
|
{
|
|
LPDEVMODE lpDevMode;
|
|
LPDEVNAMES lpDevNames;
|
|
HDC hDC;
|
|
|
|
|
|
if( !g_PageSetupDlg.hDevNames ) /* Retrieve default printer if none selected. */
|
|
{
|
|
g_PageSetupDlg.Flags |= PSD_RETURNDEFAULT;
|
|
PageSetupDlg(&g_PageSetupDlg);
|
|
g_PageSetupDlg.Flags &= ~PSD_RETURNDEFAULT;
|
|
}
|
|
|
|
if( !g_PageSetupDlg.hDevNames )
|
|
{
|
|
MessageBox( hwndSP, szLoadDrvFail, szNN, MB_APPLMODAL | MB_OK | MB_ICONEXCLAMATION);
|
|
return INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
lpDevNames= (LPDEVNAMES) GlobalLock (g_PageSetupDlg.hDevNames);
|
|
|
|
|
|
lpDevMode= NULL;
|
|
|
|
if( g_PageSetupDlg.hDevMode )
|
|
lpDevMode= (LPDEVMODE) GlobalLock( g_PageSetupDlg.hDevMode );
|
|
|
|
/* For pre 3.0 Drivers,hDevMode will be null from Commdlg so lpDevMode
|
|
* will be NULL after GlobalLock()
|
|
*/
|
|
|
|
/* The lpszOutput name is null so CreateDC will use the current setting
|
|
* from PrintMan.
|
|
*/
|
|
|
|
hDC= CreateDC (((LPTSTR)lpDevNames)+lpDevNames->wDriverOffset,
|
|
((LPTSTR)lpDevNames)+lpDevNames->wDeviceOffset,
|
|
NULL,
|
|
lpDevMode);
|
|
|
|
GlobalUnlock( g_PageSetupDlg.hDevNames );
|
|
|
|
if( g_PageSetupDlg.hDevMode )
|
|
GlobalUnlock( g_PageSetupDlg.hDevMode );
|
|
|
|
|
|
if( hDC == NULL )
|
|
{
|
|
MessageBox( hwndSP, szLoadDrvFail, szNN, MB_APPLMODAL | MB_OK | MB_ICONEXCLAMATION);
|
|
return INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
return hDC;
|
|
}
|
|
|
|
|
|
/* GetNonDefPrinterDC() - returns printer DC or INVALID_HANDLE_VALUE if none. */
|
|
/* using the name of the Printer server */
|
|
|
|
HANDLE GetNonDefPrinterDC (VOID)
|
|
{
|
|
HDC hDC;
|
|
HANDLE hPrinter;
|
|
DWORD dwBuf;
|
|
DRIVER_INFO_1 *di1;
|
|
|
|
|
|
|
|
// open the printer and retrieve the driver name.
|
|
if (!OpenPrinter(szPrinterName, &hPrinter, NULL))
|
|
{
|
|
return INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
// get the buffer size.
|
|
GetPrinterDriver(hPrinter, NULL, 1, NULL, 0, &dwBuf);
|
|
di1 = (DRIVER_INFO_1 *) LocalAlloc(LPTR, dwBuf);
|
|
if (!di1)
|
|
{
|
|
ClosePrinter(hPrinter);
|
|
return INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
if (!GetPrinterDriver(hPrinter, NULL, 1, (LPBYTE) di1, dwBuf, &dwBuf))
|
|
{
|
|
LocalFree(di1);
|
|
ClosePrinter(hPrinter);
|
|
return INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
// Initialize the PageSetup dlg to default values.
|
|
// using default printer's value for another printer !!
|
|
g_PageSetupDlg.Flags |= PSD_RETURNDEFAULT;
|
|
PageSetupDlg(&g_PageSetupDlg);
|
|
g_PageSetupDlg.Flags &= ~PSD_RETURNDEFAULT;
|
|
|
|
// create printer dc with default initialization.
|
|
hDC= CreateDC (di1->pName, szPrinterName, NULL, NULL);
|
|
|
|
// cleanup.
|
|
LocalFree(di1);
|
|
ClosePrinter(hPrinter);
|
|
|
|
if( hDC == NULL )
|
|
{
|
|
MessageBox( hwndSP, szLoadDrvFail, szNN, MB_APPLMODAL | MB_OK | MB_ICONEXCLAMATION);
|
|
return INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
return hDC;
|
|
}
|
|
|
|
|
|
/* PrintIt() - print the file, giving popup if some error */
|
|
|
|
void PrintIt(PRINT_DIALOG_TYPE type)
|
|
{
|
|
INT iError;
|
|
TCHAR* szMsg= NULL;
|
|
TCHAR msg[400]; // message info on error
|
|
|
|
/* print the file */
|
|
|
|
iError= NpPrint( type );
|
|
|
|
if(( iError != 0) &&
|
|
( iError != SP_APPABORT ) &&
|
|
( iError != SP_USERABORT ) )
|
|
{
|
|
// translate any known spooler errors
|
|
if( iError == SP_OUTOFDISK ) iError= ERROR_DISK_FULL;
|
|
if( iError == SP_OUTOFMEMORY ) iError= ERROR_OUTOFMEMORY;
|
|
if( iError == SP_ERROR ) iError= GetLastError();
|
|
/* SP_NOTREPORTED not handled. Does it happen? */
|
|
|
|
|
|
//
|
|
// iError may be 0 because the user aborted the printing.
|
|
// Just ignore.
|
|
//
|
|
|
|
if( iError == 0 ) return;
|
|
|
|
// Get system to give reasonable error message
|
|
// These will also be internationalized.
|
|
|
|
if(!FormatMessage( FORMAT_MESSAGE_IGNORE_INSERTS |
|
|
FORMAT_MESSAGE_FROM_SYSTEM,
|
|
NULL,
|
|
iError,
|
|
GetUserDefaultLangID(),
|
|
msg, // where message will end up
|
|
CharSizeOf(msg), NULL ) )
|
|
{
|
|
szMsg= szCP; // couldn't get system to say; give generic msg
|
|
}
|
|
else
|
|
{
|
|
szMsg= msg;
|
|
}
|
|
|
|
AlertBox( hwndSP, szNN, szMsg, fUntitled ? szUntitled : szFileName,
|
|
MB_APPLMODAL | MB_OK | MB_ICONEXCLAMATION);
|
|
}
|
|
}
|