| /* |
| * Copyright (c) 2007, Oracle and/or its affiliates. All rights reserved. |
| * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
| * |
| * This code is free software; you can redistribute it and/or modify it |
| * under the terms of the GNU General Public License version 2 only, as |
| * published by the Free Software Foundation. |
| * |
| * This code is distributed in the hope that it will be useful, but WITHOUT |
| * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
| * version 2 for more details (a copy is included in the LICENSE file that |
| * accompanied this code). |
| * |
| * You should have received a copy of the GNU General Public License version |
| * 2 along with this work; if not, write to the Free Software Foundation, |
| * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
| * |
| * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
| * or visit www.oracle.com if you need additional information or have any |
| * questions. |
| */ |
| |
| /** |
| * @test |
| * @bug 6531728 |
| * @summary Test printing of images which need to have src area clipped |
| * @run main/manual=yesno/timeout=900 ClippedImages |
| */ |
| |
| import java.io.*; |
| import java.awt.*; |
| import java.awt.geom.*; |
| import java.awt.event.*; |
| import java.awt.print.*; |
| import java.awt.image.BufferedImage; |
| import javax.print.*; |
| import javax.print.attribute.*; |
| |
| public class ClippedImages extends Frame implements ActionListener { |
| |
| private ClippedImageCanvas c; |
| |
| public static void main(String args[]) { |
| |
| ClippedImages f = new ClippedImages(); |
| f.setVisible(true); |
| } |
| |
| public ClippedImages() { |
| super("Clipped Src Area Image Printing Test"); |
| c = new ClippedImageCanvas(); |
| add("Center", c); |
| |
| Button paintButton = new Button("Toggle Contents"); |
| paintButton.addActionListener(this); |
| |
| Button printThisButton = new Button("Print This"); |
| printThisButton.addActionListener(this); |
| |
| Button printAllButton = new Button("Print All"); |
| printAllButton.addActionListener(this); |
| |
| Panel p = new Panel(); |
| p.add(paintButton); |
| p.add(printThisButton); |
| p.add(printAllButton); |
| add("South", p); |
| add("North", getInstructions()); |
| addWindowListener(new WindowAdapter() { |
| public void windowClosing(WindowEvent e) { |
| System.exit(0); |
| } |
| }); |
| |
| pack(); |
| } |
| |
| private TextArea getInstructions() { |
| TextArea ta = new TextArea(18, 60); |
| ta.setFont(new Font("Dialog", Font.PLAIN, 11)); |
| ta.setText |
| ("This is a manual test as it requires that you compare "+ |
| "the on-screen rendering with the printed output.\n"+ |
| "Select the 'Print All' button to print out the test\n"+ |
| "It will generate 4 sides of content: as it will print "+ |
| "each of 2 sets of transformed images in portrait, \n"+ |
| "and landscape orientations. \n"+ |
| "The sets of images are in turn made up\n"+ |
| "of two similar sets of pages: one is 'random' images,\n "+ |
| " the other is 16 squares.\n"+ |
| "Use the 'Toggle Contents' button to view the screen rendering\n"+ |
| "For each page compare the printed content to the same\n"+ |
| "on-screen one taking careful note of\n"+ |
| "a) the positions of the red/blue circles on the corners\n"+ |
| "b) that numerical text on the image is displayed similarly\n"+ |
| "e) that the green quadrilaterals match on-screen\n"+ |
| "f) that the rendering is clipped at the default (typically 1 inch) "+ |
| "margins of the page.\n"+ |
| "The test PASSES if the onscreen and printed rendering match"); |
| return ta; |
| } |
| |
| public void actionPerformed(ActionEvent e) { |
| |
| if (e.getActionCommand().equals("Print This")) { |
| printOne(); |
| } else if (e.getActionCommand().equals("Print All")) { |
| printAll(); |
| } else if (e.getActionCommand().equals("Toggle Contents")) { |
| c.toggleContents(); |
| c.repaint(); |
| } |
| } |
| |
| private void printOne() { |
| PrinterJob pj = PrinterJob.getPrinterJob(); |
| |
| PrintRequestAttributeSet attrs = new HashPrintRequestAttributeSet(); |
| if (pj != null && (false||pj.printDialog(attrs))) { |
| c.setPrinterJob(pj, false); |
| pj.setPrintable(c); |
| try { |
| pj.print(attrs); |
| } catch (PrinterException pe) { |
| pe.printStackTrace(); |
| throw new RuntimeException("Exception whilst printing."); |
| } finally { |
| System.out.println("PRINT RETURNED OK."); |
| } |
| } |
| } |
| |
| private void printAll() { |
| PrinterJob pj = PrinterJob.getPrinterJob(); |
| PrintRequestAttributeSet attrs = new HashPrintRequestAttributeSet(); |
| if (pj != null && (false||pj.printDialog(attrs))) { |
| c.setPrinterJob(pj, true); |
| pj.setPageable(c); |
| try { |
| pj.print(attrs); |
| } catch (PrinterException pe) { |
| pe.printStackTrace(); |
| throw new RuntimeException("Exception whilst printing."); |
| } finally { |
| System.out.println("PRINT RETURNED OK."); |
| } |
| } |
| } |
| } |
| |
| class ClippedImageCanvas extends Component implements Printable, Pageable { |
| |
| BufferedImage img = null; |
| int sw=50, sh=50; |
| |
| ClippedImageCanvas() { |
| img = new BufferedImage(sw, sh, BufferedImage.TYPE_INT_RGB); |
| Graphics2D g2d = img.createGraphics(); |
| g2d.setColor(Color.red); |
| g2d.fillRect(0 ,0, sw, sh); |
| g2d.setColor(Color.black); |
| int cnt = 0; |
| Font font = new Font("Serif", Font.PLAIN, 11); |
| g2d.setFont(font); |
| FontMetrics fm = g2d.getFontMetrics(); |
| for (int y=12;y<sh;y+=12) { |
| int x = 0; |
| while (x < sw) { |
| String s = (new Integer(++cnt)).toString(); |
| g2d.drawString(s, x, y); |
| x+= fm.stringWidth(s); |
| } |
| } |
| } |
| |
| private boolean paintSquares = true; |
| void toggleContents() { |
| paintSquares = !paintSquares; |
| } |
| |
| public int getNumberOfPages() { |
| if (pageable) { |
| return 4; |
| } else { |
| return 1; |
| } |
| } |
| |
| boolean pageable = false; |
| PrinterJob myPrinterJob; |
| void setPrinterJob(PrinterJob job, boolean pageable) { |
| this.myPrinterJob = job; |
| this.pageable = pageable; |
| } |
| |
| public PageFormat getPageFormat(int pageIndex) |
| throws IndexOutOfBoundsException { |
| |
| if (pageIndex < 0 || pageIndex >= getNumberOfPages()) { |
| throw new IndexOutOfBoundsException(); |
| } |
| |
| PageFormat pf = myPrinterJob.defaultPage(); |
| switch (pageIndex % 2) { |
| |
| case 0 : |
| pf.setOrientation(PageFormat.PORTRAIT); |
| break; |
| |
| case 1: |
| pf.setOrientation(PageFormat.LANDSCAPE); |
| break; |
| } |
| return pf; |
| } |
| |
| String getOrientStr(PageFormat pf) { |
| if (pf.getOrientation() == PageFormat.PORTRAIT) { |
| return "Portrait Orientation, "; |
| } else { |
| return "Landscape Orientation,"; |
| } |
| } |
| |
| public Printable getPrintable(int pageIndex) |
| throws IndexOutOfBoundsException { |
| |
| if (pageIndex < 0 || pageIndex >= getNumberOfPages()) { |
| throw new IndexOutOfBoundsException(); |
| } |
| if (pageIndex < 2) { |
| paintSquares = true; |
| } else { |
| paintSquares = false; |
| } |
| return this; |
| } |
| |
| public int print(Graphics g, PageFormat pgFmt, int pgIndex) { |
| |
| if (pgIndex > getNumberOfPages()-1) { |
| return Printable.NO_SUCH_PAGE; |
| } |
| Graphics2D g2d = (Graphics2D)g; |
| g2d.translate(pgFmt.getImageableX(), pgFmt.getImageableY()); |
| g.drawString(getOrientStr(pgFmt), 0, 12); |
| paint(g2d); |
| return Printable.PAGE_EXISTS; |
| } |
| |
| private void drawImage(Graphics g, |
| int dx1, int dy1, int dx2, int dy2, |
| int sx1, int sy1, int sx2, int sy2) { |
| |
| int rx = (dx1 < dx2) ? dx1 : dx2; |
| int ry = (dy1 < dy2) ? dy1 : dy2; |
| int rw = dx2-dx1; |
| if (rw < 0) rw = -rw; |
| int rh = dy2-dy1; |
| if (rh < 0) rh = -rh; |
| |
| g.setColor(Color.green); |
| g.drawRect(rx-1 ,ry-1, rw+1, rh+1); |
| g.drawImage(img, dx1, dy1, dx2, dy2, sx1, sy1, sx2, sy2, null); |
| g.setColor(Color.blue); |
| int r=5; |
| g.drawOval(dx1-r, dy1-r, 2*r, 2*r); |
| g.setColor(Color.red); |
| g.drawOval(dx2-r, dy2-r, 2*r, 2*r); |
| } |
| |
| private AffineTransform savedTx = null; |
| |
| private void saveTx(Graphics2D g2d) { |
| savedTx = g2d.getTransform(); |
| } |
| |
| private void restoreTx(Graphics2D g2d) { |
| g2d.setTransform(savedTx); |
| } |
| |
| public void paint(Graphics g) { |
| Dimension size = getSize(); |
| g.setColor(Color.black); |
| for (int p=0;p<size.width;p+=20) { |
| g.drawLine(p, 0, p, size.height); |
| } |
| for (int p=0;p<size.height;p+=20) { |
| g.drawLine(0, p, size.width, p); |
| } |
| if (paintSquares) { |
| paintSquares(g); |
| } else { |
| paintRandom(g); |
| } |
| } |
| |
| private void paintRandom(Graphics g) { |
| |
| int dx, dy, dw, dh; |
| |
| Graphics2D g2d = (Graphics2D)g; |
| g.setColor(Color.black); |
| |
| saveTx(g2d); |
| int sx = -20, sy=-20; |
| |
| dx=300; dy=10; dw=50; dh=50; |
| |
| drawImage(g, dx, dy, dx+dw, dy+dh ,sx,sy,1,1); |
| |
| dx=20; dy=20; dw=400; dh=80; |
| g2d.shear(0.0, Math.PI/20); |
| drawImage(g, dx, dy, dx+dw, dy+dh, sx, sy, dw/2, dh/2); |
| |
| dx=125; dy=40; |
| restoreTx(g2d); |
| |
| g2d.rotate(Math.PI/4); |
| drawImage(g, dx, dy, dx+dw, dy+dh, sx, sy, dw/2, dh/2); |
| |
| restoreTx(g2d); |
| |
| dx=290; dy=180; dw=20; dh=20; |
| drawImage(g, dx, dy, dx+dw*10, dy+dh*10, 30, sy, dw, dh); |
| g2d.scale(-1, -1); |
| dx=-280; dy=-200; |
| drawImage(g, dx, dy, dx+dw*2, dy+dh*2, 30, sy, dw, dh); |
| |
| restoreTx(g2d); |
| |
| g2d.scale(1, -1); |
| dx=430; dy=-150; |
| drawImage(g, dx, dy, dx+dw*5, dy+dh*2, 30, sy, dw, dh); |
| |
| restoreTx(g2d); |
| |
| dx = 10; dy = 290; dw = 200; dh = 200; |
| drawImage(g, dx, dy, dx+dw, dy+dh, sx, sy, sw, sh); |
| |
| dx = 0; dy = 400; dw=-30; dh=-50; |
| drawImage(g, dx, dy, dx-dw, dy-dh, dx, dy, dx-dw, dy-dh); |
| } |
| |
| private void paintSquares(Graphics g) { |
| |
| /* drawImage is required to handle mapping sx1,sy1 -> dx1,dy1 and |
| * sx2,sy2 -> dx2,dy2 which may imply flips and scales. |
| * To test this we need to test all combinations of these parameters |
| * with drawImage. |
| * If we have a source rectangle with vertices, sA, sB, sC, sD |
| * there are 4 combinations : sA+sD, sD+sA, sB+sC, sC+sB. |
| * Similarly for the destination with vertices, dA, dB, dC, dD |
| * there are 4 combinations : dA+dD, dD+dA, dB+dC, dC+dB. |
| * Thus we need 16 calls to test all combinations. |
| * Note that we set the source area coordinates (x and y -20->80) |
| * to be beyond the image size (50x50) so clipping is always needed. |
| */ |
| int sxa = -20, sya = -20; |
| int sxb = 80, syb = -20; |
| int sxc = -20, syc = 80; |
| int sxd = 80, syd = 80; |
| |
| int dxa = 0, dya = 0; |
| int dxb = 80, dyb = 0; |
| int dxc = 0, dyc = 80; |
| int dxd = 80, dyd = 80; |
| |
| int incX = 100; |
| int incY = 100; |
| |
| g.translate(20, 20); |
| |
| /* sA + sD -> dA + dD - the normal untransformed case */ |
| drawImage(g, dxa, dya, dxd, dyd, sxa, sya, sxd, syd); |
| g.translate(incX, 0); |
| |
| /* sD + sA -> dA + dD */ |
| drawImage(g, dxa, dya, dxd, dyd, sxd, syd, sxa, sya); |
| g.translate(incX, 0); |
| |
| /* sB + sC -> dA + dD */ |
| drawImage(g, dxa, dya, dxd, dyd, sxb, syb, sxc, syc); |
| g.translate(incX, 0); |
| |
| /* sC + sB -> dA + dD */ |
| drawImage(g, dxa, dya, dxd, dyd, sxc, syc, sxb, syb); |
| |
| g.translate(-3*incX, incY); |
| /******/ |
| |
| /* sA + sD -> dD + dA */ |
| drawImage(g, dxd, dyd, dxa, dya, sxa, sya, sxd, syd); |
| g.translate(incX, 0); |
| |
| /* sD + sA -> dD + dA */ |
| drawImage(g, dxd, dyd, dxa, dya, sxd, syd, sxa, sya); |
| g.translate(incX, 0); |
| |
| /* sB + sC -> dD + dA */ |
| drawImage(g, dxd, dyd, dxa, dya, sxb, syb, sxc, syc); |
| g.translate(incX, 0); |
| |
| /* sC + sB -> dD + dA */ |
| drawImage(g, dxd, dyd, dxa, dya, sxc, syc, sxb, syb); |
| |
| g.translate(-3*incX, incY); |
| /******/ |
| |
| /* sA + sD -> dB + dC */ |
| drawImage(g, dxb, dyb, dxc, dyc, sxa, sya, sxd, syd); |
| g.translate(incX, 0); |
| |
| /* sD + sA -> dB + dC */ |
| drawImage(g, dxb, dyb, dxc, dyc, sxd, syd, sxa, sya); |
| g.translate(incX, 0); |
| |
| /* sB + sC -> dB + dC */ |
| drawImage(g, dxb, dyb, dxc, dyc, sxb, syb, sxc, syc); |
| g.translate(incX, 0); |
| |
| /* sC + sB -> dB + dC */ |
| drawImage(g, dxb, dyb, dxc, dyc, sxc, syc, sxb, syb); |
| |
| g.translate(-3*incX, incY); |
| /******/ |
| |
| |
| /* sA + sD -> dC + dB */ |
| drawImage(g, dxc, dyc, dxb, dyb, sxa, sya, sxd, syd); |
| g.translate(incX, 0); |
| |
| /* sD + sA -> dC + dB */ |
| drawImage(g, dxc, dyc, dxb, dyb, sxd, syd, sxa, sya); |
| g.translate(incX, 0); |
| |
| /* sB + sC -> dC + dB */ |
| drawImage(g, dxc, dyc, dxb, dyb, sxb, syb, sxc, syc); |
| g.translate(incX, 0); |
| |
| /* sC + sB -> dC + dB */ |
| drawImage(g, dxc, dyc, dxb, dyb, sxc, syc, sxb, syb); |
| } |
| |
| |
| |
| /* Size is chosen to match default imageable width of a NA letter |
| * page. This means there will be clipping, what is clipped will |
| * depend on PageFormat orientation. |
| */ |
| public Dimension getPreferredSize() { |
| return new Dimension(468, 468); |
| } |
| |
| } |