Leaked source code of windows server 2003
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.
 
 
 
 
 
 

2773 lines
97 KiB

/*++
Copyright (c) 1990-2003 Microsoft Corporation
All rights reserved
// @@BEGIN_DDKSPLIT
Module Name:
windows\spooler\prtprocs\winprint\emf.c
// @@END_DDKSPLIT
Abstract:
Routines to facilitate printing of EMF jobs.
// @@BEGIN_DDKSPLIT
Author:
Gerrit van Wingerden (gerritv) 5-12-1995
Revision History:
Ramanathan N. Venkatapathy (ramanv) 5-15-1997
Alvin Scholten (alvins) 3-31-00
// @@END_DDKSPLIT
--*/
#include "local.h"
#include "stddef.h"
#include <windef.h>
// @@BEGIN_DDKSPLIT
#include "wingdip.h"
/*
// @@END_DDKSPLIT
#include <winppi.h>
// @@BEGIN_DDKSPLIT
*/
// @@END_DDKSPLIT
#define EMF_DUP_NONE 0
#define EMF_DUP_VERT 1
#define EMF_DUP_HORZ 2
#define EMF_DEGREE_90 0x0001
#define EMF_DEGREE_270 0x0002
#define EMF_DEGREE_SWAP 0x8000
//
// IS_DMSIZE_VALID returns TRUE if the size of the devmode is atleast as much as to
// be able to access field x in it without AV. It is assumed that the devmode is
// atleast the size pdm->dmSize
//
#define IS_DMSIZE_VALID(pdm,x) ( ( (pdm)->dmSize >= (FIELD_OFFSET(DEVMODEW, x ) + sizeof((pdm)->x )))? TRUE:FALSE)
// PAGE_NUMBER is used to save a list of the page numbers to start new sides while
// Reverse Printing.
typedef struct _PAGE_NUMBER {
struct _PAGE_NUMBER *pNext;
DWORD dwPageNumber;
} PAGE_NUMBER, *PPAGE_NUMBER;
typedef struct _UpdateRect {
double top;
double bottom;
double left;
double right;
} UpdateRect;
// The update factors for the different nup options. These factors when multiplied
// with the horizontal and vertical resolutions give the coordinates for the rectangle
// where the EMF page is to be played.
UpdateRect URect21[] = {{0, 0.5, 0, 1},
{0.5, 1, 0, 1}};
UpdateRect URect21R[] = {{0.5, 1, 0, 1},
{0, 0.5, 0, 1}};
UpdateRect URect22[] = {{0, 1, 0, 0.5},
{0, 1, 0.5, 1}};
UpdateRect URect4[] = {{0, 0.5, 0, 0.5},
{0, 0.5, 0.5, 1},
{0.5, 1, 0, 0.5},
{0.5, 1, 0.5, 1}};
UpdateRect URect61[] = {{0, 1.0/3.0, 0, 0.5},
{0, 1.0/3.0, 0.5, 1},
{1.0/3.0, 2.0/3.0, 0, 0.5},
{1.0/3.0, 2.0/3.0, 0.5, 1},
{2.0/3.0, 1, 0, 0.5},
{2.0/3.0, 1, 0.5, 1}};
UpdateRect URect61R[] = {{2.0/3.0, 1, 0, 0.5},
{1.0/3.0, 2.0/3.0, 0, 0.5},
{0, 1.0/3.0, 0, 0.5},
{2.0/3.0, 1, 0.5, 1},
{1.0/3.0, 2.0/3.0, 0.5, 1},
{0, 1.0/3.0, 0.5, 1}};
UpdateRect URect62[] = {{0, 0.5, 0, 1.0/3.0},
{0, 0.5, 1.0/3.0, 2.0/3.0},
{0, 0.5, 2.0/3.0, 1},
{0.5, 1, 0, 1.0/3.0},
{0.5, 1, 1.0/3.0, 2.0/3.0},
{0.5, 1, 2.0/3.0, 1}};
UpdateRect URect62R[] = {{0.5, 1, 0, 1.0/3.0},
{0, 0.5, 0, 1.0/3.0},
{0.5, 1, 1.0/3.0, 2.0/3.0},
{0, 0.5, 1.0/3.0, 2.0/3.0},
{0.5, 1, 2.0/3.0, 1},
{0, 0.5, 2.0/3.0, 1}};
UpdateRect URect9[] = {{0, 1.0/3.0, 0, 1.0/3.0},
{0, 1.0/3.0, 1.0/3.0, 2.0/3.0},
{0, 1.0/3.0, 2.0/3.0, 1},
{1.0/3.0, 2.0/3.0, 0, 1.0/3.0},
{1.0/3.0, 2.0/3.0, 1.0/3.0, 2.0/3.0},
{1.0/3.0, 2.0/3.0, 2.0/3.0, 1},
{2.0/3.0, 1, 0, 1.0/3.0},
{2.0/3.0, 1, 1.0/3.0, 2.0/3.0},
{2.0/3.0, 1, 2.0/3.0, 1}};
UpdateRect URect16[] = {{0, 0.25, 0, 0.25},
{0, 0.25, 0.25, 0.5},
{0, 0.25, 0.5, 0.75},
{0, 0.25, 0.75, 1},
{0.25, 0.5, 0, 0.25},
{0.25, 0.5, 0.25, 0.5},
{0.25, 0.5, 0.5, 0.75},
{0.25, 0.5, 0.75, 1},
{0.5, 0.75, 0, 0.25},
{0.5, 0.75, 0.25, 0.5},
{0.5, 0.75, 0.5, 0.75},
{0.5, 0.75, 0.75, 1},
{0.75, 1, 0, 0.25},
{0.75, 1, 0.25, 0.5},
{0.75, 1, 0.5, 0.75},
{0.75, 1, 0.75, 1}};
//
// Local function declaration
//
BOOL GdiGetDevmodeForPagePvt(
IN HANDLE hSpoolHandle,
IN DWORD dwPageNumber,
OUT PDEVMODEW *ppCurrDM,
OUT PDEVMODEW *ppLastDM
);
BOOL BIsDevmodeOfLeastAcceptableSize(
IN PDEVMODE pdm) ;
BOOL
ValidNumberForNUp(
DWORD dwPages)
/*++
Function Description: Checks if the number of pages printed on a single side is Valid.
Parameters: dwPages - Number of pages printed on a single side
Return Values: TRUE if (dwPages = 1|2|4|6|9|16)
FALSE otherwise.
--*/
{
return ((dwPages == 1) || (dwPages == 2) || (dwPages == 4) ||
(dwPages == 6) || (dwPages == 9) || (dwPages == 16));
}
BOOL
GetPageCoordinatesForNUp(
HDC hPrinterDC,
RECT *rectDocument,
RECT *rectBorder,
DWORD dwTotalNumberOfPages,
UINT uCurrentPageNumber,
DWORD dwNupBorderFlags,
LPBOOL pbRotate
)
/*++
Function Description: GetPageCoordinatesForNUp computes the rectangle on the Page where the
EMF file is to be played. It also determines if the picture is to
rotated.
Parameters: hPrinterDC - Printer Device Context
*rectDocument - pointer to RECT where the coordinates to play the
page will be returned.
*rectBorder - pointer to RECT where the page borders are to drawn.
dwTotalNumberOfPages - Total number of pages on 1 side.
uCurrentPageNumber - 1 based page number on the side.
dwNupBorderFlags - flags to draw border along logical pages.
pbRotate - pointer to BOOL which indicates if the picture must be
rotated.
Return Values: NONE.
--*/
{
UpdateRect *URect;
LONG lXPrintPage,lYPrintPage,lXPhyPage,lYPhyPage,lXFrame,lYFrame,ltemp,ldX,ldY;
LONG lXNewPhyPage,lYNewPhyPage,lXOffset,lYOffset,lNumRowCol,lRowIndex,lColIndex;
double dXleft,dXright,dYtop,dYbottom;
LONG xResolution = GetDeviceCaps(hPrinterDC, LOGPIXELSX);
LONG yResolution = GetDeviceCaps(hPrinterDC, LOGPIXELSY);
// Get the 0-based array index for the current page
uCurrentPageNumber = uCurrentPageNumber - 1;
if (dwTotalNumberOfPages==1 || xResolution==yResolution)
{
xResolution = yResolution = 1;
}
rectDocument->top = rectDocument->bottom = lYPrintPage = (GetDeviceCaps(hPrinterDC, DESKTOPVERTRES)-1) * xResolution;
rectDocument->left = rectDocument->right = lXPrintPage = (GetDeviceCaps(hPrinterDC, DESKTOPHORZRES)-1) * yResolution;
lXPhyPage = GetDeviceCaps(hPrinterDC, PHYSICALWIDTH) * yResolution;
lYPhyPage = GetDeviceCaps(hPrinterDC, PHYSICALHEIGHT) * xResolution;
//
// Down in the code, we are dividing by these values, which can lead
// to divide-by-zero errors.
//
if ( 0 == xResolution ||
0 == yResolution ||
0 == lXPhyPage ||
0 == lYPhyPage )
{
return FALSE;
}
*pbRotate = FALSE;
// Select the array containing the update factors
switch (dwTotalNumberOfPages) {
case 1: rectDocument->top = rectDocument->left = 0;
rectDocument->right += 1;
rectDocument->bottom += 1;
return TRUE;
case 2: if (lXPrintPage > lYPrintPage) { // cut vertically
URect = URect22;
lXFrame = (LONG) (lXPrintPage / 2.0);
lYFrame = lYPrintPage;
} else { // cut horizontally
URect = URect21;
lYFrame = (LONG) (lYPrintPage / 2.0);
lXFrame = lXPrintPage;
}
break;
case 4: URect = URect4;
lXFrame = (LONG) (lXPrintPage / 2.0);
lYFrame = (LONG) (lYPrintPage / 2.0);
break;
case 6: if (lXPrintPage > lYPrintPage) { // cut vertically twice
URect = URect62;
lXFrame = (LONG) (lXPrintPage / 3.0);
lYFrame = (LONG) (lYPrintPage / 2.0);
} else { // cut horizontally twice
URect = URect61;
lYFrame = (LONG) (lYPrintPage / 3.0);
lXFrame = (LONG) (lXPrintPage / 2.0);
}
break;
case 9: URect = URect9;
lXFrame = (LONG) (lXPrintPage / 3.0);
lYFrame = (LONG) (lYPrintPage / 3.0);
break;
case 16: URect = URect16;
lXFrame = (LONG) (lXPrintPage / 4.0);
lYFrame = (LONG) (lYPrintPage / 4.0);
break;
default: // Should Not Occur.
return FALSE;
}
// Set the flag if the picture has to be rotated
*pbRotate = !((lXPhyPage >= lYPhyPage) && (lXFrame >= lYFrame)) &&
!((lXPhyPage < lYPhyPage) && (lXFrame < lYFrame));
// If the picture is to be rotated, modify the rectangle selected.
if ((dwTotalNumberOfPages == 2) || (dwTotalNumberOfPages == 6)) {
if (*pbRotate) {
switch (dwTotalNumberOfPages) {
case 2: if (lXPrintPage <= lYPrintPage) {
URect = URect21R;
} // URect22 = URect22R
break;
case 6: if (lXPrintPage <= lYPrintPage) {
URect = URect61R;
} else {
URect = URect62R;
}
break;
}
}
} else {
if (*pbRotate) {
// get the number of rows/columns. switch is faster than sqrt.
switch (dwTotalNumberOfPages) {
case 4: lNumRowCol = 2;
break;
case 9: lNumRowCol = 3;
break;
case 16: lNumRowCol = 4;
break;
}
lRowIndex = (LONG) (uCurrentPageNumber / lNumRowCol);
lColIndex = (LONG) (uCurrentPageNumber % lNumRowCol);
uCurrentPageNumber = (lNumRowCol - 1 - lColIndex) * lNumRowCol + lRowIndex;
}
}
// Update the Page Coordinates.
rectDocument->top = (LONG) (rectDocument->top * URect[uCurrentPageNumber].top);
rectDocument->bottom = (LONG) (rectDocument->bottom * URect[uCurrentPageNumber].bottom);
rectDocument->left = (LONG) (rectDocument->left * URect[uCurrentPageNumber].left);
rectDocument->right = (LONG) (rectDocument->right * URect[uCurrentPageNumber].right);
// If the page border has to drawn, return the corresponding coordinates in rectBorder.
if (dwNupBorderFlags == BORDER_PRINT) {
rectBorder->top = rectDocument->top/xResolution;
rectBorder->bottom = rectDocument->bottom/xResolution - 1;
rectBorder->left = rectDocument->left/yResolution;
rectBorder->right = rectDocument->right/yResolution - 1;
}
if (*pbRotate) {
ltemp = lXFrame; lXFrame = lYFrame; lYFrame = ltemp;
}
// Get the new size of the rectangle to keep the X/Y ratio constant.
if ( ((LONG) (lYFrame*((lXPhyPage*1.0)/lYPhyPage))) >= lXFrame) {
ldX = 0;
ldY = lYFrame - ((LONG) (lXFrame*((lYPhyPage*1.0)/lXPhyPage)));
} else {
ldY = 0;
ldX = lXFrame - ((LONG) (lYFrame*((lXPhyPage*1.0)/lYPhyPage)));
}
// Adjust the position of the rectangle.
if (*pbRotate) {
if (ldX) {
rectDocument->bottom -= (LONG) (ldX / 2.0);
rectDocument->top += (LONG) (ldX / 2.0);
} else {
rectDocument->right -= (LONG) (ldY / 2.0);
rectDocument->left += (LONG) (ldY / 2.0);
}
} else {
if (ldX) {
rectDocument->left += (LONG) (ldX / 2.0);
rectDocument->right -= (LONG) (ldX / 2.0);
} else {
rectDocument->top += (LONG) (ldY / 2.0);
rectDocument->bottom -= (LONG) (ldY / 2.0);
}
}
// Adjust to get the Printable Area on the rectangle
lXOffset = GetDeviceCaps(hPrinterDC, PHYSICALOFFSETX) * yResolution;
lYOffset = GetDeviceCaps(hPrinterDC, PHYSICALOFFSETY) * xResolution;
dXleft = ( lXOffset * 1.0) / lXPhyPage;
dYtop = ( lYOffset * 1.0) / lYPhyPage;
dXright = ((lXPhyPage - (lXOffset + lXPrintPage)) * 1.0) / lXPhyPage;
dYbottom = ((lYPhyPage - (lYOffset + lYPrintPage)) * 1.0) / lYPhyPage;
lXNewPhyPage = rectDocument->right - rectDocument->left;
lYNewPhyPage = rectDocument->bottom - rectDocument->top;
if (*pbRotate) {
ltemp = lXNewPhyPage; lXNewPhyPage = lYNewPhyPage; lYNewPhyPage = ltemp;
rectDocument->left += (LONG) (dYtop * lYNewPhyPage);
rectDocument->right -= (LONG) (dYbottom * lYNewPhyPage);
rectDocument->top += (LONG) (dXright * lXNewPhyPage);
rectDocument->bottom -= (LONG) (dXleft * lXNewPhyPage);
} else {
rectDocument->left += (LONG) (dXleft * lXNewPhyPage);
rectDocument->right -= (LONG) (dXright * lXNewPhyPage);
rectDocument->top += (LONG) (dYtop * lYNewPhyPage);
rectDocument->bottom -= (LONG) (dYbottom * lYNewPhyPage);
}
if (xResolution!=yResolution)
{
rectDocument->left = rectDocument->left / yResolution;
rectDocument->right = rectDocument->right / yResolution;
rectDocument->top = rectDocument->top / xResolution;
rectDocument->bottom = rectDocument->bottom / xResolution;
}
return TRUE;
}
BOOL
PlayEMFPage(
HANDLE hSpoolHandle,
HDC hPrinterDC,
HANDLE hEMF,
DWORD dwNumberOfPagesPerSide,
DWORD dwPageNumber,
DWORD dwPageIndex,
DWORD dwNupBorderFlags,
DWORD dwAngle)
/*++
Function Description: PlayEMFPage plays the EMF in the appropriate rectangle. It performs
the required scaling, rotation and translation.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
hEMF -- handle to the contents of the page in the spool file
dwNumberOfPagesPerSide -- number of pages to be printed per side
dwPageNumber -- page number in the document
dwPageIndex -- page number in the side. (1 based)
dwNupBorderFlags -- border printing options for nup
dwAngle -- angle for rotation (if neccesary)
Return Values: TRUE if successful
FALSE otherwise
--*/
{
BOOL bReturn = FALSE, bRotate;
RECT rectDocument, rectPrinter, rectBorder = {-1, -1, -1, -1};
RECT *prectClip = NULL;
XFORM TransXForm = {1, 0, 0, 1, 0, 0}, RotateXForm = {0, -1, 1, 0, 0, 0};
HPEN hPen;
HANDLE hFormEMF;
DWORD dwPageType,dwFormPage;
// Compute the rectangle for one page.
if ( FALSE == GetPageCoordinatesForNUp(hPrinterDC,
&rectDocument,
&rectBorder,
dwNumberOfPagesPerSide,
dwPageIndex,
dwNupBorderFlags,
&bRotate) )
{
goto CleanUp;
}
// If swap flag is set, reverse rotate flag
//
if (dwAngle & EMF_DEGREE_SWAP)
bRotate = !bRotate;
if (dwAngle & EMF_DEGREE_270) {
RotateXForm.eM12 = 1;
RotateXForm.eM21 = -1;
} // EMF_DEGREE_90 case is the initialization
if (bRotate) {
rectPrinter.top = 0;
rectPrinter.bottom = rectDocument.right - rectDocument.left;
rectPrinter.left = 0;
rectPrinter.right = rectDocument.bottom - rectDocument.top;
// Set the translation matrix
if (dwAngle & EMF_DEGREE_270) {
TransXForm.eDx = (float) rectDocument.right;
TransXForm.eDy = (float) rectDocument.top;
} else {
// EMF_DEGREE_90
TransXForm.eDx = (float) rectDocument.left;
TransXForm.eDy = (float) rectDocument.bottom;
}
// Set the transformation matrix
if (!SetWorldTransform(hPrinterDC, &RotateXForm) ||
!ModifyWorldTransform(hPrinterDC, &TransXForm, MWT_RIGHTMULTIPLY)) {
ODS(("Setting transformation matrix failed\n"));
goto CleanUp;
}
}
// Add clipping for Nup
if (dwNumberOfPagesPerSide != 1) {
prectClip = &rectDocument;
}
// Print the page.
if (bRotate) {
GdiPlayPageEMF(hSpoolHandle, hEMF, &rectPrinter, &rectBorder, prectClip);
} else {
GdiPlayPageEMF(hSpoolHandle, hEMF, &rectDocument, &rectBorder, prectClip);
}
bReturn = TRUE;
CleanUp:
if (!ModifyWorldTransform(hPrinterDC, NULL, MWT_IDENTITY)) {
ODS(("Setting Identity Transformation failed\n"));
bReturn = FALSE;
}
return bReturn;
}
BOOL
SetDrvCopies(
HDC hPrinterDC,
LPDEVMODEW pDevmode,
DWORD dwNumberOfCopies)
/*++
Function Description: SetDrvCopies sets the dmCopies field in pDevmode and resets
hPrinterDC with this devmode
Parameters: hPrinterDC -- handle to the printer device context
pDevmode -- pointer to devmode
dwNumberOfCopies -- value for dmCopies
Return Values: TRUE if successful
FALSE otherwise
--*/
{
BOOL bReturn;
DWORD dmFields;
if ((pDevmode->dmFields & DM_COPIES) &&
(pDevmode->dmCopies == (short) dwNumberOfCopies)) {
return TRUE;
}
// Save the old fields structure
dmFields = pDevmode->dmFields;
pDevmode->dmFields |= DM_COPIES;
pDevmode->dmCopies = (short) dwNumberOfCopies;
if (!ResetDC(hPrinterDC, pDevmode)) {
bReturn = FALSE;
} else {
bReturn = TRUE;
}
// Restore the fields structure
pDevmode->dmFields = dmFields;
if (!SetGraphicsMode(hPrinterDC,GM_ADVANCED)) {
ODS(("Setting graphics mode failed\n"));
bReturn = FALSE;
}
return bReturn;
}
BOOL
DifferentDevmodes(
LPDEVMODE pDevmode1,
LPDEVMODE pDevmode2
)
/*++
Function Description: Compares the devmodes for differences other than dmTTOption
Parameters: pDevmode1 - devmode 1
pDevmode2 - devmode 2
Return Values: TRUE if different ; FALSE otherwise
--*/
{
DWORD dwSize1, dwSize2, dwTTOffset, dwSpecOffset, dwLogOffset;
// Same pointers are the same devmode
if (pDevmode1 == pDevmode2) {
return FALSE;
}
// Check for Null devmodes
if (!pDevmode1 || !pDevmode2) {
return TRUE;
}
dwSize1 = pDevmode1->dmSize + pDevmode1->dmDriverExtra;
dwSize2 = pDevmode2->dmSize + pDevmode2->dmDriverExtra;
// Compare devmode sizes
if (dwSize1 != dwSize2) {
return TRUE;
}
dwTTOffset = FIELD_OFFSET(DEVMODE, dmTTOption);
dwSpecOffset = FIELD_OFFSET(DEVMODE, dmSpecVersion);
dwLogOffset = FIELD_OFFSET(DEVMODE, dmLogPixels);
if (wcscmp(pDevmode1->dmDeviceName,
pDevmode2->dmDeviceName)) {
// device names are different
return TRUE;
}
if (dwTTOffset < dwSpecOffset ||
dwSize1 < dwLogOffset) {
// incorrent devmode offsets
return TRUE;
}
if (memcmp((LPBYTE) pDevmode1 + dwSpecOffset,
(LPBYTE) pDevmode2 + dwSpecOffset,
dwTTOffset - dwSpecOffset)) {
// Front half is different
return TRUE;
}
// Ignore the dmTTOption setting.
if ((pDevmode1->dmCollate != pDevmode2->dmCollate) ||
wcscmp(pDevmode1->dmFormName, pDevmode2->dmFormName)) {
// form name or collate option is different
return TRUE;
}
if (memcmp((LPBYTE) pDevmode1 + dwLogOffset,
(LPBYTE) pDevmode2 + dwLogOffset,
dwSize1 - dwLogOffset)) {
// Back half is different
return TRUE;
}
return FALSE;
}
BOOL
ResetDCForNewDevmode(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwPageNumber,
BOOL bInsidePage,
DWORD dwOptimization,
LPBOOL pbNewDevmode,
LPDEVMODE pDevmode,
LPDEVMODE *pCurrentDevmode
)
/*++
Function Description: Determines if the devmode for the page is different from the
current devmode for the printer dc and resets the dc if necessary.
The parameters allow dmTTOption to be ignored in devmode comparison.
Parameters: hSpoolHandle - spool file handle
hPrinterDC - printer dc
dwPageNumber - page number before which we search for the devmode
bInsidePage - flag to ignore changes in TT options and call EndPage
before ResetDC
dwOptimization - optimization flags
pbNewDevmode - pointer to flag to indicate if ResetDC was called
pDevmode - devmode containing changed resolution settings
Return Values: TRUE if successful; FALSE otherwise
--*/
{
BOOL bReturn = FALSE;
LPDEVMODE pLastDM, pCurrDM;
// Initialize OUT parameters
*pbNewDevmode = FALSE;
// Get the devmode just before the page
if (!GdiGetDevmodeForPagePvt(hSpoolHandle,
dwPageNumber,
&pCurrDM,
&pLastDM)) {
ODS(("GdiGetDevmodeForPagePvt failed\n"));
return bReturn;
}
// Save pointer to current devmode
if (pCurrentDevmode)
*pCurrentDevmode = pCurrDM;
// Check if the devmodes are different
if (pLastDM != pCurrDM) {
// If the pointers are different the devmodes are always different
if (!bInsidePage ||
DifferentDevmodes(pLastDM, pCurrDM)) {
*pbNewDevmode = TRUE;
}
}
// Call ResetDC on the hPrinterDC if necessary
if (*pbNewDevmode) {
if (bInsidePage &&
!GdiEndPageEMF(hSpoolHandle, dwOptimization)) {
ODS(("EndPage failed\n"));
return bReturn;
}
if (pCurrDM) {
pCurrDM->dmPrintQuality = pDevmode->dmPrintQuality;
pCurrDM->dmYResolution = pDevmode->dmYResolution;
pCurrDM->dmCopies = pDevmode->dmCopies;
// @@BEGIN_DDKSPLIT
// GdiGetDevmodeForPagePvt ensures devmode is atleast big enough for dmYResolution
// So now we check for dmCollate (which comes after dmYResolution in DEVMODE).
// @@END_DDKSPLIT
if ( IS_DMSIZE_VALID ( pCurrDM, dmCollate ) )
{
if ( IS_DMSIZE_VALID ( pDevmode, dmCollate ) )
{
pCurrDM->dmCollate = pDevmode->dmCollate;
}
else
{
pCurrDM->dmCollate = DMCOLLATE_FALSE;
}
}
}
// Ignore the return values of ResetDC and SetGraphicsMode
GdiResetDCEMF(hSpoolHandle, pCurrDM);
SetGraphicsMode(hPrinterDC, GM_ADVANCED);
}
bReturn = TRUE;
return bReturn;
}
DWORD
PrintOneSideForwardEMF(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwNumberOfPagesPerSide,
DWORD dwDrvNumberOfPagesPerSide,
DWORD dwNupBorderFlags,
BOOL bDuplex,
DWORD dwOptimization,
DWORD dwPageNumber,
DWORD dwJobNumberOfCopies,
LPBOOL pbComplete,
LPDEVMODE pDevmode)
/*++
Function Description: PrintOneSideForwardEMF plays the next physical page in the same order
as the spool file.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
dwNumberOfPagesPerSide -- number of pages to be printed per side by the print processor
dwDrvNumberOfPagesPerSide -- number of pages the driver will print per side
dwNupBorderFlags -- border printing options for nup
bDuplex -- flag to indicate duplex printing
dwOptimization -- optimization flags
dwPageNumber -- pointer to the starting page number
pbComplete -- pointer to the flag to indicate completion
pDevmode -- devmode with resolution settings
Return Values: Last Page Number if successful
0 on job completion (pbReturn set to TRUE) and
on failure (pbReturn remains FALSE)
--*/
{
DWORD dwPageIndex, dwPageType;
DWORD dwReturn = 0;
LPDEVMODEW pCurrDM;
HANDLE hEMF = NULL;
DWORD dwSides;
BOOL bNewDevmode;
DWORD cPagesToPlay;
DWORD dwAngle;
INT dmOrientation = pDevmode->dmOrientation;
// set the number of sides on this page;
dwSides = bDuplex ? 2 : 1;
*pbComplete = FALSE;
for ( ; dwSides && !*pbComplete ; --dwSides) {
// loop for a single side
for (dwPageIndex = 1;
dwPageIndex <= dwNumberOfPagesPerSide;
++dwPageIndex, ++dwPageNumber) {
if (!(hEMF = GdiGetPageHandle(hSpoolHandle,
dwPageNumber,
&dwPageType))) {
if (GetLastError() == ERROR_NO_MORE_ITEMS) {
// End of the print job
*pbComplete = TRUE;
break;
}
ODS(("GdiGetPageHandle failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
dwAngle = EMF_DEGREE_90;
if (dwPageIndex == 1)
{
// Process new devmodes in the spool file that appear before this page
if (!ResetDCForNewDevmode(hSpoolHandle,
hPrinterDC,
dwPageNumber,
(dwPageIndex != 1),
dwOptimization,
&bNewDevmode,
pDevmode,
&pCurrDM)) {
goto CleanUp;
}
if (pCurrDM)
dmOrientation = pCurrDM->dmOrientation;
}
// in case of orientation switch we need to keep track of what
// we started with and what it is now
else if (dwNumberOfPagesPerSide > 1)
{
if (GdiGetDevmodeForPagePvt(hSpoolHandle,
dwPageNumber,
&pCurrDM,
NULL))
{
if (pCurrDM && pCurrDM->dmOrientation != dmOrientation)
{
dwAngle = EMF_DEGREE_SWAP | EMF_DEGREE_90;
}
}
}
// Call StartPage for each new page
if ((dwPageIndex == 1) &&
!GdiStartPageEMF(hSpoolHandle)) {
ODS(("StartPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
if (!PlayEMFPage(hSpoolHandle,
hPrinterDC,
hEMF,
dwNumberOfPagesPerSide,
dwPageNumber,
dwPageIndex,
dwNupBorderFlags,
dwAngle)) {
ODS(("PlayEMFPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
}
//
// Explaination of the scinario set for the conditions on
// dwPageIndex1 , pbComplete and bDuplex.
// N.B. we are naming them cond.1 and cond.2
// dwPageIndex!=1 pbComplete bDuplex Condition
// 0 0 0 None
// 0 0 1 None
// 0 1 0 None
// 0 1 1 Cond2 on Second Side i.e. dwsides==1
// 1 0 0 Cond1
// 1 0 1 Cond1
// 1 1 0 Cond1
// 1 1 1 Cond1 & Cond2 on First Side i.e. dwsides==2
//
// cond.1
if (dwPageIndex != 1) {
// Call EndPage if we played any pages
if (!GdiEndPageEMF(hSpoolHandle, dwOptimization)) {
ODS(("EndPage failed\n"));
*pbComplete = FALSE;
goto CleanUp;
}
}
// cond.2
// play empty page on the back of duplex
if (*pbComplete && bDuplex && dwDrvNumberOfPagesPerSide==1) {
ODS(("PCL or PS with no N-up\n"));
//
// Checking dwsides against 2 or 1.
// depends on whether it is n-up or not.
//
if (((dwPageIndex!=1)?(dwSides==2):(dwSides==1))) {
if (!GdiStartPageEMF(hSpoolHandle) ||
!GdiEndPageEMF(hSpoolHandle, dwOptimization)) {
ODS(("EndPage failed\n"));
*pbComplete = FALSE;
goto CleanUp;
}
}
}
}
if (*pbComplete &&
dwNumberOfPagesPerSide==1 &&
dwDrvNumberOfPagesPerSide!=1 &&
dwJobNumberOfCopies!=1)
{
cPagesToPlay = dwDrvNumberOfPagesPerSide * (bDuplex ? 2 : 1);
if ((dwPageNumber-1) % cPagesToPlay)
{
//
// Number of pages played on last physical page
//
cPagesToPlay = cPagesToPlay - ((dwPageNumber-1) % cPagesToPlay);
ODS(("\nPS with N-up!\nMust fill in %u pages\n", cPagesToPlay));
for (;cPagesToPlay;cPagesToPlay--)
{
if (!GdiStartPageEMF(hSpoolHandle) || !GdiEndPageEMF(hSpoolHandle, dwOptimization))
{
ODS(("EndPage failed\n"));
goto CleanUp;
}
}
}
}
if (!(*pbComplete)) dwReturn = dwPageNumber;
CleanUp:
return dwReturn;
}
BOOL
PrintForwardEMF(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwNumberOfPagesPerSide,
DWORD dwDrvNumberOfPagesPerSide,
DWORD dwNupBorderFlags,
DWORD dwJobNumberOfCopies,
DWORD dwDrvNumberOfCopies,
BOOL bCollate,
BOOL bDuplex,
DWORD dwOptimization,
LPDEVMODEW pDevmode,
PPRINTPROCESSORDATA pData)
/*++
Function Description: PrintForwardEMF plays the EMF files in the order in which they
were spooled.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
dwNumberOfPagesPerSide -- number of pages to be printed per side by the print processor
dwDrvNumberOfPagesPerSide -- number of pages the driver will print per side
dwNupBorderFlags -- border printing options for nup
dwJobNumberOfCopies -- number of copies of the job to be printed
dwDrvNumberOfCopies -- number of copies that the driver can print
bCollate -- flag for collating the copies
bDuplex -- flag for duplex printing
dwOptimization -- optimization flags
pDevmode -- pointer to devmode for changing the copy count
pData -- needed for status and the handle of the event: pause, resume etc.
Return Values: TRUE if successful
FALSE otherwise
--*/
{
DWORD dwLastPageNumber = 1,dwPageNumber,dwPageIndex,dwRemainingCopies;
BOOL bReturn = FALSE;
// Keep printing as long as the spool file contains EMF handles.
while (dwLastPageNumber) {
//
// If the print processor is paused, wait for it to be resumed
//
if (pData->fsStatus & PRINTPROCESSOR_PAUSED) {
WaitForSingleObject(pData->semPaused, INFINITE);
}
dwPageNumber = dwLastPageNumber;
if (bCollate) {
dwLastPageNumber = PrintOneSideForwardEMF(hSpoolHandle,
hPrinterDC,
dwNumberOfPagesPerSide,
dwDrvNumberOfPagesPerSide,
dwNupBorderFlags,
bDuplex,
dwOptimization,
dwPageNumber,
dwJobNumberOfCopies,
&bReturn,
pDevmode);
} else {
dwRemainingCopies = dwJobNumberOfCopies;
while (dwRemainingCopies) {
if (dwRemainingCopies <= dwDrvNumberOfCopies) {
SetDrvCopies(hPrinterDC, pDevmode, dwRemainingCopies);
dwRemainingCopies = 0;
} else {
SetDrvCopies(hPrinterDC, pDevmode, dwDrvNumberOfCopies);
dwRemainingCopies -= dwDrvNumberOfCopies;
}
if (!(dwLastPageNumber = PrintOneSideForwardEMF(hSpoolHandle,
hPrinterDC,
dwNumberOfPagesPerSide,
dwDrvNumberOfPagesPerSide,
dwNupBorderFlags,
bDuplex,
dwOptimization,
dwPageNumber,
dwJobNumberOfCopies,
&bReturn,
pDevmode)) &&
!bReturn) {
goto CleanUp;
}
}
}
}
CleanUp:
return bReturn;
}
BOOL
PrintOneSideReverseForDriverEMF(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwDrvNumberOfPagesPerSide,
DWORD dwTotalNumberOfPages,
DWORD dwNupBorderFlags,
BOOL bDuplex,
DWORD dwOptimization,
DWORD dwPageNumber,
LPDEVMODE pDevmode)
/*++
Function Description: PrintOneSideReverseForDriverEMF plays the EMF pages on the next
physical page, in the reverse order for the driver which does the
Nup transformations.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
dwDrvNumberOfPagesPerSide -- number of pages the driver will print per side
dwTotalNumberOfPages -- total number of pages in the document
dwNupBorderFlags -- border printing options for nup
bDuplex -- flag to indicate duplex printing
dwOptimization -- optimization flags
dwPageNumber -- page number to start the side
pDevmode -- devmode with resolution settings
Return Values: TRUE if successful
FALSE otherwise
--*/
{
DWORD dwPageIndex, dwPageType, dwSides;
BOOL bReturn = FALSE, bNewDevmode,BeSmart;
LPDEVMODEW pCurrDM;
HANDLE hEMF = NULL;
DWORD dwLimit;
dwSides = bDuplex ? 2 : 1;
//
// If the document will fit on one phisical page, then this variable will prevent
// the printer from playing extra pages just to fill in one phisical page
// The exception is when the pages fit on a single phisical page, but they must
// be collated. Then because of design, the printer will also draw borders for the
// empty pages which are played so that the page gets ejected.
//
BeSmart = (dwTotalNumberOfPages<=dwDrvNumberOfPagesPerSide) &&
IS_DMSIZE_VALID(pDevmode, dmCollate) &&
(pDevmode->dmCollate != DMCOLLATE_TRUE);
for (; dwSides; --dwSides) {
// This loop may play some empty pages in the last side, since the
// driver is doing nup and it does not keep count of the page numbers
//
dwPageIndex=BeSmart?dwPageNumber:1;
dwLimit =BeSmart?dwTotalNumberOfPages:dwDrvNumberOfPagesPerSide;
for (;dwPageIndex<=dwLimit; ++dwPageIndex,++dwPageNumber) {
if (BeSmart || dwPageNumber <= dwTotalNumberOfPages) {
if (!(hEMF = GdiGetPageHandle(hSpoolHandle,
dwPageNumber,
&dwPageType))) {
ODS(("GdiGetPageHandle failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
// Process new devmodes in the spoolfile
if (!ResetDCForNewDevmode(hSpoolHandle,
hPrinterDC,
dwPageNumber,
FALSE,
dwOptimization,
&bNewDevmode,
pDevmode,
NULL)) {
}
}
if (!GdiStartPageEMF(hSpoolHandle)) {
ODS(("StartPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
if (BeSmart || dwPageNumber <= dwTotalNumberOfPages) {
if (!PlayEMFPage(hSpoolHandle,
hPrinterDC,
hEMF,
1,
dwPageNumber,
1,
dwNupBorderFlags,
EMF_DEGREE_90)) {
ODS(("PlayEMFPage failed\n"));
goto CleanUp;
}
}
if (!GdiEndPageEMF(hSpoolHandle, dwOptimization)) {
ODS(("EndPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
}
}
bReturn = TRUE;
CleanUp:
return bReturn;
}
BOOL
PrintReverseForDriverEMF(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwDrvNumberOfPagesPerSide,
DWORD dwTotalNumberOfPages,
DWORD dwNupBorderFlags,
DWORD dwJobNumberOfCopies,
DWORD dwDrvNumberOfCopies,
BOOL bCollate,
BOOL bDuplex,
BOOL bOdd,
DWORD dwOptimization,
LPDEVMODEW pDevmode,
PPAGE_NUMBER pHead,
PPRINTPROCESSORDATA pData)
/*++
Function Description: PrintReverseForDriverEMF plays the EMF pages in the reverse order
for the driver which does the Nup transformations.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
dwDrvNumberOfPagesPerSide -- number of pages the driver will print per side
dwTotalNumberOfPages -- total number of pages in the document
dwNupBorderFlags -- border printing options for nup
dwJobNumberOfCopies -- number of copies of the job to be printed
dwDrvNumberOfCopies -- number of copies that the driver can print
bCollate -- flag for collating the copies
bDuplex -- flag to indicate duplex printing
bOdd -- flag to indicate odd number of sides to print
dwOptimization -- optimization flags
pDevmode -- pointer to devmode for changing the copy count
pHead -- pointer to a linked list containing the starting
page numbers for each of the sides
pData -- needed for status and the handle of the event: pause, resume etc.
Return Values: TRUE if successful
FALSE otherwise
--*/
{
DWORD dwPageIndex,dwPageNumber,dwRemainingCopies;
BOOL bReturn = FALSE;
// select the correct page for duplex printing
if (bDuplex && !bOdd) {
if (pHead) {
pHead = pHead->pNext;
} else {
bReturn = TRUE;
goto CleanUp;
}
}
// play the sides in reverse order
while (pHead) {
//
// If the print processor is paused, wait for it to be resumed
//
if (pData->fsStatus & PRINTPROCESSOR_PAUSED) {
WaitForSingleObject(pData->semPaused, INFINITE);
}
// set the page number
dwPageNumber = pHead->dwPageNumber;
if (bCollate) {
if (!PrintOneSideReverseForDriverEMF(hSpoolHandle,
hPrinterDC,
dwDrvNumberOfPagesPerSide,
dwTotalNumberOfPages,
dwNupBorderFlags,
bDuplex,
dwOptimization,
dwPageNumber,
pDevmode)) {
goto CleanUp;
}
} else {
dwRemainingCopies = dwJobNumberOfCopies;
while (dwRemainingCopies) {
if (dwRemainingCopies <= dwDrvNumberOfCopies) {
SetDrvCopies(hPrinterDC, pDevmode, dwRemainingCopies);
dwRemainingCopies = 0;
} else {
SetDrvCopies(hPrinterDC, pDevmode, dwDrvNumberOfCopies);
dwRemainingCopies -= dwDrvNumberOfCopies;
}
if (!PrintOneSideReverseForDriverEMF(hSpoolHandle,
hPrinterDC,
dwDrvNumberOfPagesPerSide,
dwTotalNumberOfPages,
dwNupBorderFlags,
bDuplex,
dwOptimization,
dwPageNumber,
pDevmode)) {
goto CleanUp;
}
}
}
pHead = pHead->pNext;
// go to the next page for duplex printing
if (bDuplex && pHead) {
pHead = pHead->pNext;
}
}
bReturn = TRUE;
CleanUp:
return bReturn;
}
BOOL
PrintOneSideReverseEMF(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwNumberOfPagesPerSide,
DWORD dwNupBorderFlags,
BOOL bDuplex,
DWORD dwOptimization,
DWORD dwStartPage1,
DWORD dwEndPage1,
DWORD dwStartPage2,
DWORD dwEndPage2,
LPDEVMODE pDevmode)
/*++
Function Description: PrintOneSideReverseEMF plays the EMF pages for the next physical page.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
dwNumberOfPagesPerSide -- number of pages to be printed per side by the print
processor
dwNupBorderFlags -- border printing options for nup
bDuplex -- flag to indicate duplex printing
dwOptimization -- optimization flags
dwStartPage1 -- page number of the first EMF page on 1st side
dwEndPage1 -- page number of the last EMF page on 1st side
dwStartPage2 -- page number of the first EMF page on 2nd side
dwEndPage2 -- page number of the last EMF page on 2nd side
pDevmode -- devmode with resolution settings
Return Values: TRUE if successful
FALSE otherwise
--*/
{
DWORD dwPageNumber, dwPageIndex, dwPageType;
BOOL bReturn = FALSE, bNewDevmode;
LPDEVMODEW pCurrDM;
HANDLE hEMF = NULL;
DWORD dwEndPage, dwStartPage, dwSides, dwAngle;
INT dmOrientation = pDevmode->dmOrientation;
for (dwSides = bDuplex ? 2 : 1;
dwSides;
--dwSides) {
if (bDuplex && (dwSides == 1)) {
dwStartPage = dwStartPage2;
dwEndPage = dwEndPage2;
} else {
dwStartPage = dwStartPage1;
dwEndPage = dwEndPage1;
}
for (dwPageNumber = dwStartPage, dwPageIndex = 1;
dwPageNumber <= dwEndPage;
++dwPageNumber, ++dwPageIndex) {
if (!(hEMF = GdiGetPageHandle(hSpoolHandle,
dwPageNumber,
&dwPageType))) {
ODS(("GdiGetPageHandle failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
dwAngle = EMF_DEGREE_90;
if (dwPageIndex == 1) {
// Process devmodes in the spool file and call StartPage
if (!ResetDCForNewDevmode(hSpoolHandle,
hPrinterDC,
dwPageNumber,
FALSE,
dwOptimization,
&bNewDevmode,
pDevmode,
&pCurrDM) ||
!GdiStartPageEMF(hSpoolHandle)) {
goto CleanUp;
}
if (pCurrDM)
dmOrientation = pCurrDM->dmOrientation;
}
// in case of orientation switch we need to keep track of what
// we started with and what it is now
else if (dwNumberOfPagesPerSide > 1)
{
if (GdiGetDevmodeForPagePvt(hSpoolHandle,
dwPageNumber,
&pCurrDM,
NULL))
{
if (pCurrDM && pCurrDM->dmOrientation != dmOrientation)
{
dwAngle = EMF_DEGREE_SWAP | EMF_DEGREE_90;
}
}
}
if (!PlayEMFPage(hSpoolHandle,
hPrinterDC,
hEMF,
dwNumberOfPagesPerSide,
dwPageNumber,
dwPageIndex,
dwNupBorderFlags,
dwAngle)) {
ODS(("PlayEMFPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
}
if ((dwPageIndex == 1) && !GdiStartPageEMF(hSpoolHandle)) {
ODS(("StartPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
if (!GdiEndPageEMF(hSpoolHandle, dwOptimization)) {
ODS(("EndPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
}
bReturn = TRUE;
CleanUp:
return bReturn;
}
BOOL
PrintReverseEMF(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwTotalNumberOfPages,
DWORD dwNumberOfPagesPerSide,
DWORD dwNupBorderFlags,
DWORD dwJobNumberOfCopies,
DWORD dwDrvNumberOfCopies,
BOOL bCollate,
BOOL bDuplex,
BOOL bOdd,
DWORD dwOptimization,
LPDEVMODEW pDevmode,
PPAGE_NUMBER pHead,
PPRINTPROCESSORDATA pData)
/*++
Function Description: PrintReverseEMF plays the EMF pages in the reverse order and also
performs nup transformations.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
dwTotalNumberOfPages -- number of pages in the document
dwNumberOfPagesPerSide -- number of pages to be printed per side by the print
processor
dwNupBorderFlags -- border printing options for nup
dwJobNumberOfCopies -- number of copies of the job to be printed
dwDrvNumberOfCopies -- number of copies that the driver can print
bCollate -- flag for collating the copies
bDuplex -- flag to indicate duplex printing
bOdd -- flag to indicate odd number of sides to print
dwOptimization -- optimization flags
pDevmode -- pointer to devmode for changing the copy count
pHead -- pointer to a linked list containing the starting
page numbers for each of the sides
pData -- needed for status and the handle of the event: pause, resume etc.
Return Values: TRUE if successful
FALSE otherwise
--*/
{
DWORD dwPageNumber,dwPageIndex,dwRemainingCopies;
DWORD dwStartPage1,dwStartPage2,dwEndPage1,dwEndPage2;
BOOL bReturn = FALSE;
if (!pHead) {
bReturn = TRUE;
goto CleanUp;
}
// set the start and end page numbers for duplex and regular printing
if (bDuplex) {
if (bOdd) {
dwStartPage1 = pHead->dwPageNumber;
dwEndPage1 = dwTotalNumberOfPages;
dwStartPage2 = dwTotalNumberOfPages+1;
dwEndPage2 = 0;
} else {
dwStartPage2 = pHead->dwPageNumber;
dwEndPage2 = dwTotalNumberOfPages;
if (pHead = pHead->pNext) {
dwStartPage1 = pHead->dwPageNumber;
dwEndPage1 = dwStartPage2 - 1;
}
}
} else {
dwStartPage1 = pHead->dwPageNumber;
dwEndPage1 = dwTotalNumberOfPages;
dwStartPage2 = 0;
dwEndPage2 = 0;
}
while (pHead) {
//
// If the print processor is paused, wait for it to be resumed
//
if (pData->fsStatus & PRINTPROCESSOR_PAUSED) {
WaitForSingleObject(pData->semPaused, INFINITE);
}
if (bCollate) {
if (!PrintOneSideReverseEMF(hSpoolHandle,
hPrinterDC,
dwNumberOfPagesPerSide,
dwNupBorderFlags,
bDuplex,
dwOptimization,
dwStartPage1,
dwEndPage1,
dwStartPage2,
dwEndPage2,
pDevmode)) {
goto CleanUp;
}
} else {
dwRemainingCopies = dwJobNumberOfCopies;
while (dwRemainingCopies) {
if (dwRemainingCopies <= dwDrvNumberOfCopies) {
SetDrvCopies(hPrinterDC, pDevmode, dwRemainingCopies);
dwRemainingCopies = 0;
} else {
SetDrvCopies(hPrinterDC, pDevmode, dwDrvNumberOfCopies);
dwRemainingCopies -= dwDrvNumberOfCopies;
}
if (!PrintOneSideReverseEMF(hSpoolHandle,
hPrinterDC,
dwNumberOfPagesPerSide,
dwNupBorderFlags,
bDuplex,
dwOptimization,
dwStartPage1,
dwEndPage1,
dwStartPage2,
dwEndPage2,
pDevmode)) {
goto CleanUp;
}
}
}
if (bDuplex) {
if (pHead->pNext && pHead->pNext->pNext) {
dwEndPage2 = pHead->dwPageNumber - 1;
pHead = pHead->pNext;
dwStartPage2 = pHead->dwPageNumber;
dwEndPage1 = dwStartPage2 - 1;
pHead = pHead->pNext;
dwStartPage1 = pHead->dwPageNumber;
} else {
break;
}
} else {
pHead = pHead->pNext;
if (pHead) {
dwEndPage1 = dwStartPage1 - 1;
dwStartPage1 = pHead->dwPageNumber;
}
}
}
bReturn = TRUE;
CleanUp:
return bReturn;
}
BOOL
PrintOneSideBookletEMF(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwNumberOfPagesPerSide,
DWORD dwNupBorderFlags,
DWORD dwTotalNumberOfPages,
DWORD dwTotalPrintPages,
DWORD dwStartPage,
BOOL bReverseOrderPrinting,
DWORD dwOptimization,
DWORD dwDuplexMode,
LPDEVMODE pDevmode)
/*++
Function Description: PrintOneSideBookletEMF prints one page of the booklet job.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
dwNumberOfPagesPerSide -- number of pages to be printed per side by the print
processor
dwNupBorderFlags -- border printing options for nup
dwTotalNumberOfPages -- number of pages in the document
dwTotalPrintPages -- number of pages to printed (multiple of 4)
dwStartPage -- number of the starting page for the side
bReverseOrderPrinting -- flag for reverse order printing
dwOptimization -- optimization flags
dwDuplexMode -- duplex printing mode (none|horz|vert)
pDevmode -- devmode with resolution settings
Return Values: TRUE if successful
FALSE otherwise
--*/
{
DWORD dwPageArray[4];
DWORD dwPagesPrinted = 0, dwPageIndex, dwAngle, dwPageType, dwLastPage;
HANDLE hEMF = NULL;
LPDEVMODEW pCurrDM;
BOOL bReturn = FALSE ,bNewDevmode;
INT dmOrientation;
// set the order of the pages
if (bReverseOrderPrinting) {
dwPageArray[0] = dwStartPage + 1;
dwPageArray[1] = dwTotalPrintPages - dwStartPage;
if (dwDuplexMode == EMF_DUP_VERT) {
dwPageArray[2] = dwStartPage;
dwPageArray[3] = dwPageArray[1] + 1;
} else { // EMF_DUP_HORZ
dwPageArray[3] = dwStartPage;
dwPageArray[2] = dwPageArray[1] + 1;
}
} else {
dwPageArray[1] = dwStartPage;
dwPageArray[0] = dwTotalPrintPages - dwStartPage + 1;
if (dwDuplexMode == EMF_DUP_VERT) {
dwPageArray[2] = dwPageArray[0] - 1;
dwPageArray[3] = dwPageArray[1] + 1;
} else { // EMF_DUP_HORZ
dwPageArray[2] = dwPageArray[1] + 1;
dwPageArray[3] = dwPageArray[0] - 1;
}
}
// Set page number for ResetDC
dwLastPage = (dwTotalNumberOfPages < dwPageArray[0]) ? dwTotalNumberOfPages
: dwPageArray[0];
// Process devmodes in the spool file
if (!ResetDCForNewDevmode(hSpoolHandle,
hPrinterDC,
dwLastPage,
FALSE,
dwOptimization,
&bNewDevmode,
pDevmode,
&pCurrDM)) {
goto CleanUp;
}
if (pCurrDM)
dmOrientation = pCurrDM->dmOrientation;
else
dmOrientation = pDevmode->dmOrientation;
while (dwPagesPrinted < 4) {
for (dwPageIndex = 1;
dwPageIndex <= dwNumberOfPagesPerSide;
++dwPageIndex, ++dwPagesPrinted) {
if (dwPageArray[dwPagesPrinted] <= dwTotalNumberOfPages) {
if (!(hEMF = GdiGetPageHandle(hSpoolHandle,
dwPageArray[dwPagesPrinted],
&dwPageType))) {
ODS(("GdiGetPageHandle failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
}
if (dwPageIndex == 1) {
if (!GdiStartPageEMF(hSpoolHandle)) {
ODS(("StartPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
}
if (dwPageArray[dwPagesPrinted] <= dwTotalNumberOfPages) {
// in case of orientation switch we need to keep track of what
// we started with and what it is now
dwAngle = 0;
if (GdiGetDevmodeForPagePvt(hSpoolHandle,
dwPageArray[dwPagesPrinted],
&pCurrDM,
NULL))
{
if (pCurrDM && pCurrDM->dmOrientation != dmOrientation)
dwAngle |= EMF_DEGREE_SWAP;
}
if ((dwDuplexMode == EMF_DUP_VERT) &&
(dwPagesPrinted > 1)) {
dwAngle |= EMF_DEGREE_270;
} else { // EMF_DUP_HORZ or 1st side
dwAngle |= EMF_DEGREE_90;
}
if (!PlayEMFPage(hSpoolHandle,
hPrinterDC,
hEMF,
dwNumberOfPagesPerSide,
dwPageArray[dwPagesPrinted],
dwPageIndex,
dwNupBorderFlags,
dwAngle)) {
ODS(("PlayEMFPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
}
if (dwPageIndex == dwNumberOfPagesPerSide) {
if (!GdiEndPageEMF(hSpoolHandle, dwOptimization)) {
ODS(("EndPage failed\nPrinter %ws\n", pDevmode->dmDeviceName));
goto CleanUp;
}
}
}
}
bReturn = TRUE;
CleanUp:
return bReturn;
}
BOOL
PrintBookletEMF(
HANDLE hSpoolHandle,
HDC hPrinterDC,
DWORD dwNumberOfPagesPerSide,
DWORD dwTotalNumberOfPages,
DWORD dwNupBorderFlags,
DWORD dwJobNumberOfCopies,
DWORD dwDrvNumberOfCopies,
BOOL bReverseOrderPrinting,
BOOL bCollate,
DWORD dwOptimization,
DWORD dwDuplexMode,
LPDEVMODEW pDevmode,
PPRINTPROCESSORDATA pData)
/*++
Function Description: PrintBookletEMF prints the job in 2-up in booklet form.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
dwNumberOfPagesPerSide -- number of pages to be printed per side by the print
processor
dwTotalNumberOfPages -- number of pages in the document
dwNupBorderFlags -- border printing options for nup
dwJobNumberOfCopies -- number of copies of the job to be printed
dwDrvNumberOfCopies -- number of copies that the driver can print
bReverseOrderPrinting -- flag for reverse order printing
bCollate -- flag for collating the copies
dwOptimization -- optimization flags
dwDuplexMode -- duplex printing mode (none|horz|vert)
pDevmode -- pointer to devmode for changing the copy count
pData -- needed for status and the handle of the event: pause, resume etc.
Return Values: TRUE if successful
FALSE otherwise
--*/
{
BOOL bReturn = FALSE;
DWORD dwTotalPrintPages, dwNumberOfPhyPages, dwRemainingCopies, dwIndex;
// Get closest multiple of 4 greater than dwTotalNumberOfPages
dwTotalPrintPages = dwTotalNumberOfPages - (dwTotalNumberOfPages % 4);
if (dwTotalPrintPages != dwTotalNumberOfPages) {
dwTotalPrintPages += 4;
}
dwNumberOfPhyPages = (DWORD) dwTotalPrintPages / 4;
for (dwIndex = 0; dwIndex < dwNumberOfPhyPages; ++dwIndex) {
//
// If the print processor is paused, wait for it to be resumed
//
if (pData->fsStatus & PRINTPROCESSOR_PAUSED) {
WaitForSingleObject(pData->semPaused, INFINITE);
}
if (bCollate) {
if (!PrintOneSideBookletEMF(hSpoolHandle,
hPrinterDC,
dwNumberOfPagesPerSide,
dwNupBorderFlags,
dwTotalNumberOfPages,
dwTotalPrintPages,
dwIndex * 2 + 1,
bReverseOrderPrinting,
dwOptimization,
dwDuplexMode,
pDevmode)) {
goto CleanUp;
}
} else {
dwRemainingCopies = dwJobNumberOfCopies;
while (dwRemainingCopies) {
if (dwRemainingCopies <= dwDrvNumberOfCopies) {
SetDrvCopies(hPrinterDC, pDevmode, dwRemainingCopies);
dwRemainingCopies = 0;
} else {
SetDrvCopies(hPrinterDC, pDevmode, dwDrvNumberOfCopies);
dwRemainingCopies -= dwDrvNumberOfCopies;
}
if (!PrintOneSideBookletEMF(hSpoolHandle,
hPrinterDC,
dwNumberOfPagesPerSide,
dwNupBorderFlags,
dwTotalNumberOfPages,
dwTotalPrintPages,
dwIndex * 2 + 1,
bReverseOrderPrinting,
dwOptimization,
dwDuplexMode,
pDevmode)) {
goto CleanUp;
}
}
}
}
bReturn = TRUE;
CleanUp:
return bReturn;
}
BOOL
PrintEMFSingleCopy(
HANDLE hSpoolHandle,
HDC hPrinterDC,
BOOL bReverseOrderPrinting,
DWORD dwDrvNumberOfPagesPerSide,
DWORD dwNumberOfPagesPerSide,
DWORD dwTotalNumberOfPages,
DWORD dwNupBorderFlags,
DWORD dwJobNumberOfCopies,
DWORD dwDrvNumberOfCopies,
BOOL bCollate,
BOOL bOdd,
BOOL bBookletPrint,
DWORD dwOptimization,
DWORD dwDuplexMode,
LPDEVMODEW pDevmode,
PPAGE_NUMBER pHead,
PPRINTPROCESSORDATA pData)
/*++
Function Description: PrintEMFSingleCopy plays one copy of the job on hPrinterDC.
Parameters: hSpoolHandle -- handle the spool file handle
hPrinterDC -- handle to the printer device context
bReverseOrderPrinting -- flag for reverse order printing
dwDrvNumberOfPagesPerSide -- number of pages the driver will print per side
dwNumberOfPagesPerSide -- number of pages to be printed per side by the print
processor
dwTotalNumberOfPages -- number of pages in the document
dwNupBorderFlags -- border printing options for nup
dwJobNumberOfCopies -- number of copies of the job to be printed
dwDrvNumberOfCopies -- number of copies that the driver can print
bCollate -- flag for collating the copies
bOdd -- flag to indicate odd number of sides to print
bBookletPrint -- flag for booklet printing
dwOptimization -- optimization flags
dwDuplexMode -- duplex printing mode (none|horz|vert)
pDevmode -- pointer to devmode for changing the copy count
pHead -- pointer to a linked list containing the starting
page numbers for each of the sides
pData -- needed for status and the handle of the event: pause, resume etc.
Return Values: TRUE if successful
FALSE otherwise
--*/
{
BOOL bDuplex = (dwDuplexMode != EMF_DUP_NONE);
if (bBookletPrint) {
// Booklet Printing
return PrintBookletEMF(hSpoolHandle,
hPrinterDC,
dwNumberOfPagesPerSide,
dwTotalNumberOfPages,
dwNupBorderFlags,
dwJobNumberOfCopies,
dwDrvNumberOfCopies,
bReverseOrderPrinting,
bCollate,
dwOptimization,
dwDuplexMode,
pDevmode,
pData);
}
if (bReverseOrderPrinting) {
if (dwDrvNumberOfPagesPerSide != 1 || dwNumberOfPagesPerSide == 1) {
// @@BEGIN_DDKSPLIT
// Reverse printing while driver does nup / no nup required
// @@END_DDKSPLIT
return PrintReverseForDriverEMF(hSpoolHandle,
hPrinterDC,
dwDrvNumberOfPagesPerSide,
dwTotalNumberOfPages,
dwNupBorderFlags,
dwJobNumberOfCopies,
dwDrvNumberOfCopies,
bCollate,
bDuplex,
bOdd,
dwOptimization,
pDevmode,
pHead,
pData);
} else {
// Reverse printing and nup
return PrintReverseEMF(hSpoolHandle,
hPrinterDC,
dwTotalNumberOfPages,
dwNumberOfPagesPerSide,
dwNupBorderFlags,
dwJobNumberOfCopies,
dwDrvNumberOfCopies,
bCollate,
bDuplex,
bOdd,
dwOptimization,
pDevmode,
pHead,
pData);
}
} else {
// Normal printing
return PrintForwardEMF(hSpoolHandle,
hPrinterDC,
dwNumberOfPagesPerSide,
dwDrvNumberOfPagesPerSide,
dwNupBorderFlags,
dwJobNumberOfCopies,
dwDrvNumberOfCopies,
bCollate,
bDuplex,
dwOptimization,
pDevmode,
pData);
}
}
BOOL
GetStartPageList(
HANDLE hSpoolHandle,
PPAGE_NUMBER *pHead,
DWORD dwTotalNumberOfPages,
DWORD dwNumberOfPagesPerSide,
BOOL bCheckForDevmode,
LPBOOL pbOdd)
/*++
Function Description: GetStartPageList generates a list of the page numbers which
should appear on the start of each side of the job. This takes
into consideration the ResetDC calls that may appear before the
end of the page. The list generated by GetStartPageList is used
to play the job in reverse order.
Parameters: hSpoolHandle -- handle the spool file handle
pHead -- pointer to a pointer to a linked list containing the
starting page numbers for each of the sides
dwTotalNumberOfPages -- number of pages in the document
dwNumberOfPagesPerSide -- number of pages to be printed per side by the print
processor
pbOdd -- pointer to flag indicating odd number of pages to
print
Return Values: TRUE if successful
FALSE otherwise
--*/
{
DWORD dwPageIndex,dwPageNumber=1,dwPageType;
LPDEVMODEW pCurrDM, pLastDM;
PPAGE_NUMBER pTemp=NULL;
BOOL bReturn = FALSE;
BOOL bCheckDevmode;
bCheckDevmode = bCheckForDevmode && (dwNumberOfPagesPerSide != 1);
while (dwPageNumber <= dwTotalNumberOfPages) {
for (dwPageIndex = 1;
(dwPageIndex <= dwNumberOfPagesPerSide) && (dwPageNumber <= dwTotalNumberOfPages);
++dwPageIndex, ++dwPageNumber) {
if (bCheckDevmode) {
// Check if the devmode has changed requiring a new page
if (!GdiGetDevmodeForPagePvt(hSpoolHandle, dwPageNumber,
&pCurrDM, NULL)) {
ODS(("Get devmodes failed\n"));
goto CleanUp;
}
if (dwPageIndex == 1) {
// Save the Devmode for the first page on a side
pLastDM = pCurrDM;
} else {
// If the Devmode changes in a side, start a new page
if (DifferentDevmodes(pCurrDM, pLastDM)) {
dwPageIndex = 1;
pLastDM = pCurrDM;
}
}
}
// Create a node for the start of a side
if (dwPageIndex == 1) {
if (!(pTemp = AllocSplMem(sizeof(PAGE_NUMBER)))) {
ODS(("GetStartPageList - Run out of memory"));
goto CleanUp;
}
pTemp->pNext = *pHead;
pTemp->dwPageNumber = dwPageNumber;
*pHead = pTemp;
// flip the bOdd flag
*pbOdd = !*pbOdd;
}
}
}
bReturn = TRUE;
CleanUp:
// Free up the memory in case of a failure.
if (!bReturn) {
while (pTemp = *pHead) {
*pHead = (*pHead)->pNext;
FreeSplMem(pTemp);
}
}
return bReturn;
}
BOOL
CopyDevmode(
PPRINTPROCESSORDATA pData,
LPDEVMODEW *pDevmode)
/*++
Function Description: Copies the devmode in pData or the default devmode into pDevmode.
Parameters: pData - Data structure for the print job
pDevmode - pointer to devmode
Return Value: TRUE if successful
FALSE otherwise
--*/
{
HANDLE hDrvPrinter = NULL;
BOOL bReturn = FALSE;
fnWinSpoolDrv fnList;
LONG lNeeded;
HMODULE hWinSpoolDrv = NULL;
if (pData->pDevmode) {
lNeeded = pData->pDevmode->dmSize + pData->pDevmode->dmDriverExtra;
if (*pDevmode = (LPDEVMODEW) AllocSplMem(lNeeded)) {
memcpy(*pDevmode, pData->pDevmode, lNeeded);
} else {
goto CleanUp;
}
} else {
// Get the default devmode
ZeroMemory ( &fnList, sizeof (fnWinSpoolDrv) );
//
// Get the pointers to the client side functions.
//
if (!(hWinSpoolDrv = LoadLibrary(TEXT("winspool.drv"))))
{
// Could not load the client side of the spooler
goto CleanUp;
}
fnList.pfnOpenPrinter = (BOOL (*)(LPTSTR, LPHANDLE, LPPRINTER_DEFAULTS))
GetProcAddress( hWinSpoolDrv,"OpenPrinterW" );
fnList.pfnClosePrinter = (BOOL (*)(HANDLE))
GetProcAddress( hWinSpoolDrv,"ClosePrinter" );
fnList.pfnDocumentProperties = (LONG (*)(HWND, HANDLE, LPWSTR, PDEVMODE, PDEVMODE, DWORD))
GetProcAddress( hWinSpoolDrv,"DocumentPropertiesW" );
if ( NULL == fnList.pfnOpenPrinter ||
NULL == fnList.pfnClosePrinter ||
NULL == fnList.pfnDocumentProperties )
{
goto CleanUp;
}
// Get a client side printer handle to pass to the driver
if (!(* (fnList.pfnOpenPrinter))(pData->pPrinterName, &hDrvPrinter, NULL)) {
ODS(("Open printer failed\nPrinter %ws\n", pData->pPrinterName));
goto CleanUp;
}
lNeeded = (* (fnList.pfnDocumentProperties))(NULL,
hDrvPrinter,
pData->pPrinterName,
NULL,
NULL,
0);
if (lNeeded <= 0 ||
!(*pDevmode = (LPDEVMODEW) AllocSplMem(lNeeded)) ||
(* (fnList.pfnDocumentProperties))(NULL,
hDrvPrinter,
pData->pPrinterName,
*pDevmode,
NULL,
DM_OUT_BUFFER) < 0) {
if (*pDevmode) {
FreeSplMem(*pDevmode);
*pDevmode = NULL;
}
ODS(("DocumentProperties failed\nPrinter %ws\n",pData->pPrinterName));
goto CleanUp;
}
}
bReturn = TRUE;
CleanUp:
if (hDrvPrinter) {
(* (fnList.pfnClosePrinter))(hDrvPrinter);
}
if ( hWinSpoolDrv )
{
FreeLibrary (hWinSpoolDrv);
hWinSpoolDrv = NULL;
}
return bReturn;
}
BOOL
PrintEMFJob(
PPRINTPROCESSORDATA pData,
LPWSTR pDocumentName)
/*++
Function Description: Prints out a job with EMF data type.
Parameters: pData - Data structure for this job
pDocumentName - Name of this document
Return Value: TRUE if successful
FALSE if failed - GetLastError() will return reason.
--*/
{
HANDLE hSpoolHandle = NULL;
DWORD LastError;
HDC hPrinterDC = NULL;
BOOL bReverseOrderPrinting, bReturn = FALSE, bSetWorldXform = TRUE;
BOOL bCollate, bDuplex, bBookletPrint, bStartDoc = FALSE, bOdd = FALSE;
BOOL bUpdateAttributes = FALSE;
SHORT dmCollate,dmCopies;
DWORD dwNumberOfPagesPerSide, dwTotalNumberOfPages = 0, dwNupBorderFlags;
DWORD dwJobNumberOfPagesPerSide, dwDrvNumberOfPagesPerSide, dwDuplexMode;
DWORD dwJobNumberOfCopies, dwDrvNumberOfCopies,dwRemainingCopies;
DWORD dwJobOrder, dwDrvOrder, dwOptimization;
DOCINFOW DocInfo;
XFORM OldXForm;
PPAGE_NUMBER pHead = NULL,pTemp;
ATTRIBUTE_INFO_3 AttributeInfo;
LPDEVMODEW pDevmode = NULL, pFirstDM = NULL, pCopyDM;
// Copy the devmode into pDevMode
if (!CopyDevmode(pData, &pDevmode)) {
ODS(("CopyDevmode failed\nPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
}
if ( ! BIsDevmodeOfLeastAcceptableSize (pDevmode) )
{
ODS(("Devmode not big enough. Failing job.\nPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
}
// Update resolution before CreateDC for monochrome optimization
if (!GetJobAttributes(pData->pPrinterName,
pDevmode,
&AttributeInfo)) {
ODS(("GetJobAttributes failed\nPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
} else {
if (AttributeInfo.dwColorOptimization) {
if (pDevmode->dmPrintQuality != AttributeInfo.dmPrintQuality ||
pDevmode->dmYResolution != AttributeInfo.dmYResolution)
{
pDevmode->dmPrintQuality = AttributeInfo.dmPrintQuality;
pDevmode->dmYResolution = AttributeInfo.dmYResolution;
bUpdateAttributes = TRUE;
}
}
if (pDevmode->dmFields & DM_COLLATE)
dmCollate = pDevmode->dmCollate;
else
dmCollate = DMCOLLATE_FALSE;
if (pDevmode->dmFields & DM_COPIES)
dmCopies = pDevmode->dmCopies;
else
dmCopies = 0;
}
// Get spool file handle and printer device context from GDI
try {
hSpoolHandle = GdiGetSpoolFileHandle(pData->pPrinterName,
pDevmode,
pDocumentName);
if (hSpoolHandle) {
hPrinterDC = GdiGetDC(hSpoolHandle);
}
} except (EXCEPTION_EXECUTE_HANDLER) {
ODS(("PrintEMFJob gave an exceptionPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
}
if (!hPrinterDC || !hSpoolHandle) {
goto CleanUp;
}
// Use the first devmode in the spool file to update the copy count
// and the collate setting
if (GdiGetDevmodeForPagePvt(hSpoolHandle, 1, &pFirstDM, NULL) &&
pFirstDM) {
if (pFirstDM->dmFields & DM_COPIES) {
pDevmode->dmFields |= DM_COPIES;
pDevmode->dmCopies = pFirstDM->dmCopies;
}
if ( (pFirstDM->dmFields & DM_COLLATE) &&
IS_DMSIZE_VALID ( pDevmode, dmCollate) )
{
pDevmode->dmFields |= DM_COLLATE;
pDevmode->dmCollate = pFirstDM->dmCollate;
}
}
// The number of copies of the print job is the product of the number of copies set
// from the driver UI (present in the devmode) and the number of copies in pData struct
dwJobNumberOfCopies = (pDevmode->dmFields & DM_COPIES) ? pData->Copies*pDevmode->dmCopies
: pData->Copies;
pDevmode->dmCopies = (short) dwJobNumberOfCopies;
pDevmode->dmFields |= DM_COPIES;
// If collate is true this limits the ability of the driver to do multiple copies
// and causes the driver (PS) supported n-up to print blank page borders for reverse printing.
// Therefore we disable collate for 1 page multiple copy jobs or no copies but n-up since
// collate has no meaning in those cases.
//
if ((pDevmode->dmFields & DM_COLLATE) && pDevmode->dmCollate == DMCOLLATE_TRUE)
{
if (dwJobNumberOfCopies > 1)
{
// Get the number of pages in the job. This call waits till the
// last page is spooled.
try {
dwTotalNumberOfPages = GdiGetPageCount(hSpoolHandle);
} except (EXCEPTION_EXECUTE_HANDLER) {
ODS(("PrintEMFJob gave an exceptionPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto SkipCollateDisable;
}
if (dwTotalNumberOfPages > AttributeInfo.dwDrvNumberOfPagesPerSide)
goto SkipCollateDisable;
}
// if copies == 1 and driver n-up we will disable collate
//
else if (AttributeInfo.dwDrvNumberOfPagesPerSide <= 1 && dmCollate == DMCOLLATE_TRUE)
goto SkipCollateDisable;
pDevmode->dmCollate = DMCOLLATE_FALSE;
if (pFirstDM &&
IS_DMSIZE_VALID ( pFirstDM, dmCollate) )
{
pFirstDM->dmCollate = DMCOLLATE_FALSE;
}
}
SkipCollateDisable:
// Update the job attributes but only if something has changed. This is an expensive
// call so we only make a second call to GetJobAttributes if something has changed.
//
if (bUpdateAttributes || pDevmode->dmCopies != dmCopies ||
((pDevmode->dmFields & DM_COLLATE) && (pDevmode->dmCollate != dmCollate)))
{
if (!GetJobAttributes(pData->pPrinterName,
pDevmode,
&AttributeInfo)) {
ODS(("GetJobAttributes failed\nPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
}
}
// Initialize bReverseOrderPrinting, dwJobNumberOfPagesPerSide,
// dwDrvNumberOfPagesPerSide, dwNupBorderFlags, dwJobNumberOfCopies,
// dwDrvNumberOfCopies and bCollate
dwJobNumberOfPagesPerSide = AttributeInfo.dwJobNumberOfPagesPerSide;
dwDrvNumberOfPagesPerSide = AttributeInfo.dwDrvNumberOfPagesPerSide;
dwNupBorderFlags = AttributeInfo.dwNupBorderFlags;
dwJobNumberOfCopies = AttributeInfo.dwJobNumberOfCopies;
dwDrvNumberOfCopies = AttributeInfo.dwDrvNumberOfCopies;
dwJobOrder = AttributeInfo.dwJobPageOrderFlags & ( NORMAL_PRINT | REVERSE_PRINT);
dwDrvOrder = AttributeInfo.dwDrvPageOrderFlags & ( NORMAL_PRINT | REVERSE_PRINT);
bReverseOrderPrinting = (dwJobOrder != dwDrvOrder);
dwJobOrder = AttributeInfo.dwJobPageOrderFlags & BOOKLET_PRINT;
dwDrvOrder = AttributeInfo.dwDrvPageOrderFlags & BOOKLET_PRINT;
bBookletPrint = (dwJobOrder != dwDrvOrder);
bCollate = (pDevmode->dmFields & DM_COLLATE) &&
(pDevmode->dmCollate == DMCOLLATE_TRUE);
bDuplex = (pDevmode->dmFields & DM_DUPLEX) &&
(pDevmode->dmDuplex != DMDUP_SIMPLEX);
if (!dwJobNumberOfCopies) {
//
// Some applications can set the copy count to 0.
// In this case we exit.
//
bReturn = TRUE;
goto CleanUp;
}
if (bDuplex) {
dwDuplexMode = (pDevmode->dmDuplex == DMDUP_HORIZONTAL) ? EMF_DUP_HORZ
: EMF_DUP_VERT;
} else {
dwDuplexMode = EMF_DUP_NONE;
}
if (bBookletPrint) {
if (!bDuplex) {
// Not supported w/o duplex printing. Use default settings.
bBookletPrint = FALSE;
dwDrvNumberOfPagesPerSide = 1;
dwJobNumberOfPagesPerSide = 1;
} else {
// Fixed settings for pages per side.
dwDrvNumberOfPagesPerSide = 1;
dwJobNumberOfPagesPerSide = 2;
}
}
// Number of pages per side that the print processor has to play
dwNumberOfPagesPerSide = (dwDrvNumberOfPagesPerSide == 1)
? dwJobNumberOfPagesPerSide
: 1;
if (dwNumberOfPagesPerSide == 1) {
// if the print processor is not doing nup, don't draw borders
dwNupBorderFlags = NO_BORDER_PRINT;
}
//
// Color optimization may cause wrong output with duplex
//
dwOptimization = (AttributeInfo.dwColorOptimization == COLOR_OPTIMIZATION &&
!bDuplex && dwJobNumberOfPagesPerSide == 1)
? EMF_PP_COLOR_OPTIMIZATION
: 0;
// Check for Valid Option for n-up printing
if (!ValidNumberForNUp(dwNumberOfPagesPerSide)) {
ODS(("Invalid N-up option\nPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
}
if (bReverseOrderPrinting || bBookletPrint) {
// Get the number of pages in the job. This call waits till the
// last page is spooled.
try {
dwTotalNumberOfPages= GdiGetPageCount(hSpoolHandle);
} except (EXCEPTION_EXECUTE_HANDLER) {
ODS(("PrintEMFJob gave an exceptionPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
}
// Get start page list for reverse printing
// Check for a change of devmode between pages only if Nup and PCL driver
if (!GetStartPageList(hSpoolHandle,
&pHead,
dwTotalNumberOfPages,
dwJobNumberOfPagesPerSide,
FALSE,
&bOdd)) {
goto CleanUp;
}
}
// Save the old transformation on hPrinterDC
if (!SetGraphicsMode(hPrinterDC,GM_ADVANCED) ||
!GetWorldTransform(hPrinterDC,&OldXForm)) {
bSetWorldXform = FALSE;
ODS(("Transformation matrix can't be set\nPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
}
// pCopyDM will be used for changing the copy count
pCopyDM = pFirstDM ? pFirstDM : pDevmode;
pCopyDM->dmPrintQuality = pDevmode->dmPrintQuality;
pCopyDM->dmYResolution = pDevmode->dmYResolution;
try {
DocInfo.cbSize = sizeof(DOCINFOW);
DocInfo.lpszDocName = pData->pDocument;
DocInfo.lpszOutput = pData->pOutputFile;
DocInfo.lpszDatatype = NULL;
if (!GdiStartDocEMF(hSpoolHandle, &DocInfo)) goto CleanUp;
bStartDoc = TRUE;
if (bCollate) {
dwRemainingCopies = dwJobNumberOfCopies & 0x0000FFFF ;
while (dwRemainingCopies) {
if (dwRemainingCopies <= dwDrvNumberOfCopies) {
SetDrvCopies(hPrinterDC, pCopyDM, dwRemainingCopies);
dwRemainingCopies = 0;
} else {
SetDrvCopies(hPrinterDC, pCopyDM, dwDrvNumberOfCopies);
dwRemainingCopies -= dwDrvNumberOfCopies;
}
if (!PrintEMFSingleCopy(hSpoolHandle,
hPrinterDC,
bReverseOrderPrinting,
dwDrvNumberOfPagesPerSide,
dwNumberOfPagesPerSide,
dwTotalNumberOfPages,
dwNupBorderFlags,
dwJobNumberOfCopies,
dwDrvNumberOfCopies,
bCollate,
bOdd,
bBookletPrint,
dwOptimization,
dwDuplexMode,
pCopyDM,
pHead,
pData)) {
goto CleanUp;
}
}
} else {
if (!PrintEMFSingleCopy(hSpoolHandle,
hPrinterDC,
bReverseOrderPrinting,
dwDrvNumberOfPagesPerSide,
dwNumberOfPagesPerSide,
dwTotalNumberOfPages,
dwNupBorderFlags,
dwJobNumberOfCopies,
dwDrvNumberOfCopies,
bCollate,
bOdd,
bBookletPrint,
dwOptimization,
dwDuplexMode,
pCopyDM,
pHead,
pData)) {
goto CleanUp;
}
}
bStartDoc = FALSE;
if (!GdiEndDocEMF(hSpoolHandle)) goto CleanUp;
} except (EXCEPTION_EXECUTE_HANDLER) {
ODS(("PrintEMFSingleCopy gave an exception\nPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
goto CleanUp;
}
bReturn = TRUE;
CleanUp:
//
// Preserve the last error
//
LastError = bReturn ? ERROR_SUCCESS : GetLastError();
if (bStartDoc) {
GdiEndDocEMF(hSpoolHandle);
}
if (bSetWorldXform && hPrinterDC) {
SetWorldTransform(hPrinterDC, &OldXForm);
}
while (pTemp = pHead) {
pHead = pHead->pNext;
FreeSplMem(pTemp);
}
if (pDevmode) {
FreeSplMem(pDevmode);
}
try {
if (hSpoolHandle) {
GdiDeleteSpoolFileHandle(hSpoolHandle);
}
} except (EXCEPTION_EXECUTE_HANDLER) {
ODS(("GdiDeleteSpoolFileHandle failed\nPrinter %ws\nDocument %ws\nJobID %u\n", pData->pDevmode->dmDeviceName, pData->pDocument, pData->JobId));
}
SetLastError(LastError);
return bReturn;
}
/*++
Function Name
GdiGetDevmodeForPagePvt
Function Description.
In some cases, GDI's GdiGetDevmodeForPage returns a devmode
that is based on an old format of devmode. e.g. Win3.1 format. The size of such a devmode
can be smaller than the latest Devmode. This can lead to unpredictable issues.
Also, sometimes the devmode returned is even smaller than Win3.1 format (due to possible
corruption).
This function is a wrapper around GDI's GdiGetDevmodeForPage and partially takes care of this
situation by doing an extra checking for devmode.
Parameters:
hSpoolHandle -- the handle to the spool file
dwPageNumber -- the devmode related to this page number is requested.
ppCurrDM -- the devmode for the dwPageNumber is placed here.
ppLastDM -- devmode for dwPageNumber-1 is placed here. Can be NULL. (if n
ot NULL)
Return Values: TRUE if a valid devmode was obtained from GDI
FALSE otherwise
--*/
BOOL GdiGetDevmodeForPagePvt(
IN HANDLE hSpoolHandle,
IN DWORD dwPageNumber,
OUT PDEVMODEW *ppCurrDM,
OUT PDEVMODEW *ppLastDM
)
{
if ( NULL == ppCurrDM )
{
return FALSE;
}
*ppCurrDM = NULL;
if ( ppLastDM )
{
*ppLastDM = NULL;
}
if (!GdiGetDevmodeForPage(hSpoolHandle,
dwPageNumber,
ppCurrDM,
ppLastDM) )
{
ODS(("GdiGetDevmodeForPage failed\n"));
return FALSE;
}
//
// If GdiGetDevmodeForPage has succeeded, then *ppCurrDM should have valid values
// Also if ppLastDM is not NULL, then *ppLastDM should also have valid values.
//
// GDI guarantees that the size of the devmode is atleast dmSize+dmDriverExtra.
// So we dont need to check for that. But we still need to check some other dependencies
//
//
if ( NULL == *ppCurrDM ||
FALSE == BIsDevmodeOfLeastAcceptableSize (*ppCurrDM)
)
{
return FALSE;
}
//
// It is possible for GdiGetDevmodeForPage to return TRUE (i.e. success)
// but still not fill in the *ppLastDM. So NULL *ppLastDM is not an error
//
if ( ppLastDM && *ppLastDM &&
FALSE == BIsDevmodeOfLeastAcceptableSize (*ppLastDM)
)
{
// @@BEGIN_DDKSPLIT
// We could either ignore the error, set *ppLastDM to NULL and return TRUE
// or we could fail the call. I think failing the call is better.
SPLASSERT(FALSE);
// skupec - moved HSPLIT token since SPLASSERT() isn't defined in the DDK.
// @@END_DDKSPLIT
return FALSE;
}
return TRUE;
}
/*++
Function Name
BIsDevmodeOfLeastAcceptableSize
Function Description.
// @@BEGIN_DDKSPLIT
Though ideally we should be checking that pdevmode is so big that atleast we can access till
dmLogPixels(which is the last DEVMODE field), without AVing,
but due to backward compatibility issues I can only check till dmYResolution.
dmYResolution is the last field of pdevmode that is currently accessed in print proc
without checking for its validity using dmFields. (currently it is line 2519 in this file).
dmCollate is also used, but that is beyond the limit of the
Win3.1 devmode (Fields including and after dmCollate were not part of Win3.1 devmode).
So if the incoming devmode is bigger than offsetof(dmYResolution) but less than dmCollate,
it should be accepted. Then there is dmTTOption which is not really used. So lets just
ignore it.
// @@END_DDKSPLIT
Parameters:
pdm -- the pointer to the devmode.
Return Values: TRUE if devmode is of least acceptable size.
FALSE otherwise
--*/
BOOL BIsDevmodeOfLeastAcceptableSize(
IN PDEVMODE pdm)
{
if ( NULL == pdm )
{
return FALSE;
}
if ( IS_DMSIZE_VALID((pdm),dmYResolution) )
{
return TRUE;
}
return FALSE;
}