PrinterSettings.GetHdevmode()有错误吗?

时间:2010-03-12 23:55:23

标签: c# printing printer-properties

我希望能够在不打开打印机属性窗口的情况下更改打印机属性...

使用DocumentProperties(从winspool.drv导入)函数到目前为止失败了,因为虽然很容易抑制对话框显示,但似乎PrinterSettings.GetHdevmode()返回的值不反映PrinterSettings正在调用它,但是之前的打印机属性的值返回OK。例如,这给了我上次调用属性的先前(错误)值,而不是它应该来自PrinterSettings对象的值:

IntPtr hdevmode = PrinterSettings.GetHdevmode(PrinterSettings.DefaultPageSettings);
PrinterSettings.SetHdevmode(hdevmode);
PrinterSettings.DefaultPageSettings.SetHdevmode(hdevmode);

GetHdevmode也有错误或这是它应该做的事情吗?有没有C#可以解决这个问题,还是有人甚至没有任何相关信息?我甚至很难找到关于这个主题的信息。

提前感谢任何见解。

编辑: 我不想让这个问题过于个人化,但希望在这种情况下拥有所有信息可以为其他人提供一个有用的解决方案。

这是我编写的C ++ DLL,以便为此问题找到解决方法。它目前没有工作 - 它改变了其他内存,如副本,并没有成功改变“底层”纸张大小。我认为我需要做的就是指定out缓冲区标志以进行更改?

extern "C" __declspec(dllexport) DEVMODE* __stdcall GetRealHDevMode(int width, int height, char *printerName, DEVMODE* inDevMode)
    {
//declare handles and variables
HANDLE printerHandle;
LPHANDLE printerHandlePointer(&printerHandle);

//get printer handle pointer
OpenPrinter((LPWSTR)printerName, printerHandlePointer, NULL);

//Get size needed for public and private devmode data and declare devmode structure
size_t devmodeSize = DocumentProperties(NULL, printerHandle, (LPWSTR)printerName, NULL, NULL, 0);
DEVMODE* devmode = reinterpret_cast<DEVMODE*>(new char[devmodeSize + sizeof(DEVMODE) + sizeof(inDevMode->dmDriverExtra)]);

//lock memory
GlobalLock(devmode);

//fill the out buffer
DocumentProperties(NULL, printerHandle, (LPWSTR)printerName, devmode, NULL, DM_OUT_BUFFER);

//change the values as required
devmode->dmPaperWidth = width;
devmode->dmPaperLength = height;
devmode->dmPaperSize = DMPAPER_USER;

devmode->dmFields &= ~DM_PAPERSIZE;
devmode->dmFields &= ~DM_PAPERLENGTH;
devmode->dmFields &= ~DM_PAPERWIDTH;
devmode->dmFields |= (DM_PAPERSIZE | DM_PAPERLENGTH | DM_PAPERWIDTH);

//input flag on now to put the changes back in
DocumentProperties(NULL, printerHandle, (LPWSTR)printerName, devmode, devmode, DM_IN_BUFFER | DM_OUT_BUFFER);

//unlock memory
GlobalUnlock(devmode);

//return the devmode that was used to alter the settings
return devmode;
    }

我认为C ++代码足以改变设置,所以我在C#中做的就是这样:

public PrinterSettings ChangePrinterProperties(PrinterSettings inPrinterSettings)
    {
        IntPtr TemphDevMode = inPrinterSettings.GetHdevmode(inPrinterSettings.DefaultPageSettings);
        IntPtr hDevMode = GetRealHDevMode((int)(inPrinterSettings.DefaultPageSettings.PaperSize.Width * 2.54F),
            (int)(inPrinterSettings.DefaultPageSettings.PaperSize.Height * 2.54F),
            inPrinterSettings.PrinterName, TemphDevMode);
        GlobalFree(hDevMode);
        return inPrinterSettings;
    }

更新:使用dmPaperSize和dmFields稍微更改了订单。改善结果;还没到那里。

更新2:好的,我发现一个微软页面说文档错误。 MSDN说要在指定宽度和高度时将dmPaperSize设置为0,而Microsoft支持更正表示将其设置为DMPAPER_USER。 http://support.microsoft.com/kb/108924

1 个答案:

答案 0 :(得分:2)

在DEVMODE中指定纸张尺寸的方式有两个问题:

(1)如果指定DM_PAPERWIDTH或DM_PAPERLENGTH或两者,则不能同时设置DM_PAPERSIZE位。这取决于打印机驱动程序,但许多驱动程序将忽略上述代码中的DM_PAPERLENGTH / WIDTH。

(2)许多驱动程序根本不支持DM_PAPERLENGTH / WIDTH。使用此类驱动程序,您无法像上面尝试的那样设置纸张尺寸。您只能选择一个预定义的dmPaperSizes。

您可以使用DeviceCapabilities(DC_FIELDS)来确定您的驱动程序是否支持DM_PAPERLENGTH / WIDTH。

您可以使用DeviceCapabilities(DC_PAPERS)枚举允许的dmPaperSizes。