26

一段时间以来,我一直在尝试确定一种使用标准 Java Print 库打印具有某些属性的文件(特别是 PDF 文档)的方法,特别是打印到某些托盘或使用双面打印。

有很多关于如何做到这一点的文档,事实上,我已经研究并尝试了这些方法。典型的方式是这样的:

public static void main (String [] args) {
    try {

        PrintService[] pservices = PrintServiceLookup.lookupPrintServices(null, null);

        //Acquire Printer
        PrintService printer = null;
        for (PrintService serv: pservices) {
            System.out.println(serv.toString());
            if (serv.getName().equals("PRINTER_NAME_BLAH")) {
                printer = serv;
            }
        }

        if (printer != null) {
            System.out.println("Found!");


            //Open File
            FileInputStream fis = new FileInputStream("FILENAME_BLAH_BLAH.pdf");

            //Create Doc out of file, autosense filetype
            Doc pdfDoc = new SimpleDoc(fis, DocFlavor.INPUT_STREAM.AUTOSENSE, null);

            //Create job for printer
            DocPrintJob printJob = printer.createPrintJob();

            //Create AttributeSet
            PrintRequestAttributeSet pset = new HashPrintRequestAttributeSet();

            //Add MediaTray to AttributeSet
            pset.add(MediaTray.TOP);

            //Add Duplex Option to AttributeSet
            pset.add(Sides.DUPLEX);

            //Print using Doc and Attributes
            printJob.print(pdfDoc, pset);

            //Close File
            fis.close();

        }

    }
    catch (Throwable t) {
        t.printStackTrace();
    }
}

简而言之,您执行以下操作

  1. 查找打印机
  2. 创建打印机作业
  3. 创建一个属性集
  4. 将属性添加到 AttributeSet,例如 Tray 和 Duplex
  5. 使用 AttributeSet 在打印机作业上调用 print

这里的问题是,尽管有记录的方法,以及我从几个教程中发现的方法,但这种方法......不起作用。现在请记住,我知道这听起来不太描述,但请听我说。 我不轻易说...

PrinterJob 的官方文档实际上提到了 AttributeSet 在默认实现中被忽略。 这里看到的源代码表明这是真的 - 属性被传入并完全忽略。

显然,您需要该类的某种扩展版本,它可能基于特定的打印机及其功能?我试图编写一些测试代码来告诉我这些功能 - 我们在办公室设置了各种各样的打印机,大小不一,简单或充满花里胡哨 - 更不用说我电脑上的几个驱动程序只是为了伪- 打印机驱动程序,无需任何硬件即可创建文档和模拟打印机。测试代码如下:

public static void main (String [] args) {

    PrintService[] pservices = PrintServiceLookup.lookupPrintServices(null, null);

    for (PrintService serv: pservices) {
        System.out.println(serv.toString());

        printFunctionality(serv, "Trays", MediaTray.class);
        printFunctionality(serv, "Copies", Copies.class);
        printFunctionality(serv, "Print Quality", PrintQuality.class);
        printFunctionality(serv, "Color", ColorSupported.class);
        printFunctionality(serv, "Media Size", MediaSize.class);
        printFunctionality(serv, "Accepting Jobs", PrinterIsAcceptingJobs.class);
    }
}

private static void printFunctionality(PrintService serv, String attrName, Class<? extends Attribute> attr) {
    boolean isSupported = serv.isAttributeCategorySupported(attr);
    System.out.println("    " + attrName + ": " + (isSupported ? "Y" : "N"));
}

我发现的结果是,每台打印机都无一例外地返回支持“副本”,而所有其他属性都不支持。此外,无论这看起来多么难以置信,每台打印机的功能都是相同的。

不可避免的问题是多层次的:如何以注册的方式发送属性?此外,如何正确检测打印机的功能?确实,PrinterJob 类实际上是否以可用的方式扩展,或者属性总是被忽略?

我在 Internet 上找到的示例似乎向我表明,后一个问题的答案是“不,它们总是被忽略”,这对我来说似乎很荒谬(但随着我筛选数百页,它变得越来越可信)。 这段代码是 Sun 简单设置但从未工作到完成状态的代码吗?如果是这样,是否有任何替代方案?

4

5 回答 5

10

问题在于 Java 打印 API 是世界之间的桥梁。打印机制造商不发布 JVM 驱动程序。他们发布了适用于 Windows、Macintosh 的驱动程序,并且也许有人拥有适用于一个或多个 *nix 平台的给定打印机的驱动程序。

随之而来的是一些在某个主机系统上的 JVM 中运行的 Java 代码。当您开始查询打印机功能时,您不是在与打印机对话——您是在与 java.awt.print 中的一个桥接类对话,该桥接类连接到 JVM,它连接到主机操作系统,连接到任何特定的为给定的打印机安装了驱动程序。所以有几个地方可能会崩溃......您所在的特定JVM可能会也可能不会完全实现用于查询打印机功能的API,更不用说为给定作业传递这些参数了。

几点建议:

  1. 研究 javax.print 类作为 java.awt.print 的替代品——我从那里获得了更多的打印运气。
  2. 尝试为您的打印机使用替代打印驱动程序——您可以定义到给定打印机的多个命名连接,每个连接都有不同的驱动程序。如果您有制造商提供的驱动程序,请尝试更通用的驱动程序,如果您有通用驱动程序,请尝试安装更具体的驱动程序。
  3. 在您的平台的替代 JVM 实现下运行您的代码
于 2013-01-24T16:28:09.583 回答
6

因此,我们不可避免地找到了一种方法,可以使用不同的设置打印到不同的托盘,但不是直接打印。我们发现通过 printJob.print 方法发送属性是不可能的,而且很多都没有改变。但是,我们能够设置打印作业的名称,然后使用低级 Perl 脚本拦截打印作业,解析名称,并在那里设置纸盘和双面打印设置。这是一个极端的黑客,但它有效。Java 打印机属性不起作用仍然是事实,如果您想设置它们,您将需要找到另一种方法。

于 2013-07-18T21:48:42.597 回答
1

我们对打印 PDF 有类似的要求,希望将一些页面发送到特定托盘,并且还希望将文档装订。我们使用Java 代码+ghost 脚本组合首先将PDF 转换为ghost 脚本,然后在ghost 脚本文件中添加PJL(打印作业语言)命令来选择托盘并装订文档。然后将编辑后的幽灵脚本文件发送到打印机。

这是用Java编写的完整示例

http://reddymails.blogspot.com/2014/07/how-to-print-documents-using-java-how.html

-内存

于 2015-01-22T18:40:23.947 回答
1

我发现打印机托盘的诀窍是遍历Media.classusing getSupportedAttributeValues(...),匹配人类可读的名称,然后选择该特定值。在具有多种托盘配置的 Windows、MacOS 上进行了测试。

String tray = "1";

// Handle human-readable names, see PRINTER_TRAY_ALIASES usage below for context.  Adjust as needed.
List<String> PRINTER_TRAY_ALIASES = Arrays.asList("", "Tray ", "Paper Cassette ");

// Get default printer
PrintService printService = PrintServiceLookup.lookupDefaultPrintService();

// Attributes to be provided at print time
PrintRequestAttributeSet pset = new HashPrintRequestAttributeSet();

Media[] supported = printService.getSupportedAttributeValues(Media.class, null, null);
for(Media m : supported) {           
    for(String pta : PRINTER_TRAY_ALIASES) {
        // Matches "1", "Tray 1", or "Paper Cassette 1"
        if (m.toString().trim().equalsIgnoreCase(pta + tray)) {
            attributes.add(m);
            break;
        }
    }
}

// Print, etc
// printJob.print(pdfDoc, pset);
于 2019-07-05T20:50:26.760 回答
1

这是它在 javafx 中的样子 托盘可能会有所不同,它还会打印出所有可用的托盘,只需更改托盘名称即可

private void printImage(Node node) {
    PrinterJob job = PrinterJob.createPrinterJob();
    if (job != null) {
        JobSettings js = job.getJobSettings();
        PaperSource papersource = js.getPaperSource();
        System.out.println("PaperSource=" + papersource);
        PrinterAttributes pa = printer.getPrinterAttributes();
        Set<PaperSource> s = pa.getSupportedPaperSources();
        System.out.println("# of papersources=" + s.size());
        if (s != null) {
            for (PaperSource newPaperSource : s) {
                System.out.println("newpapersource= " + newPaperSource);
                //Here is where you would put the tray name that is appropriate
                //in the contains section
                if(newPaperSource.toString().contains("Tray 2"))
                    js.setPaperSource(newPaperSource);
            }
        }
        job.getJobSettings().setJobName("Whatever");
        ObjectProperty<PaperSource> sources = job.getJobSettings().paperSourceProperty();
        System.out.println(sources.toString());
        boolean success = job.printPage(node);
        if (success) {
            System.out.println("PRINTING FINISHED");
            job.endJob();
            //Stage mainStage = (Stage) root.getScene().getWindow();
            //mainStage.close();
        }
    }
}

这是我的输出:

PaperSource=Paper source : Automatic
# of papersources=6
newpapersource= Paper source :
newpapersource= Paper source :  Manual Feed in Tray 1
newpapersource= Paper source :  Printer auto select
newpapersource= Paper source :  Tray 1
newpapersource= Paper source :  Tray 2
newpapersource= Paper source : Form-Source
ObjectProperty [bean:  Collation = UNCOLLATED
 Copies = 1
 Sides = ONE_SIDED
 JobName = Whatever
 Page ranges = null
 Print color = COLOR
 Print quality = NORMAL
 Print resolution = Feed res=600dpi. Cross Feed res=600dpi.
 Paper source = Paper source :  Tray 2
 Page layout = Paper=Paper: Letter size=8.5x11.0 INCH Orient=PORTRAIT leftMargin=54.0 rightMargin=54.0 topMargin=54.0 bottomMargin=54.0, name: paperSource, value: Paper source :  Tray 2]
PRINTING FINISHED
于 2017-09-06T18:56:44.123 回答