/* * Copyright (c) 2004, 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 4980035 * @summary Unit test for new methods: * * AffineTransform.getRotateInstance(double x, double y); * AffineTransform.setToRotation(double x, double y); * AffineTransform.rotate(double x, double y); * * AffineTransform.getQuadrantRotateInstance(int numquads); * AffineTransform.setToQuadrantRotation(int numquads); * AffineTransform.quadrantRotate(int numquads); * * @author flar * @run main TestRotateMethods */ import java.awt.geom.AffineTransform; import java.awt.geom.Point2D; public class TestRotateMethods { /* The maximum errors allowed, measured in double precision "ulps" * Note that for most fields, the tests are extremely accurate - to * within 3 ulps of the smaller value in the comparison * For the translation components, the tests are still very accurate, * but the absolute number of ulps can be noticeably higher when we * use one of the rotate methods that takes an anchor point. * Since a double precision value has 56 bits of precision, even * 1024 ulps is extremely small as a ratio of the value. */ public static final double MAX_ULPS = 3.0; public static final double MAX_ANCHOR_TX_ULPS = 1024.0; public static double MAX_TX_ULPS = MAX_ULPS; // Vectors for quadrant rotations public static final double quadxvec[] = { 1.0, 0.0, -1.0, 0.0 }; public static final double quadyvec[] = { 0.0, 1.0, 0.0, -1.0 }; // Run tests once for each type of method: // tx = AffineTransform.getInstance() // tx.set() // tx.() public static enum Mode { GET, SET, MOD }; // Used to accumulate and report largest differences encountered by tests public static double maxulps = 0.0; public static double maxtxulps = 0.0; // Sample anchor points for testing. public static Point2D zeropt = new Point2D.Double(0, 0); public static Point2D testtxpts[] = { new Point2D.Double( 5, 5), new Point2D.Double( 20, -10), new Point2D.Double(-Math.PI, Math.E), }; public static void main(String argv[]) { test(Mode.GET); test(Mode.SET); test(Mode.MOD); System.out.println("Max scale and shear difference: "+maxulps+" ulps"); System.out.println("Max translate difference: "+maxtxulps+" ulps"); } public static void test(Mode mode) { MAX_TX_ULPS = MAX_ULPS; // Stricter tx testing with no anchor point test(mode, 0.5, null); test(mode, 1.0, null); test(mode, 3.0, null); // Anchor points make the tx values less reliable MAX_TX_ULPS = MAX_ANCHOR_TX_ULPS; for (int i = 0; i < testtxpts.length; i++) { test(mode, 1.0, testtxpts[i]); } MAX_TX_ULPS = MAX_ULPS; // Restore to default } public static void verify(AffineTransform at1, AffineTransform at2, Mode mode, double vectorscale, Point2D txpt, String message, double num, String units) { if (!compare(at1, at2)) { System.out.println("mode == "+mode); System.out.println("vectorscale == "+vectorscale); System.out.println("txpt == "+txpt); System.out.println(at1+", type = "+at1.getType()); System.out.println(at2+", type = "+at2.getType()); System.out.println("ScaleX values differ by "+ ulps(at1.getScaleX(), at2.getScaleX())+" ulps"); System.out.println("ScaleY values differ by "+ ulps(at1.getScaleY(), at2.getScaleY())+" ulps"); System.out.println("ShearX values differ by "+ ulps(at1.getShearX(), at2.getShearX())+" ulps"); System.out.println("ShearY values differ by "+ ulps(at1.getShearY(), at2.getShearY())+" ulps"); System.out.println("TranslateX values differ by "+ ulps(at1.getTranslateX(), at2.getTranslateX())+" ulps"); System.out.println("TranslateY values differ by "+ ulps(at1.getTranslateY(), at2.getTranslateY())+" ulps"); throw new RuntimeException(message + num + units); } } public static void test(Mode mode, double vectorscale, Point2D txpt) { AffineTransform at1, at2, at3; for (int deg = -720; deg <= 720; deg++) { if ((deg % 90) == 0) continue; double radians = Math.toRadians(deg); double vecy = Math.sin(radians) * vectorscale; double vecx = Math.cos(radians) * vectorscale; at1 = makeAT(mode, txpt, radians); at2 = makeAT(mode, txpt, vecx, vecy); verify(at1, at2, mode, vectorscale, txpt, "vector and radians do not match for ", deg, " degrees"); if (txpt == null) { // Make sure output was same as a with a 0,0 anchor point if (vectorscale == 1.0) { // Only need to test radians method for one scale factor at3 = makeAT(mode, zeropt, radians); verify(at1, at3, mode, vectorscale, zeropt, "radians not invariant with 0,0 translate at ", deg, " degrees"); } // But test vector methods with all scale factors at3 = makeAT(mode, zeropt, vecx, vecy); verify(at2, at3, mode, vectorscale, zeropt, "vector not invariant with 0,0 translate at ", deg, " degrees"); } } for (int quad = -8; quad <= 8; quad++) { double degrees = quad * 90.0; double radians = Math.toRadians(degrees); double vecx = quadxvec[quad & 3] * vectorscale; double vecy = quadyvec[quad & 3] * vectorscale; at1 = makeAT(mode, txpt, radians); at2 = makeAT(mode, txpt, vecx, vecy); verify(at1, at2, mode, vectorscale, txpt, "quadrant vector and radians do not match for ", degrees, " degrees"); at2 = makeQuadAT(mode, txpt, quad); verify(at1, at2, mode, vectorscale, txpt, "quadrant and radians do not match for ", quad, " quadrants"); if (txpt == null) { at3 = makeQuadAT(mode, zeropt, quad); verify(at2, at3, mode, vectorscale, zeropt, "quadrant not invariant with 0,0 translate at ", quad, " quadrants"); } } } public static AffineTransform makeRandomAT() { AffineTransform at = new AffineTransform(); at.scale(Math.random() * -10.0, Math.random() * 100.0); at.rotate(Math.random() * Math.PI); at.shear(Math.random(), Math.random()); at.translate(Math.random() * 300.0, Math.random() * -20.0); return at; } public static AffineTransform makeAT(Mode mode, Point2D txpt, double radians) { AffineTransform at; double tx = (txpt == null) ? 0.0 : txpt.getX(); double ty = (txpt == null) ? 0.0 : txpt.getY(); switch (mode) { case GET: if (txpt != null) { at = AffineTransform.getRotateInstance(radians, tx, ty); } else { at = AffineTransform.getRotateInstance(radians); } break; case SET: at = makeRandomAT(); if (txpt != null) { at.setToRotation(radians, tx, ty); } else { at.setToRotation(radians); } break; case MOD: at = makeRandomAT(); at.setToIdentity(); if (txpt != null) { at.rotate(radians, tx, ty); } else { at.rotate(radians); } break; default: throw new InternalError("unrecognized mode: "+mode); } return at; } public static AffineTransform makeAT(Mode mode, Point2D txpt, double vx, double vy) { AffineTransform at; double tx = (txpt == null) ? 0.0 : txpt.getX(); double ty = (txpt == null) ? 0.0 : txpt.getY(); switch (mode) { case GET: if (txpt != null) { at = AffineTransform.getRotateInstance(vx, vy, tx, ty); } else { at = AffineTransform.getRotateInstance(vx, vy); } break; case SET: at = makeRandomAT(); if (txpt != null) { at.setToRotation(vx, vy, tx, ty); } else { at.setToRotation(vx, vy); } break; case MOD: at = makeRandomAT(); at.setToIdentity(); if (txpt != null) { at.rotate(vx, vy, tx, ty); } else { at.rotate(vx, vy); } break; default: throw new InternalError("unrecognized mode: "+mode); } return at; } public static AffineTransform makeQuadAT(Mode mode, Point2D txpt, int quads) { AffineTransform at; double tx = (txpt == null) ? 0.0 : txpt.getX(); double ty = (txpt == null) ? 0.0 : txpt.getY(); switch (mode) { case GET: if (txpt != null) { at = AffineTransform.getQuadrantRotateInstance(quads, tx, ty); } else { at = AffineTransform.getQuadrantRotateInstance(quads); } break; case SET: at = makeRandomAT(); if (txpt != null) { at.setToQuadrantRotation(quads, tx, ty); } else { at.setToQuadrantRotation(quads); } break; case MOD: at = makeRandomAT(); at.setToIdentity(); if (txpt != null) { at.quadrantRotate(quads, tx, ty); } else { at.quadrantRotate(quads); } break; default: throw new InternalError("unrecognized mode: "+mode); } return at; } public static boolean compare(AffineTransform at1, AffineTransform at2) { maxulps = Math.max(maxulps, ulps(at1.getScaleX(), at2.getScaleX())); maxulps = Math.max(maxulps, ulps(at1.getScaleY(), at2.getScaleY())); maxulps = Math.max(maxulps, ulps(at1.getShearX(), at2.getShearX())); maxulps = Math.max(maxulps, ulps(at1.getShearY(), at2.getShearY())); maxtxulps = Math.max(maxtxulps, ulps(at1.getTranslateX(), at2.getTranslateX())); maxtxulps = Math.max(maxtxulps, ulps(at1.getTranslateY(), at2.getTranslateY())); return (getModifiedType(at1) == getModifiedType(at2) && (compare(at1.getScaleX(), at2.getScaleX(), MAX_ULPS)) && (compare(at1.getScaleY(), at2.getScaleY(), MAX_ULPS)) && (compare(at1.getShearX(), at2.getShearX(), MAX_ULPS)) && (compare(at1.getShearY(), at2.getShearY(), MAX_ULPS)) && (compare(at1.getTranslateX(), at2.getTranslateX(), MAX_TX_ULPS)) && (compare(at1.getTranslateY(), at2.getTranslateY(), MAX_TX_ULPS))); } public static int getModifiedType(AffineTransform at) { int type = at.getType(); // Some of the vector methods can introduce a tiny uniform scale // at some angles... if ((type & AffineTransform.TYPE_UNIFORM_SCALE) != 0) { maxulps = Math.max(maxulps, ulps(at.getDeterminant(), 1.0)); if (ulps(at.getDeterminant(), 1.0) <= MAX_ULPS) { // Really tiny - we will ignore it type &= (~AffineTransform.TYPE_UNIFORM_SCALE); } } return type; } public static boolean compare(double val1, double val2, double maxulps) { return (ulps(val1, val2) <= maxulps); } public static double ulps(double val1, double val2) { double diff = Math.abs(val1 - val2); double ulpmax = Math.min(Math.ulp(val1), Math.ulp(val2)); return (diff / ulpmax); } }