b8d560b6cd
Reviewed-by: honkar
353 lines
13 KiB
Java
353 lines
13 KiB
Java
/*
|
|
* Copyright (c) 2015, 2024, 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.
|
|
*/
|
|
|
|
import java.awt.Color;
|
|
import java.awt.Dimension;
|
|
import java.awt.FlowLayout;
|
|
import java.awt.Point;
|
|
import java.awt.Rectangle;
|
|
import java.awt.Robot;
|
|
import java.awt.event.FocusAdapter;
|
|
import java.awt.event.FocusEvent;
|
|
import java.awt.image.BufferedImage;
|
|
import java.awt.image.ColorModel;
|
|
import java.awt.image.Raster;
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.util.concurrent.CountDownLatch;
|
|
import java.util.concurrent.TimeUnit;
|
|
import java.util.concurrent.atomic.AtomicReference;
|
|
import javax.imageio.ImageIO;
|
|
import javax.swing.JButton;
|
|
import javax.swing.JFrame;
|
|
import javax.swing.SwingUtilities;
|
|
|
|
/*
|
|
* @test
|
|
* @bug 8338103
|
|
* @key headful
|
|
* @summary Verifies that the OpenGL pipeline does not create artifacts
|
|
* with swing components after window is zoomed to maximum size and then
|
|
* resized back to normal. The test case simulates this operation using
|
|
* a JButton. A file image of the component will be saved before and after
|
|
* the window resize if the test fails. The test passes if both the button
|
|
* images are the same.
|
|
* @run main/othervm -Dsun.java2d.opengl=true -Dsun.java2d.opengl.fbobject=false SwingButtonResizeTestWithOpenGL
|
|
* @run main/othervm -Dsun.java2d.opengl=true -Dsun.java2d.opengl.fbobject=true SwingButtonResizeTestWithOpenGL
|
|
* @run main/othervm -Dsun.java2d.opengl=false SwingButtonResizeTestWithOpenGL
|
|
* @run main/othervm SwingButtonResizeTestWithOpenGL
|
|
*/
|
|
/*
|
|
* @test
|
|
* @key headful
|
|
* @requires (os.family == "windows")
|
|
* @run main/othervm -Dsun.java2d.d3d=false SwingButtonResizeTestWithOpenGL
|
|
* @run main/othervm -Dsun.java2d.d3d=true SwingButtonResizeTestWithOpenGL
|
|
*/
|
|
/*
|
|
* @test
|
|
* @key headful
|
|
* @requires (os.family == "linux")
|
|
* @run main/othervm -Dsun.java2d.xrender=false SwingButtonResizeTestWithOpenGL
|
|
* @run main/othervm -Dsun.java2d.xrender=true SwingButtonResizeTestWithOpenGL
|
|
*/
|
|
/*
|
|
* @test
|
|
* @key headful
|
|
* @requires (os.family == "mac")
|
|
* @run main/othervm -Dsun.java2d.metal=false SwingButtonResizeTestWithOpenGL
|
|
* @run main/othervm -Dsun.java2d.metal=true SwingButtonResizeTestWithOpenGL
|
|
*/
|
|
public class SwingButtonResizeTestWithOpenGL {
|
|
private static Robot robot;
|
|
private static CountDownLatch focusGainedLatch;
|
|
private JFrame frame;
|
|
private JButton button;
|
|
|
|
public SwingButtonResizeTestWithOpenGL() {
|
|
|
|
try {
|
|
SwingUtilities.invokeAndWait(() -> createGUI());
|
|
} catch (Exception e) {
|
|
throw new RuntimeException("Problems creating GUI");
|
|
}
|
|
}
|
|
|
|
private void createGUI() {
|
|
frame = new JFrame("SwingButtonResizeTestWithOpenGL");
|
|
button = new JButton("Button A");
|
|
frame.setLocation(200, 200);
|
|
frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
|
|
button.setPreferredSize(new Dimension(300, 300));
|
|
button.setFocusPainted(false);
|
|
button.addFocusListener(new FocusAdapter() {
|
|
public void focusGained(FocusEvent fe) {
|
|
focusGainedLatch.countDown();
|
|
}
|
|
});
|
|
frame.getContentPane().setLayout(new FlowLayout());
|
|
frame.getContentPane().add(button);
|
|
frame.pack();
|
|
frame.setVisible(true);
|
|
frame.toFront();
|
|
}
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
focusGainedLatch = new CountDownLatch(1);
|
|
SwingButtonResizeTestWithOpenGL test =
|
|
new SwingButtonResizeTestWithOpenGL();
|
|
test.runTest();
|
|
}
|
|
|
|
public void runTest() throws Exception {
|
|
BufferedImage bimage1;
|
|
BufferedImage bimage2;
|
|
|
|
try {
|
|
robot = new Robot();
|
|
robot.waitForIdle();
|
|
robot.delay(1000);
|
|
if (focusGainedLatch.await(3, TimeUnit.SECONDS)) {
|
|
System.out.println("Button focus gained...");
|
|
} else {
|
|
System.out.println("Button focus not gained...");
|
|
throw new RuntimeException(
|
|
"Can't gain focus on button even after waiting " +
|
|
"too long..");
|
|
}
|
|
|
|
System.out.println("Getting initial button image..image1");
|
|
bimage1 = getButtonImage();
|
|
|
|
// some platforms may not support maximize frame
|
|
if (frame.getToolkit().isFrameStateSupported(
|
|
JFrame.MAXIMIZED_BOTH)) {
|
|
// maximize frame from normal size
|
|
frame.setExtendedState(JFrame.MAXIMIZED_BOTH);
|
|
System.out.println("Frame is maximized");
|
|
robot.waitForIdle();
|
|
robot.delay(100);
|
|
|
|
if (frame.getToolkit().isFrameStateSupported(JFrame.NORMAL)) {
|
|
System.out.println("Frame is back to normal");
|
|
// resize from maximum size to normal
|
|
frame.setExtendedState(JFrame.NORMAL);
|
|
robot.waitForIdle();
|
|
robot.delay(100);
|
|
// capture image of JButton after resize
|
|
System.out.println(
|
|
"Getting image of JButton after resize..image2");
|
|
bimage2 = getButtonImage();
|
|
|
|
// compare button images from before and after frame resize
|
|
DiffImage di = new DiffImage(bimage1.getWidth(),
|
|
bimage1.getHeight());
|
|
System.out.println(
|
|
"Taking the diff of two images, image1 and image2");
|
|
if (!di.compare(bimage1, bimage2)) {
|
|
throw new RuntimeException(
|
|
"Button renderings are different after window "
|
|
+ "resize, num of Diff Pixels="
|
|
+ di.getNumDiffPixels());
|
|
} else {
|
|
System.out.println("Test passed...");
|
|
}
|
|
|
|
} else {
|
|
System.out.println(
|
|
"Test skipped: JFrame.NORMAL resize is " +
|
|
"not supported");
|
|
}
|
|
|
|
} else {
|
|
System.out.println(
|
|
"Test skipped: JFrame.MAXIMIZED_BOTH resize is " +
|
|
"not supported");
|
|
}
|
|
} finally {
|
|
SwingUtilities.invokeAndWait(() -> disposeFrame());
|
|
}
|
|
}
|
|
|
|
// Capture button rendering as a BufferedImage
|
|
private BufferedImage getButtonImage() {
|
|
try {
|
|
robot.waitForIdle();
|
|
robot.delay(500);
|
|
|
|
AtomicReference<Point> buttonLocRef = new AtomicReference<>();
|
|
SwingUtilities.invokeAndWait(
|
|
() -> buttonLocRef.set(button.getLocationOnScreen()));
|
|
Point buttonLoc = buttonLocRef.get();
|
|
System.out.println("Button loc: " + buttonLoc);
|
|
return robot.createScreenCapture(
|
|
new Rectangle(buttonLoc.x, buttonLoc.y, button.getWidth(),
|
|
button.getHeight()));
|
|
} catch (Exception e) {
|
|
throw new RuntimeException(
|
|
"Problems capturing button image from Robot", e);
|
|
}
|
|
}
|
|
|
|
private void disposeFrame() {
|
|
if(frame != null) {
|
|
frame.dispose();
|
|
}
|
|
}
|
|
|
|
// Save BufferedImage to PNG file
|
|
private void saveButtonImage(BufferedImage image, File file) {
|
|
if (image != null) {
|
|
try {
|
|
System.out.println(
|
|
"Saving button image to " + file.getAbsolutePath());
|
|
ImageIO.write(image, "PNG", file);
|
|
} catch (Exception e) {
|
|
throw new RuntimeException("Could not write image file");
|
|
}
|
|
} else {
|
|
throw new RuntimeException("BufferedImage was set to null");
|
|
}
|
|
}
|
|
|
|
private class DiffImage extends BufferedImage {
|
|
|
|
public boolean diff = false;
|
|
public int nDiff = -1;
|
|
|
|
Color bgColor;
|
|
|
|
int threshold = 0;
|
|
|
|
public DiffImage(int w, int h) {
|
|
super(w, h, BufferedImage.TYPE_INT_ARGB);
|
|
bgColor = Color.LIGHT_GRAY;
|
|
}
|
|
|
|
public int getNumDiffPixels() {
|
|
return nDiff;
|
|
}
|
|
|
|
public boolean compare(BufferedImage img1, BufferedImage img2)
|
|
throws IOException {
|
|
|
|
int minx1 = img1.getMinX();
|
|
int minx2 = img2.getMinX();
|
|
int miny1 = img1.getMinY();
|
|
int miny2 = img2.getMinY();
|
|
|
|
int w1 = img1.getWidth();
|
|
int w2 = img2.getWidth();
|
|
int h1 = img1.getHeight();
|
|
int h2 = img2.getHeight();
|
|
|
|
if ((minx1 != minx2) || (miny1 != miny2) || (w1 != w2)
|
|
|| (h1 != h2)) {
|
|
// image sizes are different
|
|
throw new RuntimeException(
|
|
"img1: <" + minx1 + "," + miny1 + "," + w1 + "x" + h1
|
|
+ ">" + " img2: " + minx2 + "," + miny2 + "," + w2 + "x"
|
|
+ h2 + ">" + " are different sizes");
|
|
}
|
|
// Get the actual data behind the images
|
|
Raster ras1 = img1.getData();
|
|
Raster ras2 = img2.getData();
|
|
|
|
ColorModel cm1 = img1.getColorModel();
|
|
ColorModel cm2 = img2.getColorModel();
|
|
|
|
int r1, r2; // red
|
|
int g1, g2; // green
|
|
int b1, b2; // blue
|
|
|
|
Object o1 = null;
|
|
Object o2 = null;
|
|
nDiff = 0;
|
|
for (int x = minx1; x < (minx1 + w1); x++) {
|
|
for (int y = miny1; y < (miny1 + h1); y++) {
|
|
// Causes rasters to allocate data
|
|
o1 = ras1.getDataElements(x, y, o1);
|
|
// and we reuse the data on every loop
|
|
o2 = ras2.getDataElements(x, y, o2);
|
|
|
|
r1 = cm1.getRed(o1);
|
|
r2 = cm2.getRed(o2);
|
|
g1 = cm1.getGreen(o1);
|
|
g2 = cm2.getGreen(o2);
|
|
b1 = cm1.getBlue(o1);
|
|
b2 = cm2.getBlue(o2);
|
|
|
|
int redAbs = Math.abs(r1 - r2);
|
|
int greenAbs = Math.abs(g1 - g2);
|
|
int blueAbs = Math.abs(b1 - b2);
|
|
if ((redAbs > threshold)
|
|
|| (greenAbs > threshold)
|
|
|| (blueAbs > threshold)) {
|
|
// pixel is different
|
|
setDiffPixel(x, y, redAbs, greenAbs, blueAbs);
|
|
nDiff++;
|
|
} else {
|
|
setSamePixel(x, y);
|
|
}
|
|
|
|
}
|
|
}
|
|
if (nDiff != 0) {
|
|
ImageIO.write(this, "png",
|
|
new File("diffImage.png"));
|
|
saveButtonImage(img1, new File("image1.png"));
|
|
saveButtonImage(img2, new File("image2.png"));
|
|
}
|
|
return nDiff == 0;
|
|
}
|
|
|
|
void setDiffPixel(int x, int y, int r, int g, int b) {
|
|
diff = true;
|
|
setPixelValue(x, y, 255, r, g, b);
|
|
}
|
|
|
|
void setSamePixel(int x, int y) {
|
|
if (bgColor != null) {
|
|
setPixelValue(x, y, 255, bgColor.getRed(),
|
|
bgColor.getGreen(),
|
|
bgColor.getBlue());
|
|
} else {
|
|
setPixelValue(x, y, 255, Color.black.getRed(),
|
|
Color.black.getGreen(), Color.black.getBlue());
|
|
}
|
|
}
|
|
|
|
void setPixelValue(int x, int y, int a, int r, int g, int b) {
|
|
// setRGB uses BufferedImage.TYPE_INT_ARGB format
|
|
int pixel =
|
|
((a & 0xff) << 24) + ((r & 0xff) << 16) + ((g & 0xff) << 8)
|
|
+ ((b & 0xff));
|
|
setRGB(x, y, pixel);
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|