344 lines
15 KiB
Java
344 lines
15 KiB
Java
|
/*
|
||
|
* Copyright (c) 2022, 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
|
||
|
* @summary Pass if app exits without error code
|
||
|
* @bug 8264999
|
||
|
*/
|
||
|
|
||
|
import java.awt.*;
|
||
|
import java.io.*;
|
||
|
import java.util.List;
|
||
|
import java.util.ArrayList;
|
||
|
import java.awt.geom.*;
|
||
|
import javax.imageio.*;
|
||
|
import java.awt.image.*;
|
||
|
|
||
|
/**
|
||
|
* This tests redundant line segments. That is: if you draw a line from A to B, and then a line from
|
||
|
* B to B, then the expected behavior is for the last redundant segment to NOT affect the miter stroke.
|
||
|
*/
|
||
|
public class JoinMiterRedundantLineSegmentsTest {
|
||
|
|
||
|
public static void main(String[] args) throws Exception {
|
||
|
System.out.println("This test defines a series of shapes with optional shape data. The optional data (which is enclosed in brackets) should not make a difference in how the shape is rendered. This test renders the shape data with and without the bracketed segments and tests to see if those renderings are identical.");
|
||
|
|
||
|
List<Test> tests = createTests();
|
||
|
int sampleCtr = 1;
|
||
|
boolean[] booleans = new boolean[] {false, true};
|
||
|
boolean failed = false;
|
||
|
String header = null;
|
||
|
for (Test test : tests) {
|
||
|
header = null;
|
||
|
|
||
|
for (Object strokeHint : new Object[] { RenderingHints.VALUE_STROKE_PURE, RenderingHints.VALUE_STROKE_NORMALIZE } ) {
|
||
|
for (boolean createStrokedShape : booleans) {
|
||
|
for (boolean closePath : booleans) {
|
||
|
try {
|
||
|
test.run(strokeHint, createStrokedShape, closePath);
|
||
|
} catch(TestException e) {
|
||
|
failed = true;
|
||
|
|
||
|
if (header == null) {
|
||
|
System.out.println();
|
||
|
|
||
|
header = "#############################\n";
|
||
|
header += "## " + test.name + "\n";
|
||
|
header += "## " + test.description + "\n";
|
||
|
header += "## " + test.shapeString + "\n";
|
||
|
header += "#############################";
|
||
|
System.out.println(header);
|
||
|
}
|
||
|
|
||
|
System.out.println();
|
||
|
System.out.println("# sample index = " + (sampleCtr));
|
||
|
System.out.println("strokeHint = " + strokeHint);
|
||
|
System.out.println("createStrokedShape = " + createStrokedShape);
|
||
|
System.out.println("closePath = " + closePath);
|
||
|
System.out.println("FAILED");
|
||
|
e.printStackTrace(System.out);
|
||
|
BufferedImage bi = e.getImage();
|
||
|
File file = new File("failure-"+sampleCtr+".png");
|
||
|
ImageIO.write(bi, "png", file);
|
||
|
}
|
||
|
|
||
|
sampleCtr++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (failed)
|
||
|
System.exit(1);
|
||
|
}
|
||
|
|
||
|
private static List<Test> createTests() {
|
||
|
List<Test> tests = new ArrayList<>();
|
||
|
|
||
|
tests.add(new Test("Redundant diagonal line endpoint",
|
||
|
"m 0 0 l 10 10 [l 10 10]",
|
||
|
"This creates a diagonal line with a redundant endpoint; this is the core problem demonstrated in JDK-8264999."));
|
||
|
|
||
|
tests.add(new Test("jdk-8264999",
|
||
|
"m 24.954517 159 l 21.097446 157.5 [l 21.097446 157.5] l 17.61364 162 [l 17.61364 162] l 13.756569 163.5 [l 13.756569 163.5] l 11.890244 160.5",
|
||
|
"This is the original shape reported in https://bugs.openjdk.org/browse/JDK-8264999"));
|
||
|
|
||
|
tests.add(new Test("2x and 3x redundant lines",
|
||
|
"m 24.954517 159 l 21.097446 157.5 [l 21.097446 157.5 l 21.097446 157.5] l 17.61364 162 [l 17.61364 162 l 17.61364 162 l 17.61364 162] l 13.756569 163.5 [l 13.756569 163.5 l 13.756569 163.5 l 13.756569 163.5] l 11.890244 160.5",
|
||
|
"This is a derivative of JDK-8264999 that includes two or three redundant lines (instead of just one)."));
|
||
|
|
||
|
tests.add(new Test("cubic curve with redundant line",
|
||
|
"m 17 100 c 7 130 27 130 17 100 [l 17 100]",
|
||
|
"This creates a simple cubic curve (a teardrop shape) with one redundant line at the end."));
|
||
|
|
||
|
tests.add(new Test("degenerate cubic curve",
|
||
|
"m 19 180 l 20 181 [c 20 181 20 181 20 181]",
|
||
|
"This creates a degenerate cubic curve after the last end point."));
|
||
|
|
||
|
tests.add(new Test("degenerate quadratic curve",
|
||
|
"m 19 180 l 20 181 [q 20 181 20 181]",
|
||
|
"This creates a degenerate quadratic curve after the last end point."));
|
||
|
|
||
|
// This test reaches the line in Stroker.java where we detect a change of (+0, +0)
|
||
|
// and manually change the dx:
|
||
|
// dx = 1.0d;
|
||
|
|
||
|
// tests.add(new Test("Redundant lineTo after moveTo",
|
||
|
// "m 0 0 [l 0 0] l 10 10",
|
||
|
// "This creates a diagonal line that may include a redundant lineTo after the moveTo"));
|
||
|
|
||
|
// This test does NOT reach the same "dx = 1.0d" line. I'm not sure what the expected behavior here is.
|
||
|
|
||
|
// tests.add(new Test("lineTo after close",
|
||
|
// "m 0 0 z [l 10 10]",
|
||
|
// "This tests a lineTo after a close (but without a second moveTo)"));
|
||
|
|
||
|
// The following 2 tests fail because the mitered stroke covers different ares.
|
||
|
// They might (?) be working as expected, and I just don't understand the expected behavior?
|
||
|
|
||
|
// tests.add(new Test("Diagonal line, optional lineTo back",
|
||
|
// "m 0 0 l 20 20 [l 0 0]",
|
||
|
// "This creates a diagonal line and optionally returns to the starting point."));
|
||
|
//
|
||
|
// tests.add(new Test("Diagonal line, optional close",
|
||
|
// "m 0 0 l 20 20 l 0 0 [z]",
|
||
|
// "This creates a diagonal line, returns to the starting point, and optionally closes the path."));
|
||
|
|
||
|
// We've decided the following commented-out tests are invalid. The current interpretation is:
|
||
|
// "a moveTo statement without any additional information should NOT result in rendering anything"
|
||
|
|
||
|
// tests.add(new Test("empty line",
|
||
|
// "m 19 180 [l 19 180]",
|
||
|
// "This creates an empty shape with a lineTo the starting point."));
|
||
|
//
|
||
|
// tests.add(new Test("empty degenerate cubic curve",
|
||
|
// "m 19 180 [c 19 180 19 180 19 180]",
|
||
|
// "This creates an empty degenerate cubic curve that is effectively a line to the starting point."));
|
||
|
//
|
||
|
// tests.add(new Test("empty degenerate quadratic curve",
|
||
|
// "m 19 180 [q 19 180 19 180]",
|
||
|
// "This creates an empty degenerate quadratic curve that is effectively a line to the starting point."));
|
||
|
//
|
||
|
// tests.add(new Test("moveTo then close",
|
||
|
// "m 19 180 [z]",
|
||
|
// "This moves to a starting position and then optionally closes the path."));
|
||
|
|
||
|
return tests;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class TestException extends Exception {
|
||
|
BufferedImage bi;
|
||
|
|
||
|
public TestException(Throwable t, BufferedImage bi) {
|
||
|
super(t);
|
||
|
this.bi = bi;
|
||
|
}
|
||
|
|
||
|
public BufferedImage getImage() {
|
||
|
return bi;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class Test {
|
||
|
Path2D path_expected, path_actual;
|
||
|
String name, description, shapeString;
|
||
|
|
||
|
/**
|
||
|
* @param name a short name of this test
|
||
|
* @param shape shape data, including optional phrases in brackets. The shape should render the same
|
||
|
* whether the data in brackets is included or not.
|
||
|
* @param description a sentence describing this test
|
||
|
*/
|
||
|
public Test(String name, String shape, String description) {
|
||
|
// make sure the test contains optional path data. Because if it doesn't: this test
|
||
|
// is meaningless because nothing will change.
|
||
|
if (!shape.contains("["))
|
||
|
throw new IllegalArgumentException("The shape must contain optional path data.");
|
||
|
|
||
|
this.shapeString = shape;
|
||
|
this.name = name;
|
||
|
this.description = description;
|
||
|
path_expected = parse(shape, false);
|
||
|
path_actual = parse(shape, true);
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public String toString() {
|
||
|
return name;
|
||
|
}
|
||
|
|
||
|
private String stripBracketPhrases(String str) {
|
||
|
StringBuffer sb = new StringBuffer();
|
||
|
int ctr = 0;
|
||
|
for (int a = 0; a < str.length(); a++) {
|
||
|
char ch = str.charAt(a);
|
||
|
if (ch == '[') {
|
||
|
ctr++;
|
||
|
} else if (ch == ']') {
|
||
|
ctr--;
|
||
|
} else if (ctr == 0) {
|
||
|
sb.append(ch);
|
||
|
}
|
||
|
}
|
||
|
return sb.toString();
|
||
|
}
|
||
|
|
||
|
private Path2D.Double parse(String str, boolean includeBrackets) {
|
||
|
if (includeBrackets) {
|
||
|
str = str.replace('[', ' ');
|
||
|
str = str.replace(']', ' ');
|
||
|
} else {
|
||
|
str = stripBracketPhrases(str);
|
||
|
}
|
||
|
Path2D.Double path = new Path2D.Double();
|
||
|
String[] terms = str.split(" ");
|
||
|
int a = 0;
|
||
|
while (a < terms.length) {
|
||
|
if ("m".equals(terms[a])) {
|
||
|
path.moveTo(Double.parseDouble(terms[a + 1]), Double.parseDouble(terms[a + 2]));
|
||
|
a += 3;
|
||
|
} else if ("l".equals(terms[a])) {
|
||
|
path.lineTo( Double.parseDouble(terms[a+1]), Double.parseDouble(terms[a+2]) );
|
||
|
a += 3;
|
||
|
} else if ("q".equals(terms[a])) {
|
||
|
path.quadTo( Double.parseDouble(terms[a+1]), Double.parseDouble(terms[a+2]),
|
||
|
Double.parseDouble(terms[a+3]), Double.parseDouble(terms[a+4]) );
|
||
|
a += 5;
|
||
|
} else if ("c".equals(terms[a])) {
|
||
|
path.curveTo( Double.parseDouble(terms[a+1]), Double.parseDouble(terms[a+2]),
|
||
|
Double.parseDouble(terms[a+3]), Double.parseDouble(terms[a+4]),
|
||
|
Double.parseDouble(terms[a+5]), Double.parseDouble(terms[a+6]) );
|
||
|
a += 7;
|
||
|
} else if ("z".equals(terms[a])) {
|
||
|
path.closePath();
|
||
|
a += 1;
|
||
|
} else if(terms[a].trim().isEmpty()) {
|
||
|
a += 1;
|
||
|
} else {
|
||
|
throw new RuntimeException("\""+terms[a]+"\" in \""+str+"\"");
|
||
|
}
|
||
|
}
|
||
|
return path;
|
||
|
}
|
||
|
|
||
|
public void run(Object strokeRenderingHint, boolean createStrokedShape, boolean closePath) throws Exception {
|
||
|
BufferedImage bi_expected = new BufferedImage(400, 400, BufferedImage.TYPE_INT_ARGB);
|
||
|
BufferedImage bi_actual = new BufferedImage(400, 400, BufferedImage.TYPE_INT_ARGB);
|
||
|
|
||
|
paint(path_expected, bi_expected, Color.black, strokeRenderingHint, createStrokedShape, closePath);
|
||
|
paint(path_actual, bi_actual, Color.black, strokeRenderingHint, createStrokedShape, closePath);
|
||
|
|
||
|
try {
|
||
|
assertEquals(bi_expected, bi_actual);
|
||
|
} catch(Exception e) {
|
||
|
BufferedImage composite = new BufferedImage(400, 400, BufferedImage.TYPE_INT_ARGB);
|
||
|
paint(path_expected, composite, Color.blue, strokeRenderingHint, createStrokedShape, closePath);
|
||
|
paint(path_actual, composite, new Color(255,0,0,100), strokeRenderingHint, createStrokedShape, closePath);
|
||
|
throw new TestException(e, composite);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Throw an exception if two images are not equal.
|
||
|
*/
|
||
|
private static void assertEquals(BufferedImage bi1, BufferedImage bi2) {
|
||
|
int w = bi1.getWidth();
|
||
|
int h = bi1.getHeight();
|
||
|
int[] row1 = new int[w];
|
||
|
int[] row2 = new int[w];
|
||
|
for (int y = 0; y < h; y++) {
|
||
|
bi1.getRaster().getDataElements(0,y,w,1,row1);
|
||
|
bi2.getRaster().getDataElements(0,y,w,1,row2);
|
||
|
for (int x = 0; x < w; x++) {
|
||
|
if (row1[x] != row2[x])
|
||
|
throw new RuntimeException("failure at ("+x+", "+y+"): 0x"+Integer.toHexString(row1[x])+" != 0x"+Integer.toHexString(row2[x]));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create a transform that maps from one rectangle to another.
|
||
|
*/
|
||
|
private AffineTransform createTransform(Rectangle2D oldRect,Rectangle2D newRect) {
|
||
|
double scaleX = newRect.getWidth() / oldRect.getWidth();
|
||
|
double scaleY = newRect.getHeight() / oldRect.getHeight();
|
||
|
|
||
|
double translateX = -oldRect.getX() * scaleX + newRect.getX();
|
||
|
double translateY = -oldRect.getY() * scaleY + newRect.getY();
|
||
|
return new AffineTransform(scaleX, 0, 0, scaleY, translateX, translateY);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Paint a path to an image.
|
||
|
*/
|
||
|
private void paint(Path2D path, BufferedImage dst, Color color, Object strokeRenderingHint,
|
||
|
boolean createStrokedShape, boolean closePath) {
|
||
|
Rectangle2D pathBounds = path.getBounds2D();
|
||
|
pathBounds.setFrame(pathBounds.getX() - 10,
|
||
|
pathBounds.getY() - 10,
|
||
|
pathBounds.getWidth() + 20,
|
||
|
pathBounds.getHeight() + 20);
|
||
|
Rectangle imageBounds = new Rectangle(0, 0, dst.getWidth(), dst.getHeight());
|
||
|
|
||
|
Path2D p = new Path2D.Double();
|
||
|
p.append(path, false);
|
||
|
if (closePath)
|
||
|
p.closePath();
|
||
|
|
||
|
Graphics2D g = dst.createGraphics();
|
||
|
g.transform(createTransform(pathBounds, imageBounds));
|
||
|
g.setColor(color);
|
||
|
g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
|
||
|
g.setRenderingHint(RenderingHints.KEY_STROKE_CONTROL, strokeRenderingHint);
|
||
|
Stroke stroke = new BasicStroke(3);
|
||
|
if (createStrokedShape) {
|
||
|
g.fill( stroke.createStrokedShape(p) );
|
||
|
} else {
|
||
|
g.setStroke(stroke);
|
||
|
g.draw(p);
|
||
|
}
|
||
|
g.dispose();
|
||
|
}
|
||
|
}
|