HOWTO: Set Duplex Printing for Word AutomationID: Q230743
|
Microsoft Word for Windows does not provide a method for Automation clients to set the duplex print flag before starting a print job. Although there is a Duplex parameter in the PrintOut method, it applies to the Macintosh version only. However, developers can work around this limitation on Windows systems by changing the duplex flag for the active printer driver before calling Word's PrintOut function.
This article demonstrates how to use the Windows API to change the duplex setting of the active printer and allow a Word document to be printed in duplex.
This code uses the DocumentProperties API to change the print settings of the printer driver to enable duplex printing. For this code to work successfully, the end user will need adequate permissions to change the global print settings for the printer. If a user does not have the proper permission to change driver settings, they will get an Access Denied error on the OpenPrinter API call.
For users of Windows NT who need to print to a shared network printer, this can be a problem because the print driver does not reside on the local machine but on the print server. Although it is possible for an administrator to configure the print server to give end users the proper permission to change global settings, it is NOT desirable to do so in most cases. To work around this problem, it is possible to install a local print driver for the network printer, and let each user control the settings for their local systems.
\\printserver\printername (using the exact path name to the printer)
Option Explicit
Public Type PRINTER_DEFAULTS
pDatatype As Long
pDevmode As Long
DesiredAccess As Long
End Type
Public Type PRINTER_INFO_2
pServerName As Long
pPrinterName As Long
pShareName As Long
pPortName As Long
pDriverName As Long
pComment As Long
pLocation As Long
pDevmode As Long ' Pointer to DEVMODE
pSepFile As Long
pPrintProcessor As Long
pDatatype As Long
pParameters As Long
pSecurityDescriptor As Long ' Pointer to SECURITY_DESCRIPTOR
Attributes As Long
Priority As Long
DefaultPriority As Long
StartTime As Long
UntilTime As Long
Status As Long
cJobs As Long
AveragePPM As Long
End Type
Public Type DEVMODE
dmDeviceName As String * 32
dmSpecVersion As Integer
dmDriverVersion As Integer
dmSize As Integer
dmDriverExtra As Integer
dmFields As Long
dmOrientation As Integer
dmPaperSize As Integer
dmPaperLength As Integer
dmPaperWidth As Integer
dmScale As Integer
dmCopies As Integer
dmDefaultSource As Integer
dmPrintQuality As Integer
dmColor As Integer
dmDuplex As Integer
dmYResolution As Integer
dmTTOption As Integer
dmCollate As Integer
dmFormName As String * 32
dmUnusedPadding As Integer
dmBitsPerPel As Integer
dmPelsWidth As Long
dmPelsHeight As Long
dmDisplayFlags As Long
dmDisplayFrequency As Long
dmICMMethod As Long
dmICMIntent As Long
dmMediaType As Long
dmDitherType As Long
dmReserved1 As Long
dmReserved2 As Long
End Type
Public Const DM_DUPLEX = &H1000&
Public Const DM_IN_BUFFER = 8
Public Const DM_OUT_BUFFER = 2
Public Const PRINTER_ACCESS_ADMINISTER = &H4
Public Const PRINTER_ACCESS_USE = &H8
Public Const STANDARD_RIGHTS_REQUIRED = &HF0000
Public Const PRINTER_ALL_ACCESS = (STANDARD_RIGHTS_REQUIRED Or _
PRINTER_ACCESS_ADMINISTER Or PRINTER_ACCESS_USE)
Public Declare Function ClosePrinter Lib "winspool.drv" _
(ByVal hPrinter As Long) As Long
Public Declare Function DocumentProperties Lib "winspool.drv" _
Alias "DocumentPropertiesA" (ByVal hwnd As Long, _
ByVal hPrinter As Long, ByVal pDeviceName As String, _
ByVal pDevModeOutput As Long, ByVal pDevModeInput As Long, _
ByVal fMode As Long) As Long
Public Declare Function GetPrinter Lib "winspool.drv" Alias _
"GetPrinterA" (ByVal hPrinter As Long, ByVal Level As Long, _
pPrinter As Byte, ByVal cbBuf As Long, pcbNeeded As Long) As Long
Public Declare Function OpenPrinter Lib "winspool.drv" Alias _
"OpenPrinterA" (ByVal pPrinterName As String, phPrinter As Long, _
pDefault As PRINTER_DEFAULTS) As Long
Public Declare Function SetPrinter Lib "winspool.drv" Alias _
"SetPrinterA" (ByVal hPrinter As Long, ByVal Level As Long, _
pPrinter As Byte, ByVal Command As Long) As Long
Public Declare Sub CopyMemory Lib "kernel32" Alias "RtlMoveMemory" _
(pDest As Any, pSource As Any, ByVal cbLength As Long)
' ==================================================================
' SetPrinterDuplex
'
' sPrinterName is the name of the printer
' nDuplexSetting is standard settings
' 1 = None
' 2 = Duplex on long edge (book)
' 3 = Duplex on short edge (legal)
' ==================================================================
Public Sub SetPrinterDuplex(ByVal sPrinterName As String, _
ByVal nDuplexSetting As Long)
Dim hPrinter As Long
Dim pd As PRINTER_DEFAULTS
Dim dm As DEVMODE
Dim pinfo As PRINTER_INFO_2
Dim yPInfoMemory() As Byte
Dim nBytesNeeded As Long
Dim nRet As Long, nJunk As Long
On Error GoTo cleanup
If (nDuplexSetting < 1) Or (nDuplexSetting > 3) Then
MsgBox "Error: dwDuplexSetting is incorrect."
Exit Sub
End If
pd.DesiredAccess = PRINTER_ALL_ACCESS
nRet = OpenPrinter(sPrinterName, hPrinter, pd)
If (nRet = 0) Or (hPrinter = 0) Then
If Err.LastDllError = 5 Then
MsgBox "Access denied -- See the article for more info."
Else
MsgBox "Open Printer failed for some reason."
End If
Exit Sub
End If
Call GetPrinter(hPrinter, 2, 0, 0, nBytesNeeded)
If (nBytesNeeded = 0) Then GoTo cleanup
ReDim yPInfoMemory(0 To nBytesNeeded) As Byte
nRet = GetPrinter(hPrinter, 2, yPInfoMemory(0), nBytesNeeded, nJunk)
If (nRet = 0) Then GoTo cleanup
Call CopyMemory(pinfo, yPInfoMemory(0), Len(pinfo))
If (pinfo.pDevmode = 0) Then GoTo cleanup
Call CopyMemory(dm, ByVal pinfo.pDevmode, Len(dm))
If Not CBool(dm.dmFields & DM_DUPLEX) Then GoTo cleanup
dm.dmDuplex = nDuplexSetting
Call CopyMemory(ByVal pinfo.pDevmode, dm, Len(dm))
nRet = DocumentProperties(0, hPrinter, sPrinterName, _
pinfo.pDevmode, pinfo.pDevmode, DM_IN_BUFFER Or DM_OUT_BUFFER)
pinfo.pSecurityDescriptor = 0
Call CopyMemory(yPInfoMemory(0), pinfo, Len(pinfo))
Call SetPrinter(hPrinter, 2, yPInfoMemory(0), 0)
cleanup:
If (hPrinter <> 0) Then Call ClosePrinter(hPrinter)
End Sub
Option Explicit
Private Sub Command1_Click()
Dim oWord As Object
Dim oDoc As Object
Set oWord = CreateObject("Word.application")
oWord.Visible = True
Set oDoc = oWord.Documents.Add
oDoc.Range.Select
oWord.Selection.TypeText "This is on page 1" & vbCr
oWord.Selection.InsertBreak 1
oWord.Selection.TypeText "This is page 2"
SetPrinterDuplex Printer.DeviceName, 2
oDoc.PrintOut Background:=False
SetPrinterDuplex Printer.DeviceName, 1
MsgBox "Print Done", vbMsgBoxSetForeground
oDoc.Saved = True
oDoc.Close
Set oDoc = Nothing
oWord.Quit
Set oWord = Nothing
End Sub
For more information on problems you might see while trying to print Word documents using a duplex printer, please see the following articles in the Microsoft Knowledge Base:
Q176189 WD97: Shading of Solid Black or Gray Won't Print Duplex on NT
Q196857 WD97: First Page Prints on Back of Last Page with Duplex
Q214683 WD97: Duplex Printing Does Not Duplex with Objects on Page
© Microsoft Corporation 1999, All Rights Reserved.
Contributions by Richard R. Taylor, Microsoft Corporation
Additional query words:
Keywords : kbAPI kbAutomation kbPrinting KbVBA kbVBp500 kbVBp600 kbWord kbGrpDSO kbOffice2000 kbword2000
Version : WINDOWS:2000,5.0,6.0,97; :
Platform : WINDOWS
Issue type : kbhowto
Last Reviewed: June 8, 1999