8015739: Background of JInternalFrame is located out of JInternalFrame

Reviewed-by: kizune, aivanov
This commit is contained in:
Harshitha Onkar 2022-11-04 19:48:31 +00:00 committed by Alexey Ivanov
parent b847fb6877
commit f857f795a9
2 changed files with 395 additions and 43 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 1998, 2021, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1998, 2022, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -25,24 +25,46 @@
package javax.swing.plaf.metal; package javax.swing.plaf.metal;
import javax.swing.*; import java.awt.BasicStroke;
import javax.swing.border.*;
import javax.swing.plaf.*;
import javax.swing.plaf.basic.BasicBorders;
import javax.swing.text.JTextComponent;
import java.awt.Component;
import java.awt.Insets;
import java.awt.Color; import java.awt.Color;
import java.awt.Component;
import java.awt.Dialog; import java.awt.Dialog;
import java.awt.Frame; import java.awt.Frame;
import java.awt.Graphics; import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Insets;
import java.awt.Stroke;
import java.awt.Window; import java.awt.Window;
import java.awt.geom.AffineTransform;
import javax.swing.AbstractButton;
import javax.swing.ButtonModel;
import javax.swing.JButton;
import javax.swing.JComponent;
import javax.swing.JInternalFrame;
import javax.swing.JMenu;
import javax.swing.JMenuBar;
import javax.swing.JMenuItem;
import javax.swing.JOptionPane;
import javax.swing.JScrollPane;
import javax.swing.JToolBar;
import javax.swing.SwingConstants;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.border.AbstractBorder;
import javax.swing.border.Border;
import javax.swing.border.CompoundBorder;
import javax.swing.border.EmptyBorder;
import javax.swing.border.LineBorder;
import javax.swing.border.MatteBorder;
import javax.swing.plaf.BorderUIResource;
import javax.swing.plaf.UIResource;
import javax.swing.plaf.basic.BasicBorders;
import javax.swing.text.JTextComponent;
import sun.swing.StringUIClientPropertyKey; import sun.swing.StringUIClientPropertyKey;
import sun.swing.SwingUtilities2; import sun.swing.SwingUtilities2;
/** /**
* Factory object that can vend Borders appropriate for the metal L & F. * Factory object that can vend Borders appropriate for the metal L & F.
* @author Steve Wilson * @author Steve Wilson
@ -217,16 +239,28 @@ public class MetalBorders {
*/ */
@SuppressWarnings("serial") // Superclass is not serializable across versions @SuppressWarnings("serial") // Superclass is not serializable across versions
public static class InternalFrameBorder extends AbstractBorder implements UIResource { public static class InternalFrameBorder extends AbstractBorder implements UIResource {
private static final int corner = 14; private static final int CORNER = 14;
/** /**
* Constructs a {@code InternalFrameBorder}. * Constructs a {@code InternalFrameBorder}.
*/ */
public InternalFrameBorder() {} public InternalFrameBorder() {}
/**
* Rounds a double to the nearest integer. It rounds 0.5 down,
* for example 1.5 is rounded to 1.0.
*
* @param d number to be rounded
* @return the rounded value
*/
private static int roundHalfDown(double d) {
double decP = (Math.ceil(d) - d);
return (int)((decP == 0.5) ? Math.floor(d) : Math.round(d));
}
public void paintBorder(Component c, Graphics g, int x, int y, public void paintBorder(Component c, Graphics g, int x, int y,
int w, int h) { int w, int h) {
Color background; Color background;
Color highlight; Color highlight;
Color shadow; Color shadow;
@ -241,39 +275,91 @@ public class MetalBorders {
shadow = MetalLookAndFeel.getControlInfo(); shadow = MetalLookAndFeel.getControlInfo();
} }
Graphics2D g2d = (Graphics2D) g;
AffineTransform at = g2d.getTransform();
Stroke oldStk = g2d.getStroke();
Color oldColor = g2d.getColor();
int stkWidth = 1;
// if m01 or m10 is non-zero, then there is a rotation or shear
// skip resetting the transform
boolean resetTransform = ((at.getShearX() == 0) && (at.getShearY() == 0));
int xtranslation;
int ytranslation;
int width;
int height;
if (resetTransform) {
g2d.setTransform(new AffineTransform());
stkWidth = roundHalfDown(Math.min(at.getScaleX(), at.getScaleY()));
double xx = at.getScaleX() * x + at.getTranslateX();
double yy = at.getScaleY() * y + at.getTranslateY();
xtranslation = roundHalfDown(xx);
ytranslation = roundHalfDown(yy);
width = roundHalfDown(at.getScaleX() * w + xx) - xtranslation;
height = roundHalfDown(at.getScaleY() * h + yy) - ytranslation;
} else {
width = w;
height = h;
xtranslation = x;
ytranslation = y;
}
g2d.translate(xtranslation, ytranslation);
// scaled border
int thickness = (int) Math.ceil(4 * at.getScaleX());
g.setColor(background); g.setColor(background);
// Draw outermost lines
g.drawLine( 1, 0, w-2, 0);
g.drawLine( 0, 1, 0, h-2);
g.drawLine( w-1, 1, w-1, h-2);
g.drawLine( 1, h-1, w-2, h-1);
// Draw the bulk of the border // Draw the bulk of the border
for (int i = 1; i < 5; i++) { for (int i = 0; i <= thickness; i++) {
g.drawRect(x+i,y+i,w-(i*2)-1, h-(i*2)-1); g.drawRect(i, i, width - (i * 2), height - (i * 2));
} }
if (c instanceof JInternalFrame && if (c instanceof JInternalFrame && ((JInternalFrame)c).isResizable()) {
((JInternalFrame)c).isResizable()) { // set new stroke to draw shadow and highlight lines
g.setColor(highlight); g2d.setStroke(new BasicStroke((float) stkWidth));
// midpoint at which highlight & shadow lines
// are positioned on the border
int midPoint = thickness / 2;
int offset = ((at.getScaleX() - stkWidth) >= 0 && stkWidth % 2 != 0) ? 1 : 0;
int loc1 = thickness % 2 == 0 ? midPoint + stkWidth / 2 - stkWidth : midPoint;
int loc2 = thickness % 2 == 0 ? midPoint + stkWidth / 2 : midPoint + stkWidth;
// scaled corner
int corner = (int) Math.round(CORNER * at.getScaleX());
// Draw the Long highlight lines // Draw the Long highlight lines
g.drawLine( corner+1, 3, w-corner, 3); g.setColor(highlight);
g.drawLine( 3, corner+1, 3, h-corner); g.drawLine(corner + 1, loc2, width - corner, loc2); //top
g.drawLine( w-2, corner+1, w-2, h-corner); g.drawLine(loc2, corner + 1, loc2, height - corner); //left
g.drawLine( corner+1, h-2, w-corner, h-2); g.drawLine((width - offset) - loc1, corner + 1,
(width - offset) - loc1, height - corner); //right
g.drawLine(corner + 1, (height - offset) - loc1,
width - corner, (height - offset) - loc1); //bottom
g.setColor(shadow);
// Draw the Long shadow lines // Draw the Long shadow lines
g.drawLine( corner, 2, w-corner-1, 2); g.setColor(shadow);
g.drawLine( 2, corner, 2, h-corner-1); g.drawLine(corner, loc1, width - corner - 1, loc1);
g.drawLine( w-3, corner, w-3, h-corner-1); g.drawLine(loc1, corner, loc1, height - corner - 1);
g.drawLine( corner, h-3, w-corner-1, h-3); g.drawLine((width - offset) - loc2, corner,
(width - offset) - loc2, height - corner - 1);
g.drawLine(corner, (height - offset) - loc2,
width - corner - 1, (height - offset) - loc2);
} }
// restore previous transform
g2d.translate(-xtranslation, -ytranslation);
if (resetTransform) {
g2d.setColor(oldColor);
g2d.setTransform(at);
g2d.setStroke(oldStk);
}
} }
public Insets getBorderInsets(Component c, Insets newInsets) { public Insets getBorderInsets(Component c, Insets newInsets) {
newInsets.set(5, 5, 5, 5); newInsets.set(4, 4, 4, 4);
return newInsets; return newInsets;
} }
} }

View File

@ -0,0 +1,266 @@
/*
* 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.
*/
import java.awt.AWTException;
import java.awt.Color;
import java.awt.GridBagLayout;
import java.awt.Image;
import java.awt.Point;
import java.awt.Rectangle;
import java.awt.Robot;
import java.awt.image.MultiResolutionImage;
import java.awt.image.RenderedImage;
import java.io.File;
import java.lang.reflect.InvocationTargetException;
import java.util.List;
import javax.imageio.ImageIO;
import javax.swing.JFrame;
import javax.swing.JInternalFrame;
import javax.swing.JLabel;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
/*
* @test
* @bug 8015739
* @key headful
* @summary Tests whether background color of JInternalFrame is visible
* in the border region at different scales by checking the midpoints
* and corners of the border.
*
* @requires (os.family == "windows")
* @run main/othervm -Dsun.java2d.uiScale=1 InternalFrameBorderTest
* @run main/othervm -Dsun.java2d.uiScale=1.25 InternalFrameBorderTest
* @run main/othervm -Dsun.java2d.uiScale=1.5 InternalFrameBorderTest
* @run main/othervm -Dsun.java2d.uiScale=1.75 InternalFrameBorderTest
* @run main/othervm -Dsun.java2d.uiScale=2 InternalFrameBorderTest
* @run main/othervm -Dsun.java2d.uiScale=2.5 InternalFrameBorderTest
* @run main/othervm -Dsun.java2d.uiScale=3 InternalFrameBorderTest
*/
/*
* @test
* @bug 8015739
* @key headful
* @summary Tests whether background color of JInternalFrame is visible
* in the border region at different scales by checking the midpoints
* and corners of the border.
*
* @requires (os.family == "mac" | os.family == "linux")
* @run main/othervm -Dsun.java2d.uiScale=1 InternalFrameBorderTest
* @run main/othervm -Dsun.java2d.uiScale=2 InternalFrameBorderTest
*/
public class InternalFrameBorderTest {
private static final int FRAME_SIZE = 300;
private static final int INTFRAME_SIZE = 150;
private static final int MIDPOINT = INTFRAME_SIZE / 2;
private static final int BORDER_THICKNESS = 4;
private static final StringBuffer errorLog = new StringBuffer();
private static JFrame jFrame;
private static Rectangle jFrameBounds;
private static JInternalFrame iFrame;
private static Point iFrameLoc;
private static int iFrameMaxX;
private static int iFrameMaxY;
private static Robot robot;
private static String uiScale;
public static void main(String[] args) throws AWTException,
InterruptedException, InvocationTargetException {
try {
UIManager.setLookAndFeel("javax.swing.plaf.metal.MetalLookAndFeel");
} catch (Exception e) {
System.out.println("Metal LAF class not supported");
return;
}
try {
robot = new Robot();
robot.setAutoDelay(200);
uiScale = System.getProperty("sun.java2d.uiScale");
SwingUtilities.invokeAndWait(InternalFrameBorderTest::createAndShowGUI);
robot.waitForIdle();
robot.delay(500);
SwingUtilities.invokeAndWait(() -> {
iFrameLoc = iFrame.getLocationOnScreen();
iFrameMaxX = iFrameLoc.x + INTFRAME_SIZE;
iFrameMaxY = iFrameLoc.y + INTFRAME_SIZE;
jFrameBounds = jFrame.getBounds();
});
// Check Borders
checkBorderMidPoints("TOP");
checkBorderMidPoints("RIGHT");
checkBorderMidPoints("BOTTOM");
checkBorderMidPoints("LEFT");
// Check Corner Diagonals
checkCorners("TOP_LEFT");
checkCorners("TOP_RIGHT");
checkCorners("BOTTOM_RIGHT");
checkCorners("BOTTOM_LEFT");
if (!errorLog.isEmpty()) {
saveScreenCapture("JIF_uiScale_" + uiScale + ".png");
throw new RuntimeException("Following error(s) occurred: \n"
+ errorLog);
}
} finally {
if (jFrame != null) {
jFrame.dispose();
}
robot.delay(500);
}
}
private static void checkBorderMidPoints(String borderDirection) {
int x, y;
int start, stop;
switch (borderDirection) {
case "TOP" -> {
x = iFrameLoc.x + MIDPOINT;
y = iFrameLoc.y + BORDER_THICKNESS;
start = iFrameLoc.y;
stop = iFrameLoc.y + BORDER_THICKNESS - 1;
}
case "RIGHT" -> {
x = iFrameMaxX - BORDER_THICKNESS;
y = iFrameLoc.y + MIDPOINT;
start = iFrameMaxX - BORDER_THICKNESS + 1;
stop = iFrameMaxX;
}
case "BOTTOM" -> {
x = iFrameLoc.x + MIDPOINT;
y = iFrameMaxY - BORDER_THICKNESS;
start = iFrameMaxY - BORDER_THICKNESS + 1;
stop = iFrameMaxY;
}
case "LEFT" -> {
x = iFrameLoc.x;
y = iFrameLoc.y + MIDPOINT;
start = iFrameLoc.x;
stop = iFrameLoc.x + BORDER_THICKNESS - 1;
}
default -> throw new IllegalStateException("Unexpected value: "
+ borderDirection);
}
boolean isVertical = borderDirection.equals("RIGHT")
|| borderDirection.equals("LEFT");
boolean isHorizontal = borderDirection.equals("TOP")
|| borderDirection.equals("BOTTOM");
robot.mouseMove(x, y);
for (int i = start; i < stop; i++) {
int locX = isVertical ? i : (iFrameLoc.x + MIDPOINT);
int locY = isHorizontal ? i : (iFrameLoc.y + MIDPOINT);
if (Color.RED.equals(robot.getPixelColor(locX, locY))) {
errorLog.append("At uiScale: " + uiScale
+ ", Red background color detected at "
+ borderDirection + " border.\n");
break;
}
}
robot.delay(300);
}
private static void checkCorners(String cornerLocation) {
int x, y;
switch (cornerLocation) {
case "TOP_LEFT" -> {
x = iFrameLoc.x;
y = iFrameLoc.y;
}
case "TOP_RIGHT" -> {
x = iFrameMaxX;
y = iFrameLoc.y;
}
case "BOTTOM_RIGHT" -> {
x = iFrameMaxX;
y = iFrameMaxY;
}
case "BOTTOM_LEFT" -> {
x = iFrameLoc.x;
y = iFrameMaxY;
}
default -> throw new IllegalStateException("Unexpected value: "
+ cornerLocation);
}
boolean isTop = cornerLocation.equals("TOP_LEFT")
|| cornerLocation.equals("TOP_RIGHT");
boolean isLeft = cornerLocation.equals("TOP_LEFT")
|| cornerLocation.equals("BOTTOM_LEFT");
robot.mouseMove(x, y);
for (int i = 0; i < BORDER_THICKNESS - 1; i++) {
int locX = isLeft ? (x + i) : (x - i);
int locY = isTop ? (y + i) : (y - i);
if (Color.RED.equals(robot.getPixelColor(locX, locY))) {
errorLog.append("At uiScale: " + uiScale + ", Red background color"
+ " detected at " + cornerLocation + " corner.\n");
break;
}
}
robot.delay(300);
}
private static void createAndShowGUI() {
jFrame = new JFrame();
jFrame.setSize(FRAME_SIZE, FRAME_SIZE);
jFrame.setLayout(null);
jFrame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
JLabel scale = new JLabel("UI Scale: " + uiScale);
iFrame = new JInternalFrame("iframe", true);
iFrame.setLayout(new GridBagLayout());
iFrame.setBackground(Color.RED);
iFrame.add(scale);
iFrame.setLocation(30, 30);
jFrame.getContentPane().add(iFrame);
iFrame.setSize(INTFRAME_SIZE, INTFRAME_SIZE);
iFrame.setVisible(true);
jFrame.setLocation(150, 150);
jFrame.setVisible(true);
}
private static void saveScreenCapture(String filename) {
MultiResolutionImage mrImage = robot.createMultiResolutionScreenCapture(jFrameBounds);
List<Image> variants = mrImage.getResolutionVariants();
RenderedImage image = (RenderedImage) variants.get(variants.size() - 1);
try {
ImageIO.write(image, "png", new File(filename));
} catch (Exception e) {
e.printStackTrace();
}
}
}