diff dwt/printing/Printer.d @ 0:380af2bdd8e5

Upload of whole dwt tree
author Jacob Carlborg <doob@me.com> <jacob.carlborg@gmail.com>
date Sat, 09 Aug 2008 17:00:02 +0200
parents
children 649b8e223d5a
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/dwt/printing/Printer.d	Sat Aug 09 17:00:02 2008 +0200
@@ -0,0 +1,609 @@
+/*******************************************************************************
+ * Copyright (c) 2000, 2007 IBM Corporation and others.
+ * All rights reserved. This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License v1.0
+ * which accompanies this distribution, and is available at
+ * http://www.eclipse.org/legal/epl-v10.html
+ *
+ * Contributors:
+ *     IBM Corporation - initial API and implementation
+ *******************************************************************************/
+module dwt.printing.Printer;
+
+import dwt.dwthelper.utils;
+
+import dwt.DWT;
+import dwt.DWTError;
+import dwt.DWTException;
+import dwt.graphics.Device;
+import dwt.graphics.DeviceData;
+import dwt.graphics.GCData;
+import dwt.graphics.Point;
+import dwt.graphics.Rectangle;
+import dwt.internal.cocoa.NSArray;
+import dwt.internal.cocoa.NSPrinter;
+import dwt.internal.cocoa.NSString;
+
+/**
+ * Instances of this class are used to print to a printer.
+ * Applications create a GC on a printer using <code>new GC(printer)</code>
+ * and then draw on the printer GC using the usual graphics calls.
+ * <p>
+ * A <code>Printer</code> object may be constructed by providing
+ * a <code>PrinterData</code> object which identifies the printer.
+ * A <code>PrintDialog</code> presents a print dialog to the user
+ * and returns an initialized instance of <code>PrinterData</code>.
+ * Alternatively, calling <code>new Printer()</code> will construct a
+ * printer object for the user's default printer.
+ * </p><p>
+ * Application code must explicitly invoke the <code>Printer.dispose()</code> 
+ * method to release the operating system resources managed by each instance
+ * when those instances are no longer required.
+ * </p>
+ *
+ * @see PrinterData
+ * @see PrintDialog
+ */
+public final class Printer extends Device {
+    PrinterData data;
+    NSPrinter printer;
+    bool inPage, isGCCreated;
+
+    static final String DRIVER = "Mac";
+    static final String PRINTER_DRIVER = "Printer";
+    static final String FILE_DRIVER = "File";
+    static final String PREVIEW_DRIVER = "Preview";
+    static final String FAX_DRIVER = "Fax";
+
+/**
+ * Returns an array of <code>PrinterData</code> objects
+ * representing all available printers.
+ *
+ * @return the list of available printers
+ */
+public static PrinterData[] getPrinterList() {
+    NSArray printers = NSPrinter.printerNames();
+    int count = printers.count();
+    PrinterData[] result = new PrinterData[count];
+    for (int i = 0; i < count; i++) {
+        NSString str = new NSString(printers.objectAtIndex(i));
+        char[] buffer = new char[str.length()];
+        str.getCharacters_(buffer);
+        result[i] = new PrinterData(DRIVER, new String(buffer));
+    }
+    return result;
+}
+
+/**
+ * Returns a <code>PrinterData</code> object representing
+ * the default printer or <code>null</code> if there is no 
+ * printer available on the System.
+ *
+ * @return the default printer data or null
+ * 
+ * @since 2.1
+ */
+public static PrinterData getDefaultPrinterData() {
+    //TODO - get default
+    PrinterData[] printers = getPrinterList();
+    if (printers.length > 0) return printers[0];
+    return null;
+}
+//static int packData(int handle, byte[] buffer, int offset) {
+//  int length = OS.GetHandleSize (handle);
+//  buffer[offset++] = (byte)((length & 0xFF) >> 0);
+//  buffer[offset++] = (byte)((length & 0xFF00) >> 8);
+//  buffer[offset++] = (byte)((length & 0xFF0000) >> 16);
+//  buffer[offset++] = (byte)((length & 0xFF000000) >> 24);
+//  int [] ptr = new int [1];
+//  OS.HLock(handle);
+//  OS.memmove(ptr, handle, 4);
+//  byte[] buffer1 = new byte[length];
+//  OS.memmove(buffer1, ptr [0], length);
+//  OS.HUnlock(handle);
+//  System.arraycopy(buffer1, 0, buffer, offset, length);
+//  return offset + length;
+//}
+//static int unpackData(int[] handle, byte[] buffer, int offset) {
+//  int length = 
+//      ((buffer[offset++] & 0xFF) << 0) |
+//      ((buffer[offset++] & 0xFF) << 8) |
+//      ((buffer[offset++] & 0xFF) << 16) |
+//      ((buffer[offset++] & 0xFF) << 24);
+//  handle[0] = OS.NewHandle(length);
+//  if (handle[0] is 0) DWT.error(DWT.ERROR_NO_HANDLES);
+//  int[] ptr = new int[1];
+//  OS.HLock(handle[0]);
+//  OS.memmove(ptr, handle[0], 4);
+//  byte[] buffer1 = new byte[length];
+//  System.arraycopy(buffer, offset, buffer1, 0, length);
+//  OS.memmove(ptr[0], buffer1, length);
+//  OS.HUnlock(handle[0]);
+//  return offset + length;
+//}
+
+/**
+ * Constructs a new printer representing the default printer.
+ * <p>
+ * You must dispose the printer when it is no longer required. 
+ * </p>
+ *
+ * @exception DWTError <ul>
+ *    <li>ERROR_NO_HANDLES - if there are no valid printers
+ * </ul>
+ *
+ * @see Device#dispose
+ */
+public Printer() {
+    this(null);
+}
+
+/**
+ * Constructs a new printer given a <code>PrinterData</code>
+ * object representing the desired printer.
+ * <p>
+ * You must dispose the printer when it is no longer required. 
+ * </p>
+ *
+ * @param data the printer data for the specified printer
+ *
+ * @exception IllegalArgumentException <ul>
+ *    <li>ERROR_INVALID_ARGUMENT - if the specified printer data does not represent a valid printer
+ * </ul>
+ * @exception DWTError <ul>
+ *    <li>ERROR_NO_HANDLES - if there are no valid printers
+ * </ul>
+ *
+ * @see Device#dispose
+ */
+public Printer(PrinterData data) {
+    super (checkNull(data));
+}
+
+/**
+ * Given a desired <em>client area</em> for the receiver
+ * (as described by the arguments), returns the bounding
+ * rectangle which would be required to produce that client
+ * area.
+ * <p>
+ * In other words, it returns a rectangle such that, if the
+ * receiver's bounds were set to that rectangle, the area
+ * of the receiver which is capable of displaying data
+ * (that is, not covered by the "trimmings") would be the
+ * rectangle described by the arguments (relative to the
+ * receiver's parent).
+ * </p><p>
+ * Note that there is no setBounds for a printer. This method
+ * is usually used by passing in the client area (the 'printable
+ * area') of the printer. It can also be useful to pass in 0, 0, 0, 0.
+ * </p>
+ * 
+ * @param x the desired x coordinate of the client area
+ * @param y the desired y coordinate of the client area
+ * @param width the desired width of the client area
+ * @param height the desired height of the client area
+ * @return the required bounds to produce the given client area
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ *
+ * @see #getBounds
+ * @see #getClientArea
+ */
+public Rectangle computeTrim(int x, int y, int width, int height) {
+    checkDevice();
+//  PMRect pageRect = new PMRect();
+//  PMRect paperRect = new PMRect();
+//  OS.PMGetAdjustedPageRect(pageFormat, pageRect);
+//  OS.PMGetAdjustedPaperRect(pageFormat, paperRect);
+//  return new Rectangle(x+(int)paperRect.left, y+(int)paperRect.top, width+(int)(paperRect.right-pageRect.right), height+(int)(paperRect.bottom-pageRect.bottom));
+    return null;
+}
+
+/**  
+ * Creates the printer handle.
+ * This method is called internally by the instance creation
+ * mechanism of the <code>Device</code> class.
+ * @param deviceData the device data
+ */
+protected void create(DeviceData deviceData) {
+    data = (PrinterData)deviceData;
+    
+    printer = NSPrinter.static_printerWithName_(NSString.stringWith(data.name));
+    printer.retain();
+    
+//  int[] buffer = new int[1];
+//  if (OS.PMCreateSession(buffer) !is OS.noErr) DWT.error(DWT.ERROR_NO_HANDLES);
+//  printSession = buffer[0];
+//  if (printSession is 0) DWT.error(DWT.ERROR_NO_HANDLES);
+//      
+//  if (data.otherData !is null) {
+//      /* Deserialize settings */
+//      int offset = 0;
+//      byte[] otherData = data.otherData;
+//      offset = unpackData(buffer, otherData, offset);
+//      int flatSettings = buffer[0];
+//      offset = unpackData(buffer, otherData, offset);
+//      int flatFormat = buffer[0];
+//      if (OS.PMUnflattenPrintSettings(flatSettings, buffer) !is OS.noErr) DWT.error(DWT.ERROR_NO_HANDLES);
+//      printSettings = buffer[0];
+//      if (printSettings is 0) DWT.error(DWT.ERROR_NO_HANDLES);
+//      if (OS.PMUnflattenPageFormat(flatFormat, buffer) !is OS.noErr) DWT.error(DWT.ERROR_NO_HANDLES);
+//      pageFormat = buffer[0];
+//      if (pageFormat is 0) DWT.error(DWT.ERROR_NO_HANDLES);
+//      OS.DisposeHandle(flatSettings);
+//      OS.DisposeHandle(flatFormat);
+//  } else {
+//      /* Create default settings */
+//      if (OS.PMCreatePrintSettings(buffer) !is OS.noErr) DWT.error(DWT.ERROR_NO_HANDLES);
+//      printSettings = buffer[0];
+//      if (printSettings is 0) DWT.error(DWT.ERROR_NO_HANDLES);
+//      OS.PMSessionDefaultPrintSettings(printSession, printSettings);
+//      if (OS.PMCreatePageFormat(buffer) !is OS.noErr) DWT.error(DWT.ERROR_NO_HANDLES);
+//      pageFormat = buffer[0];
+//      if (pageFormat is 0) DWT.error(DWT.ERROR_NO_HANDLES);
+//      OS.PMSessionDefaultPageFormat(printSession, pageFormat);
+//  }
+//  
+//  if (PREVIEW_DRIVER.equals(data.driver)) {
+//      OS.PMSessionSetDestination(printSession, printSettings, (short) OS.kPMDestinationPreview, 0, 0);
+//  }
+//  String name = data.name;
+//  char[] buffer1 = new char[name.length ()];
+//  name.getChars(0, buffer1.length, buffer1, 0);
+//  int ptr = OS.CFStringCreateWithCharacters(OS.kCFAllocatorDefault, buffer1, buffer1.length);
+//  if (ptr !is 0) {
+//      OS.PMSessionSetCurrentPrinter(printSession, ptr); 
+//      OS.CFRelease(ptr);
+//  }
+//  
+//  OS.PMSessionValidatePrintSettings(printSession, printSettings, null);
+//  OS.PMSessionValidatePageFormat(printSession, pageFormat, null); 
+//  
+//  int graphicsContextsArray = OS.CFArrayCreateMutable(OS.kCFAllocatorDefault, 1, 0);
+//  if (graphicsContextsArray !is 0) {
+//      OS.CFArrayAppendValue(graphicsContextsArray, OS.kPMGraphicsContextCoreGraphics());
+//      OS.PMSessionSetDocumentFormatGeneration(printSession, OS.kPMDocumentFormatPDF(), graphicsContextsArray, 0);
+//      OS.CFRelease(graphicsContextsArray);
+//  }
+}
+
+/**  
+ * Destroys the printer handle.
+ * This method is called internally by the dispose
+ * mechanism of the <code>Device</code> class.
+ */
+protected void destroy() {
+    if (printer !is null) printer.release();
+    printer = null;
+}
+
+/**  
+ * Invokes platform specific functionality to allocate a new GC handle.
+ * <p>
+ * <b>IMPORTANT:</b> This method is <em>not</em> part of the public
+ * API for <code>Printer</code>. It is marked public only so that it
+ * can be shared within the packages provided by DWT. It is not
+ * available on all platforms, and should never be called from
+ * application code.
+ * </p>
+ *
+ * @param data the platform specific GC data 
+ * @return the platform specific GC handle
+ */
+public int internal_new_GC(GCData data) {
+    if (isDisposed()) DWT.error(DWT.ERROR_GRAPHIC_DISPOSED);
+    setupNewPage();
+//  if (data !is null) {
+//      if (isGCCreated) DWT.error(DWT.ERROR_INVALID_ARGUMENT);
+//      data.device = this;
+//      data.background = getSystemColor(DWT.COLOR_WHITE).handle;
+//      data.foreground = getSystemColor(DWT.COLOR_BLACK).handle;
+//      data.font = getSystemFont ();
+//      PMRect paperRect= new PMRect();
+//      OS.PMGetAdjustedPaperRect(pageFormat, paperRect);
+//      Rect portRect = new Rect();
+//      portRect.left = (short)paperRect.left;
+//      portRect.right = (short)paperRect.right;
+//      portRect.top = (short)paperRect.top;
+//      portRect.bottom = (short)paperRect.bottom;
+//      data.portRect = portRect;
+//      isGCCreated = true;
+//  }
+//  return context;
+    return 0;
+}
+
+protected void init () {
+    super.init();
+}
+
+/**  
+ * Invokes platform specific functionality to dispose a GC handle.
+ * <p>
+ * <b>IMPORTANT:</b> This method is <em>not</em> part of the public
+ * API for <code>Printer</code>. It is marked public only so that it
+ * can be shared within the packages provided by DWT. It is not
+ * available on all platforms, and should never be called from
+ * application code.
+ * </p>
+ *
+ * @param hDC the platform specific GC handle
+ * @param data the platform specific GC data 
+ */
+public void internal_dispose_GC(int context, GCData data) {
+    if (data !is null) isGCCreated = false;
+}
+
+/**  
+ * Releases any internal state prior to destroying this printer.
+ * This method is called internally by the dispose
+ * mechanism of the <code>Device</code> class.
+ */
+protected void release () {
+    super.release();
+}
+
+/**
+ * Starts a print job and returns true if the job started successfully
+ * and false otherwise.
+ * <p>
+ * This must be the first method called to initiate a print job,
+ * followed by any number of startPage/endPage calls, followed by
+ * endJob. Calling startPage, endPage, or endJob before startJob
+ * will result in undefined behavior.
+ * </p>
+ * 
+ * @param jobName the name of the print job to start
+ * @return true if the job started successfully and false otherwise.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ *
+ * @see #startPage
+ * @see #endPage
+ * @see #endJob
+ */
+public bool startJob(String jobName) {
+    checkDevice();
+//  if (jobName !is null && jobName.length() !is 0) {
+//      char[] buffer = new char[jobName.length ()];
+//      jobName.getChars(0, buffer.length, buffer, 0);
+//      int ptr = OS.CFStringCreateWithCharacters(OS.kCFAllocatorDefault, buffer, buffer.length);
+//      if (ptr !is 0) {
+//          OS.PMSetJobNameCFString(printSettings, ptr); 
+//          OS.CFRelease (ptr);
+//      }
+//  }
+//  return OS.PMSessionBeginDocumentNoDialog(printSession, printSettings, pageFormat) is OS.noErr;
+    return false;
+}
+
+/**
+ * Ends the current print job.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ *
+ * @see #startJob
+ * @see #startPage
+ * @see #endPage
+ */
+public void endJob() {
+    checkDevice();
+//  if (inPage) {
+//      OS.PMSessionEndPageNoDialog(printSession);
+//      inPage = false;
+//  }
+//  OS.PMSessionEndDocumentNoDialog(printSession);
+//  context = 0;
+}
+
+/**
+ * Cancels a print job in progress. 
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ */
+public void cancelJob() {
+    checkDevice();
+//  OS.PMSessionSetError(printSession, OS.kPMCancel);
+//  if (inPage) {
+//      OS.PMSessionEndPageNoDialog(printSession);
+//      inPage = false;
+//  }
+//  OS.PMSessionEndDocumentNoDialog(printSession);
+//  context = 0;
+}
+
+static DeviceData checkNull (PrinterData data) {
+    if (data is null) data = new PrinterData();
+    if (data.driver is null || data.name is null) {
+        PrinterData defaultPrinter = getDefaultPrinterData();
+        if (defaultPrinter is null) DWT.error(DWT.ERROR_NO_HANDLES);
+        data.driver = defaultPrinter.driver;
+        data.name = defaultPrinter.name;        
+    }
+    return data;
+}
+
+/**
+ * Starts a page and returns true if the page started successfully
+ * and false otherwise.
+ * <p>
+ * After calling startJob, this method may be called any number of times
+ * along with a matching endPage.
+ * </p>
+ * 
+ * @return true if the page started successfully and false otherwise.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ *
+ * @see #endPage
+ * @see #startJob
+ * @see #endJob
+ */
+public bool startPage() {
+    checkDevice();
+//  if (OS.PMSessionError(printSession) !is OS.noErr) return false;
+//  setupNewPage();
+//  return context !is 0;
+    return false;
+}
+
+/**
+ * Ends the current page.
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ *
+ * @see #startPage
+ * @see #startJob
+ * @see #endJob
+ */
+public void endPage() {
+    checkDevice();
+//  if (inPage) {
+//      OS.PMSessionEndPageNoDialog(printSession);
+//      inPage = false;
+//  }
+}
+
+/**
+ * Returns a point whose x coordinate is the horizontal
+ * dots per inch of the printer, and whose y coordinate
+ * is the vertical dots per inch of the printer.
+ *
+ * @return the horizontal and vertical DPI
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ */
+public Point getDPI() {
+    checkDevice();
+//  PMResolution resolution = new PMResolution();
+//  OS.PMGetResolution(pageFormat, resolution);
+//  return new Point((int)resolution.hRes, (int)resolution.vRes);
+    return null;
+}
+
+/**
+ * Returns a rectangle describing the receiver's size and location.
+ * For a printer, this is the size of a physical page, in pixels.
+ *
+ * @return the bounding rectangle
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ *
+ * @see #getClientArea
+ * @see #computeTrim
+ */
+public Rectangle getBounds() {
+    checkDevice();
+//  PMRect paperRect = new PMRect();
+//  OS.PMGetAdjustedPaperRect(pageFormat, paperRect);
+//  return new Rectangle(0, 0, (int)(paperRect.right-paperRect.left), (int)(paperRect.bottom-paperRect.top));
+    return null;
+}
+
+/**
+ * Returns a rectangle which describes the area of the
+ * receiver which is capable of displaying data.
+ * For a printer, this is the size of the printable area
+ * of a page, in pixels.
+ * 
+ * @return the client area
+ *
+ * @exception DWTException <ul>
+ *    <li>ERROR_DEVICE_DISPOSED - if the receiver has been disposed</li>
+ * </ul>
+ *
+ * @see #getBounds
+ * @see #computeTrim
+ */
+public Rectangle getClientArea() {
+    checkDevice();
+//  PMRect pageRect = new PMRect();
+//  OS.PMGetAdjustedPageRect(pageFormat, pageRect);
+//  return new Rectangle(0, 0, (int)(pageRect.right-pageRect.left), (int)(pageRect.bottom-pageRect.top));
+    return null;
+}
+
+/**
+ * Returns a <code>PrinterData</code> object representing the
+ * target printer for this print job.
+ * 
+ * @return a PrinterData object describing the receiver
+ */
+public PrinterData getPrinterData() {
+    checkDevice();
+    return data;
+}
+
+/**
+ * On the Mac the core graphics context for printing is only valid between PMSessionBeginPage and PMSessionEndPage,
+ * so printing code has to retrieve and initializes a graphic context for every page like this:
+ * 
+ * <pre>
+ * PMSessionBeginDocument
+ *    PMSessionBeginPage
+ *       PMSessionGetGraphicsContext
+ *       // ... use context
+ *    PMSessionEndPage
+ * PMSessionEndDocument
+ * </pre>
+ * 
+ * In DWT it is OK to create a GC once between startJob / endJob and use it for all pages in between:
+ * 
+ * <pre>
+ * startJob(...);
+ *    GC gc= new GC(printer);
+ *    startPage();
+ *       // ... use gc
+ *    endPage();
+ *    gc.dispose();
+ * endJob();
+ * </pre>
+ * 
+ * The solution to resolve this difference is to rely on the fact that Mac OS X returns the same but
+ * reinitialized graphics context for every page. So we only have to account for the fact that DWT assumes
+ * that the graphics context keeps it settings across a page break when it actually does not.
+ * So we have to copy some settings that exist in the CGC before a PMSessionEndPage to the CGC after a PMSessionBeginPage.
+ * <p>
+ * In addition to this we have to cope with the situation that in DWT we can create a GC before a call to
+ * PMSessionBeginPage. For this we decouple the call to PMSessionBeginPage from
+ * DWT's method startPage as follows: if a new GC is created before a call to startPage, internal_new_GC
+ * does the PMSessionBeginPage and the next following startPage does nothing.
+ * </p>
+ */
+void setupNewPage() {
+//  if (!inPage) {
+//      inPage= true;
+//      OS.PMSessionBeginPageNoDialog(printSession, pageFormat, null);
+//      int[] buffer = new int[1];
+//      OS.PMSessionGetGraphicsContext(printSession, 0, buffer);
+//      if (context is 0) {
+//          context = buffer[0];
+//      } else {
+//          if (context !is buffer[0]) DWT.error(DWT.ERROR_UNSPECIFIED);
+//      }
+//      PMRect paperRect= new PMRect();
+//      OS.PMGetAdjustedPaperRect(pageFormat, paperRect);
+//      OS.CGContextScaleCTM(context, 1, -1);
+//      OS.CGContextTranslateCTM(context, 0, -(float)(paperRect.bottom-paperRect.top));
+//      OS.CGContextSetStrokeColorSpace(context, colorspace);
+//      OS.CGContextSetFillColorSpace(context, colorspace);
+//  }
+}
+
+}