428 lines
13 KiB
C
428 lines
13 KiB
C
|
/************************************************************/
|
|||
|
/* Windows Write, Copyright 1985-1990 Microsoft Corporation */
|
|||
|
/************************************************************/
|
|||
|
|
|||
|
/* This file contains the dialog box routines for the print dialog box and the
|
|||
|
printer initialization code. */
|
|||
|
|
|||
|
#define NOGDICAPMASKS
|
|||
|
#define NOVIRTUALKEYCODES
|
|||
|
#define NOWINSTYLES
|
|||
|
#define NOSYSMETRICS
|
|||
|
#define NOMENUS
|
|||
|
#define NOCLIPBOARD
|
|||
|
#define NOKEYSTATE
|
|||
|
#define NOSYSCOMMANDS
|
|||
|
#define NORASTEROPS
|
|||
|
#define NOSHOWWINDOW
|
|||
|
#define NOATOM
|
|||
|
#define NOCREATESTRUCT
|
|||
|
#define NODRAWTEXT
|
|||
|
#define NOMB
|
|||
|
#define NOMEMMGR
|
|||
|
#define NOMETAFILE
|
|||
|
#define NOWH
|
|||
|
#define NOWNDCLASS
|
|||
|
#define NOSOUND
|
|||
|
#define NOCOLOR
|
|||
|
#define NOSCROLL
|
|||
|
#define NOCOMM
|
|||
|
#include <windows.h>
|
|||
|
#include "mw.h"
|
|||
|
#include "cmddefs.h"
|
|||
|
#include "dlgdefs.h"
|
|||
|
#include "str.h"
|
|||
|
#include "printdef.h"
|
|||
|
#include "fmtdefs.h"
|
|||
|
#include "propdefs.h"
|
|||
|
|
|||
|
|
|||
|
fnPrPrinter()
|
|||
|
{
|
|||
|
/* This routine is the outside world's interface to the print code. */
|
|||
|
|
|||
|
extern HWND hParentWw;
|
|||
|
extern HANDLE hMmwModInstance;
|
|||
|
extern CHAR *vpDlgBuf;
|
|||
|
extern int docCur;
|
|||
|
CHAR rgbDlgBuf[sizeof(int) + 2 * sizeof(BOOL)];
|
|||
|
#ifdef INEFFLOCKDOWN
|
|||
|
extern FARPROC lpDialogPrint;
|
|||
|
#else
|
|||
|
BOOL far PASCAL DialogPrint(HWND, unsigned, WORD, LONG);
|
|||
|
FARPROC lpDialogPrint;
|
|||
|
if (!(lpDialogPrint = MakeProcInstance(DialogPrint, hMmwModInstance)))
|
|||
|
{
|
|||
|
WinFailure();
|
|||
|
return;
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
vpDlgBuf = &rgbDlgBuf[0];
|
|||
|
switch (OurDialogBox(hMmwModInstance, MAKEINTRESOURCE(dlgPrint), hParentWw,
|
|||
|
lpDialogPrint))
|
|||
|
{
|
|||
|
case idiOk:
|
|||
|
/* Force all of the windows to clean up their act. */
|
|||
|
DispatchPaintMsg();
|
|||
|
|
|||
|
/* At this point, we have the following :
|
|||
|
vfPrPages = true if print page range else print all pages
|
|||
|
vpgnBegin = starting page number (if vfPrPages)
|
|||
|
vpgnEnd = ending page number (if vfPrPages)
|
|||
|
vcCopies = number of copies to print */
|
|||
|
PrintDoc(docCur, TRUE);
|
|||
|
break;
|
|||
|
|
|||
|
case -1:
|
|||
|
/* We didn't even have enough memory to create the dialog box. */
|
|||
|
#ifdef WIN30
|
|||
|
WinFailure();
|
|||
|
#else
|
|||
|
Error(IDPMTNoMemory);
|
|||
|
#endif
|
|||
|
break;
|
|||
|
}
|
|||
|
#ifndef INEFFLOCKDOWN
|
|||
|
FreeProcInstance(lpDialogPrint);
|
|||
|
#endif
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
BOOL far PASCAL DialogPrint( hDlg, message, wParam, lParam )
|
|||
|
HWND hDlg;
|
|||
|
unsigned message;
|
|||
|
WORD wParam;
|
|||
|
LONG lParam;
|
|||
|
{
|
|||
|
/* This routine handles input to the Print dialog box. */
|
|||
|
extern CHAR *vpDlgBuf;
|
|||
|
extern int vfPrPages; /* true if print page range */
|
|||
|
extern int vpgnBegin; /* starting page number to print */
|
|||
|
extern int vpgnEnd; /* ending page number to print */
|
|||
|
extern int vcCopies; /* nubmer of copies to print */
|
|||
|
extern BOOL vfPrinterValid;
|
|||
|
extern HDC vhDCPrinter;
|
|||
|
extern int vfDraftMode;
|
|||
|
extern HWND vhWndMsgBoxParent;
|
|||
|
extern ferror;
|
|||
|
extern HCURSOR vhcArrow;
|
|||
|
extern int vfCursorVisible;
|
|||
|
extern CHAR (**hszPrinter)[];
|
|||
|
extern CHAR (**hszPrDriver)[];
|
|||
|
extern CHAR (**hszPrPort)[];
|
|||
|
|
|||
|
int *pidiRBDown = (int *)vpDlgBuf;
|
|||
|
BOOL *pfDraftMode = (BOOL *)(vpDlgBuf + sizeof(int));
|
|||
|
BOOL *pfDraftSupport = (BOOL *)(vpDlgBuf + sizeof(int) + sizeof(BOOL));
|
|||
|
int iEscape;
|
|||
|
CHAR szPrDescrip[cchMaxProfileSz];
|
|||
|
|
|||
|
switch (message)
|
|||
|
{
|
|||
|
case WM_INITDIALOG:
|
|||
|
BuildPrSetupSz(szPrDescrip, &(**hszPrinter)[0], &(**hszPrPort)[0]);
|
|||
|
SetDlgItemText(hDlg, idiPrtDest, (LPSTR)szPrDescrip);
|
|||
|
SetDlgItemText(hDlg, idiPrtCopies, (LPSTR)"1");
|
|||
|
SelectIdiText(hDlg, idiPrtCopies);
|
|||
|
if (vfPrPages)
|
|||
|
{
|
|||
|
*pidiRBDown = idiPrtFrom;
|
|||
|
SetDlgItemInt(hDlg, idiPrtPageFrom, vpgnBegin, TRUE);
|
|||
|
SetDlgItemInt(hDlg, idiPrtPageTo, vpgnEnd, TRUE);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
*pidiRBDown = idiPrtAll;
|
|||
|
}
|
|||
|
|
|||
|
iEscape = DRAFTMODE;
|
|||
|
if (*pfDraftSupport = vfPrinterValid && vhDCPrinter &&
|
|||
|
Escape(vhDCPrinter, QUERYESCSUPPORT, sizeof(int),
|
|||
|
(LPSTR)&iEscape, (LPSTR)NULL))
|
|||
|
{
|
|||
|
CheckDlgButton(hDlg, idiPrtDraft, *pfDraftMode = vfDraftMode);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
EnableWindow(GetDlgItem(hDlg, idiPrtDraft), FALSE);
|
|||
|
if (!vhDCPrinter) /* we've got a timing thing whereby they
|
|||
|
managed to get into the print dialog
|
|||
|
inbetween the time printer.setup had
|
|||
|
unhooked the old printer and the hookup
|
|||
|
of the new one! ..pault */
|
|||
|
EnableWindow(GetDlgItem(hDlg, idiOk), FALSE);
|
|||
|
}
|
|||
|
|
|||
|
CheckDlgButton(hDlg, *pidiRBDown, TRUE);
|
|||
|
EnableOtherModeless(FALSE);
|
|||
|
break;
|
|||
|
|
|||
|
case WM_SETVISIBLE:
|
|||
|
if (wParam)
|
|||
|
{
|
|||
|
EndLongOp(vhcArrow);
|
|||
|
}
|
|||
|
return(FALSE);
|
|||
|
|
|||
|
case WM_ACTIVATE:
|
|||
|
if (wParam)
|
|||
|
{
|
|||
|
vhWndMsgBoxParent = hDlg;
|
|||
|
}
|
|||
|
if (vfCursorVisible)
|
|||
|
{
|
|||
|
ShowCursor(wParam);
|
|||
|
}
|
|||
|
return (FALSE);
|
|||
|
|
|||
|
case WM_COMMAND:
|
|||
|
switch (wParam)
|
|||
|
{
|
|||
|
BOOL fPages;
|
|||
|
int pgnBegin;
|
|||
|
int pgnEnd;
|
|||
|
int cCopies;
|
|||
|
|
|||
|
case idiOk:
|
|||
|
if (fPages = (*pidiRBDown == idiPrtFrom))
|
|||
|
{
|
|||
|
/* Get the range of pages to print. */
|
|||
|
if (!WPwFromItW3Id(&pgnBegin, hDlg, idiPrtPageFrom,
|
|||
|
pgnMin, pgnMax, wNormal, IDPMTNPI))
|
|||
|
{
|
|||
|
/* Reset error condition, so as to report any further error.
|
|||
|
*/
|
|||
|
ferror = FALSE;
|
|||
|
return(TRUE);
|
|||
|
}
|
|||
|
if (!WPwFromItW3Id(&pgnEnd, hDlg, idiPrtPageTo,
|
|||
|
pgnMin, pgnMax, wNormal, IDPMTNPI))
|
|||
|
{
|
|||
|
/* Reset error condition, so as to report any further error.
|
|||
|
*/
|
|||
|
ferror = FALSE;
|
|||
|
return(TRUE);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/* Get the number of copies to print. */
|
|||
|
if (!WPwFromItW3IdFUt(&cCopies, hDlg, idiPrtCopies, 1, 32767,
|
|||
|
wNormal, IDPMTNPI, FALSE, 0
|
|||
|
))
|
|||
|
{
|
|||
|
/* Reset error condition, so as to report any further error. */
|
|||
|
ferror = FALSE;
|
|||
|
return(TRUE);
|
|||
|
}
|
|||
|
|
|||
|
/* If we have gotten this far, then everything must be okey-dokey.
|
|||
|
*/
|
|||
|
vfDraftMode = *pfDraftSupport ? *pfDraftMode : FALSE;
|
|||
|
if (vfPrPages = fPages)
|
|||
|
{
|
|||
|
vpgnBegin = pgnBegin;
|
|||
|
vpgnEnd = pgnEnd;
|
|||
|
}
|
|||
|
vcCopies = cCopies;
|
|||
|
|
|||
|
case idiCancel:
|
|||
|
OurEndDialog(hDlg, wParam);
|
|||
|
break;
|
|||
|
|
|||
|
case idiPrtPageFrom:
|
|||
|
case idiPrtPageTo:
|
|||
|
if (HIWORD(lParam) == EN_CHANGE)
|
|||
|
{
|
|||
|
if (SendMessage(LOWORD(lParam), WM_GETTEXTLENGTH, 0, 0L) &&
|
|||
|
*pidiRBDown != idiPrtFrom)
|
|||
|
{
|
|||
|
CheckDlgButton(hDlg, *pidiRBDown, FALSE);
|
|||
|
CheckDlgButton(hDlg, *pidiRBDown = idiPrtFrom, TRUE);
|
|||
|
}
|
|||
|
return(TRUE);
|
|||
|
}
|
|||
|
return(FALSE);
|
|||
|
|
|||
|
case idiPrtAll:
|
|||
|
case idiPrtFrom:
|
|||
|
CheckDlgButton(hDlg, *pidiRBDown, FALSE);
|
|||
|
CheckDlgButton(hDlg, *pidiRBDown = wParam, TRUE);
|
|||
|
|
|||
|
// set focus to the edit field automatically
|
|||
|
|
|||
|
if (wParam == idiPrtFrom)
|
|||
|
SetFocus(GetDlgItem(hDlg, idiPrtPageFrom));
|
|||
|
|
|||
|
break;
|
|||
|
|
|||
|
case idiPrtDraft:
|
|||
|
CheckDlgButton(hDlg, wParam, *pfDraftMode = !(*pfDraftMode));
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
return(FALSE);
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
return(FALSE);
|
|||
|
}
|
|||
|
return(TRUE);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
BOOL FInitHeaderFooter(fHeader, ppgn, phrgpld, pcpld)
|
|||
|
BOOL fHeader;
|
|||
|
unsigned *ppgn;
|
|||
|
struct PLD (***phrgpld)[];
|
|||
|
int *pcpld;
|
|||
|
{
|
|||
|
/* This routine initializes the array of print line descriptors used in
|
|||
|
positioning the header/footer on the printed page. FALSE is returned if an
|
|||
|
error occurs; TRUE otherwise. */
|
|||
|
|
|||
|
extern typeCP cpMinHeader;
|
|||
|
extern typeCP cpMacHeader;
|
|||
|
extern typeCP cpMinFooter;
|
|||
|
extern typeCP cpMacFooter;
|
|||
|
extern int docCur;
|
|||
|
extern struct PAP vpapAbs;
|
|||
|
extern struct SEP vsepAbs;
|
|||
|
extern int dxaPrOffset;
|
|||
|
extern int dyaPrOffset;
|
|||
|
extern int dxpPrPage;
|
|||
|
extern int dxaPrPage;
|
|||
|
extern int dypPrPage;
|
|||
|
extern int dyaPrPage;
|
|||
|
extern struct FLI vfli;
|
|||
|
extern int vfOutOfMemory;
|
|||
|
|
|||
|
typeCP cpMin;
|
|||
|
typeCP cpMac;
|
|||
|
|
|||
|
/* Get the cpMin and the cpMac for the header/footer. */
|
|||
|
if (fHeader)
|
|||
|
{
|
|||
|
cpMin = cpMinHeader;
|
|||
|
cpMac = cpMacHeader;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
cpMin = cpMinFooter;
|
|||
|
cpMac = cpMacFooter;
|
|||
|
}
|
|||
|
|
|||
|
/* Is there a header/footer. */
|
|||
|
if (cpMac - cpMin > ccpEol)
|
|||
|
{
|
|||
|
int cpld = 0;
|
|||
|
int cpldReal = 0;
|
|||
|
int cpldMax;
|
|||
|
int xp;
|
|||
|
int yp;
|
|||
|
int ichCp = 0;
|
|||
|
typeCP cpMacDoc = CpMacText(docCur);
|
|||
|
|
|||
|
/* Compute the page number of the start of the headers/footers. */
|
|||
|
CacheSect(docCur, cpMin);
|
|||
|
if ((*ppgn = vsepAbs.pgnStart) == pgnNil)
|
|||
|
{
|
|||
|
*ppgn = 1;
|
|||
|
}
|
|||
|
|
|||
|
/* Does the header/footer appear on the first page. */
|
|||
|
CachePara(docCur, cpMin);
|
|||
|
if (!(vpapAbs.rhc & RHC_fFirst))
|
|||
|
{
|
|||
|
(*ppgn)++;
|
|||
|
}
|
|||
|
|
|||
|
/* Calculate the bounds of the header/footer in pixels. */
|
|||
|
xp = MultDiv(vsepAbs.xaLeft - dxaPrOffset, dxpPrPage, dxaPrPage);
|
|||
|
yp = fHeader ? MultDiv(vsepAbs.yaRH1 - dyaPrOffset, dypPrPage,
|
|||
|
dyaPrPage) : 0;
|
|||
|
|
|||
|
/* Initialize the array of print line descriptors for the header/footer.
|
|||
|
*/
|
|||
|
if (FNoHeap(*phrgpld = (struct PLD (**)[])HAllocate((cpldMax = cpldRH) *
|
|||
|
cwPLD)))
|
|||
|
{
|
|||
|
*phrgpld = NULL;
|
|||
|
return (FALSE);
|
|||
|
}
|
|||
|
|
|||
|
/* We now have to calculate the array of print line descriptors for the
|
|||
|
header/footer. */
|
|||
|
cpMac -= ccpEol;
|
|||
|
while (cpMin < cpMac)
|
|||
|
{
|
|||
|
/* Format this line of the header/footer for the printer. */
|
|||
|
FormatLine(docCur, cpMin, ichCp, cpMacDoc, flmPrinting);
|
|||
|
|
|||
|
/* Bail out if an error occurred. */
|
|||
|
if (vfOutOfMemory)
|
|||
|
{
|
|||
|
return (FALSE);
|
|||
|
}
|
|||
|
|
|||
|
/* Is the array of print line descriptors big enough? */
|
|||
|
if (cpld >= cpldMax && !FChngSizeH(*phrgpld, (cpldMax += cpldRH) *
|
|||
|
cwPLD, FALSE))
|
|||
|
{
|
|||
|
return (FALSE);
|
|||
|
}
|
|||
|
|
|||
|
/* Fill the print line descriptor for this line. */
|
|||
|
{
|
|||
|
register struct PLD *ppld = &(***phrgpld)[cpld++];
|
|||
|
|
|||
|
ppld->cp = cpMin;
|
|||
|
ppld->ichCp = ichCp;
|
|||
|
ppld->rc.left = xp + vfli.xpLeft;
|
|||
|
ppld->rc.right = xp + vfli.xpReal;
|
|||
|
ppld->rc.top = yp;
|
|||
|
ppld->rc.bottom = yp + vfli.dypLine;
|
|||
|
}
|
|||
|
|
|||
|
/* Keep track of the non-blank lines in the header/footer */
|
|||
|
if ((vfli.ichReal > 0) || vfli.fGraphics)
|
|||
|
{
|
|||
|
cpldReal = cpld;
|
|||
|
}
|
|||
|
|
|||
|
/* Bump the counters. */
|
|||
|
cpMin = vfli.cpMac;
|
|||
|
ichCp = vfli.ichCpMac;
|
|||
|
yp += vfli.dypLine;
|
|||
|
}
|
|||
|
|
|||
|
/* If this is a footer, then we have to move the positions of the lines
|
|||
|
around so that the footer ends where the user has requested. */
|
|||
|
if (!fHeader && cpldReal > 0)
|
|||
|
{
|
|||
|
register struct PLD *ppld = &(***phrgpld)[cpldReal - 1];
|
|||
|
int dyp = MultDiv(vsepAbs.yaRH2 - dyaPrOffset, dypPrPage, dyaPrPage)
|
|||
|
- ppld->rc.bottom;
|
|||
|
int ipld;
|
|||
|
|
|||
|
for (ipld = cpldReal; ipld > 0; ipld--, ppld--)
|
|||
|
{
|
|||
|
ppld->rc.top += dyp;
|
|||
|
ppld->rc.bottom += dyp;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/* Record the number of non-blank lines in the head/footer. */
|
|||
|
*pcpld = cpldReal;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
/* Indicate there is no header/footer. */
|
|||
|
*ppgn = pgnNil;
|
|||
|
*phrgpld = NULL;
|
|||
|
*pcpld = 0;
|
|||
|
}
|
|||
|
return (TRUE);
|
|||
|
}
|
|||
|
|