javawinapiprintingprinter-control-language

Why does getSupportedAttributeValues return Paper Types


The following code snippet should return the Media Trays available to a printer.

However, with some drivers, specifically Ricoh PCL6 Driver for Universal Print and HP Universal Printing PCL 6, in addition to Printer Trays, these drivers also list paper types such as Recycled, Thick, Matte, etc.

From what I can tell, OpenJDK is properly using DC_BINNAMES when calling DeviceCapabilities. OpenJDK doesn't even seem to use DC_MEDIATYPENAMES at all in the source code, so I wouldn't expect e.g. Purple Paper to even be a queryable property, yet it lists when querying trays from the Ricoh driver.

So what's wrong? Are these PCL 6 drivers just bugged? Is DeviceCapabilities at fault? Or does the bug live in OpenJDK?

import javax.print.DocFlavor;
import javax.print.PrintService;
import javax.print.PrintServiceLookup;
import javax.print.attribute.standard.Media;

public class TrayListing {
    public static void main(String ... args) {
        String printerName = "HP LaserJet ...";  // TODO: change this to the actual printer name
        PrintService[] allPrinters = PrintServiceLookup.lookupPrintServices(null, null);
        for(PrintService ps : allPrinters) {
            if(ps.getName().equalsIgnoreCase(printerName)) {
                // loop over media trays
                System.out.println("\n\nFound MediaTray:");

                // Some HP, Ricoh printers/drivers list items that aren't printer trays, such as paper types
                for(Media m : (Media[])ps.getSupportedAttributeValues(Media.class, DocFlavor.SERVICE_FORMATTED.PAGEABLE, null)) {
                    if (m instanceof javax.print.attribute.standard.MediaTray) {
                        System.out.println("- " + m + " (" + m.getClass().getName() + ")");
                    }
                }
            }
        }
    }
}

Additional keywords:

PCL XL Feature Reference


Solution

  • The drivers are bugged. Workarounds exists, but they are complex.

    The short:

    The long:

    Some drivers such as HP expose the printer trays properly in other areas, for example:

    ... however this isn't true for drivers such as Ricoh.

    After examining a lot of drivers (HP, Ricoh, Xerox, Konica, etc) I've isolated the issue to the following:

    In cases with both vendors, the DC_BINS value is always > 1000, which is partially explained in the PCL6 MediaSource specification, quoting:

    "... External input trays 1 through 248 are selected by substituting the values 8 through 255 for the enumerated values. Example, 8 = first external input tray, 9 = second external input tray, etc. ..."

    Although there's nothing about the 1,000 specifically, and vendors such as Xerox use values over 7000 without the bug. That said, with the problematic vendors, what is observed is that when values are > 1,000 they tend to be actually valid MediaType values (NOT MediaSource values), but incremented by 1,000.

    Oddly enough, this is very limited to HP and Ricoh and does not apply to other PCL drivers. For example:

    So to "fix" this issue, I wrote a series of custom parsing to find out the driver vendor and the tray id.

    To locate the trayId from Java:

    // Get default printer
    PrintService ps = PrintServiceLookup.lookupDefaultPrintService();
    
    for(Media mediaTray : (Media[])ps.getSupportedAttributeValues(Media.class, null, null)) {
       // Warning: Reflective operation on Windows-only class
       Method method = ps.getClass().getMethod("findTrayID", MediaTray.class);
       Object trayId = method.invoke(ps, new Object[]{mediaTray});
       System.out.println(trayId);
    }
    

    To obtain the driver vendor from Java:

    Calculating the driver name is some nuanced code, but can be used reliably. For a comprehensive code example, see here. Take special note of the special character replacement in the keys/values.

    Once the driver is found, matching the terms Ricoh, HP, PCL6 and PCL 6 using a regular expression will allow filtering trayIDs over 1,000.

    By combining the above techniques of trayId numbering, vendor matching and the keywords "PCL6" and "PCL 6", the bad trays can be filtered out programatically.