8

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

到目前为止,使用 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。它目前不起作用 - 它会更改其他内存(例如副本),并且无法成功更改“基础”纸张大小。我认为我需要做的就是指定输出缓冲区标志以进行更改?

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

4

1 回答 1

2

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

(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。

于 2010-04-02T11:18:27.557 回答