Mercurial > projects > dwt-mac
view dwt/printing/Printer.d @ 156:969e7de37c3d default tip
Fixes to get dwt to work with dmd and ldc
author | Jacob Carlborg <doob@me.com> |
---|---|
date | Wed, 08 Jul 2009 21:56:44 +0200 |
parents | 3d4579727e0e |
children |
line wrap: on
line source
/******************************************************************************* * Copyright (c) 2000, 2008 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 * * Port to the D programming language: * Jacob Carlborg <doob@me.com> *******************************************************************************/ 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.NSAffineTransform; import dwt.internal.cocoa.NSArray; import dwt.internal.cocoa.NSBezierPath; import dwt.internal.cocoa.NSData; import dwt.internal.cocoa.NSGraphicsContext; import dwt.internal.cocoa.NSKeyedUnarchiver; import dwt.internal.cocoa.NSPoint; import dwt.internal.cocoa.NSPrintInfo; import dwt.internal.cocoa.NSPrintOperation; import dwt.internal.cocoa.NSPrinter; import dwt.internal.cocoa.NSRect; import dwt.internal.cocoa.NSSize; import dwt.internal.cocoa.NSString; import dwt.internal.cocoa.NSView; import dwt.internal.cocoa.NSWindow; import dwt.internal.cocoa.OS; import Carbon = dwt.internal.c.Carbon; import dwt.internal.objc.cocoa.Cocoa; import objc = dwt.internal.objc.runtime; import dwt.printing.PrinterData; /** * 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 * @see <a href="http://www.eclipse.org/swt/snippets/#printing">Printing snippets</a> * @see <a href="http://www.eclipse.org/swt/">Sample code and further information</a> */ public final class Printer : Device { PrinterData data; NSPrinter printer; NSPrintInfo printInfo; NSPrintOperation operation; NSView view; NSWindow window; bool isGCCreated; static const String DRIVER = "Mac"; /** * 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(); NSUInteger count = printers.count(); PrinterData[] result = new PrinterData[count]; for (NSUInteger i = 0; i < count; i++) { NSString str = new NSString(printers.objectAtIndex(i)); result[i] = new PrinterData(DRIVER, str.getString()); } 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() { NSPrinter printer = NSPrintInfo.defaultPrinter(); if (printer is null) return null; NSString str = printer.name(); return new PrinterData(DRIVER, str.getString()); } /** * 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 this() { 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 this(PrinterData data) { super (checkNull(data)); } /** * Given a <em>client area</em> (as described by the arguments), * returns a rectangle, relative to the client area's coordinates, * that is the client area expanded by the printer's trim (or minimum margins). * <p> * Most printers have a minimum margin on each edge of the paper where the * printer device is unable to print. This margin is known as the "trim." * This method can be used to calculate the printer's minimum margins * by passing in a client area of 0, 0, 0, 0 and then using the resulting * x and y coordinates (which will be <= 0) to determine the minimum margins * for the top and left edges of the paper, and the resulting width and height * (offset by the resulting x and y) to determine the minimum margins for the * bottom and right edges of the paper, as follows: * <ul> * <li>The left trim width is -x pixels</li> * <li>The top trim height is -y pixels</li> * <li>The right trim width is (x + width) pixels</li> * <li>The bottom trim height is (y + height) pixels</li> * </ul> * </p> * * @param x the x coordinate of the client area * @param y the y coordinate of the client area * @param width the width of the client area * @param height the height of the client area * @return a rectangle, relative to the client area's coordinates, that is * the client area expanded by the printer's trim (or minimum margins) * * @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(); NSSize paperSize = printInfo.paperSize(); NSRect bounds = printInfo.imageablePageBounds(); Point dpi = getDPI (), screenDPI = getIndependentDPI(); x -= (bounds.x * dpi.x / screenDPI.x); y -= (bounds.y * dpi.y / screenDPI.y); width += (paperSize.width - bounds.width) * dpi.x / screenDPI.x; height += (paperSize.height - bounds.height) * dpi.y / screenDPI.y; return new Rectangle(x, y, width, height); } /** * 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 = cast(PrinterData)deviceData; if (data.otherData !is null) { NSData nsData = NSData.dataWithBytes(data.otherData.ptr, data.otherData.length); printInfo = new NSPrintInfo(NSKeyedUnarchiver.unarchiveObjectWithData(nsData).id); } else { printInfo = NSPrintInfo.sharedPrintInfo(); } printInfo.retain(); printer = NSPrinter.printerWithName(NSString.stringWith(data.name)); if (printer !is null) { printer.retain(); printInfo.setPrinter(printer); } /* * Bug in Cocoa. For some reason, the output still goes to the printer when * the user chooses the preview button. The fix is to reset the job disposition. */ NSString job = printInfo.jobDisposition(); if (job.isEqual(new NSString(OS.NSPrintPreviewJob_))) { printInfo.setJobDisposition(job); } NSRect rect = NSRect(); window = cast(NSWindow)(new NSWindow()).alloc(); window.initWithContentRect(rect, OS.NSBorderlessWindowMask, OS.NSBackingStoreBuffered, false); view = cast(NSView)(new NSView()).alloc(); view.initWithFrame(rect); window.setContentView(view); operation = NSPrintOperation.printOperationWithView(view, printInfo); operation.retain(); operation.setShowsPrintPanel(false); operation.setShowsProgressPanel(false); } /** * 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(); if (printInfo !is null) printInfo.release(); if (view !is null) view.release(); if (window !is null) window.release(); if (operation !is null) operation.release(); printer = null; printInfo = null; view = null; operation = 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 objc.id internal_new_GC(GCData data) { if (isDisposed()) DWT.error(DWT.ERROR_GRAPHIC_DISPOSED); 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 (); data.sizeStruct = printInfo.paperSize(); data.size = &data.sizeStruct; isGCCreated = true; } return operation.context().id; } 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(objc.id 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) { operation.setJobTitle(NSString.stringWith(jobName)); } printInfo.setUpPrintOperationDefaultValues(); NSPrintOperation.setCurrentOperation(operation); NSGraphicsContext context = operation.createContext(); if (context !is null) { view.beginDocument(); return true; } 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(); view.endDocument(); operation.deliverResult(); operation.destroyContext(); operation.cleanUpOperation(); } /** * 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(); operation.destroyContext(); operation.cleanUpOperation(); } 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(); NSSize paperSize = printInfo.paperSize(); NSRect rect = NSRect(); rect.width = paperSize.width; rect.height = paperSize.height; view.beginPageInRect(rect, NSPoint()); NSRect imageBounds = printInfo.imageablePageBounds(); NSBezierPath.bezierPathWithRect(imageBounds).setClip(); NSAffineTransform transform = NSAffineTransform.transform(); transform.translateXBy(imageBounds.x, rect.height - imageBounds.y); transform.scaleXBy(1, -1); Point dpi = getDPI (), screenDPI = getIndependentDPI(); transform.scaleXBy(screenDPI.x / cast(Carbon.CGFloat)dpi.x, screenDPI.y / cast(Carbon.CGFloat)dpi.y); transform.concat(); return true; } /** * 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(); view.endPage(); } /** * 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(); //TODO get output resolution return getIndependentDPI(); } Point getIndependentDPI() { return super.getDPI(); } /** * Returns a rectangle describing the receiver's size and location. * <p> * For a printer, this is the size of the physical page, in pixels. * </p> * * @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(); NSSize size = printInfo.paperSize(); Point dpi = getDPI (), screenDPI = getIndependentDPI(); return new Rectangle (0, 0, cast(int)(size.width * dpi.x / screenDPI.x), cast(int)(size.height * dpi.y / screenDPI.y)); } /** * Returns a rectangle which describes the area of the * receiver which is capable of displaying data. * <p> * For a printer, this is the size of the printable area * of the page, in pixels. * </p> * * @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(); NSRect rect = printInfo.imageablePageBounds(); Point dpi = getDPI (), screenDPI = getIndependentDPI(); return new Rectangle(0, 0, cast(int)(rect.width * dpi.x / screenDPI.x), cast(int)(rect.height * dpi.y / screenDPI.y)); } /** * 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; } }