blob: 41e334934e063cc3038dd93f05cd1c0884dfdd60 [file] [log] [blame]
/*
* Copyright (c) 2018, 2020, 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.Dimension;
import java.awt.EventQueue;
import java.awt.FlowLayout;
import java.awt.Font;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.Callable;
import javax.swing.JButton;
import javax.swing.JCheckBox;
import javax.swing.JComboBox;
import javax.swing.JComponent;
import javax.swing.JEditorPane;
import javax.swing.JFormattedTextField;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JList;
import javax.swing.JMenu;
import javax.swing.JMenuItem;
import javax.swing.JRadioButton;
import javax.swing.JScrollPane;
import javax.swing.JSpinner;
import javax.swing.JTable;
import javax.swing.JTextArea;
import javax.swing.JTextField;
import javax.swing.JToolTip;
import javax.swing.JTree;
import javax.swing.Popup;
import javax.swing.PopupFactory;
import javax.swing.SpinnerListModel;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.tree.DefaultMutableTreeNode;
import sun.swing.MenuItemLayoutHelper;
import static javax.swing.UIManager.getInstalledLookAndFeels;
/**
* @test
* @key headful
* @bug 8201552 8213535
* @summary Initial layout of the component should use correct graphics config.
* It is checked by SwingUtilities.updateComponentTreeUI(), if layout
* was correct the call to updateComponentTreeUI() will be no-op.
* @modules java.desktop/sun.swing
* @compile -encoding utf-8 StalePreferredSize.java
* @run main/othervm/timeout=400 StalePreferredSize
* @run main/othervm/timeout=400 -Dsun.java2d.uiScale=1 StalePreferredSize
* @run main/othervm/timeout=400 -Dsun.java2d.uiScale=2.25 StalePreferredSize
*/
public final class StalePreferredSize {
// Some text to be tested
static final String TEXT[] = new String[]{
"<span>A few words to get started before the "
+ "bug</span><span>overlapping text</span>",
"A quick brown fox jumps over the lazy dog",
"El veloz murciélago hindú comía feliz cardillo y kiwi. La cigüeña "
+ "tocaba el saxofón detrás del palenque de paja",
"Voix ambiguë d’un cœur qui au zéphyr préfère les jattes de kiwis",
"다람쥐 헌 쳇바퀴에 타고파",
"Съешь ещё этих мягких французских булок да выпей же чаю"};
static JFrame frame;
static Popup popup;
static JComponent component;
static int typeFont = 0; // 0 - default, 1 - bold, 2 - italic
static boolean addViaPopup;
public static void main(final String[] args) throws Exception {
for (final UIManager.LookAndFeelInfo laf : getInstalledLookAndFeels()) {
EventQueue.invokeAndWait(() -> setLookAndFeel(laf));
for (typeFont = 0; typeFont < 3; typeFont++) {
System.err.println("typeFont = " + typeFont);
for (boolean usePopup : new boolean[]{true, false}) {
addViaPopup = usePopup;
System.err.println("Use popup: " + usePopup);
for (final boolean html : new boolean[]{true, false}) {
for (String text : TEXT) {
if (html) {
text = "<html>" + text + "</html>";
}
test(text);
}
}
}
}
}
}
private static void test(String text) throws Exception {
System.err.println("text = " + text);
// Each Callable create a component to be tested
final List<Callable<JComponent>> comps = List.of(
() -> new JLabel(text),
() -> new JButton(text),
() -> new JMenuItem(text),
() -> new JMenu(text),
() -> new JList<>(new String[]{text}),
() -> new JComboBox<>(new String[]{text}),
() -> new JTextField(text),
() -> new JTextArea(text),
() -> new JCheckBox(text),
() -> new JFormattedTextField(text),
() -> new JRadioButton(text),
() -> new JTree(new DefaultMutableTreeNode(text)),
() -> new JSpinner(new SpinnerListModel(new String[]{text})),
() -> {
JToolTip tip = new JToolTip();
tip.setTipText(text);
return tip;
},
() -> {
JEditorPane pane = new JEditorPane();
pane.setText(text);
return pane;
},
() -> {
JTable table = new JTable(1, 1);
table.getModel().setValueAt(text, 0, 0);
return table;
}
);
for (final Callable<JComponent> creator : comps) {
checkComponent(creator);
}
}
static void checkComponent(Callable<JComponent> creator) throws Exception {
EventQueue.invokeAndWait(() -> {
try {
component = creator.call();
} catch (Exception e) {
throw new RuntimeException(e);
}
component.setEnabled(false); // minimize paint/focus events amount
Font font = component.getFont();
if (typeFont == 1) {
component.setFont(new Font(font.deriveFont(Font.BOLD).getAttributes()));
}
if (typeFont == 2) {
component.setFont(new Font(font.deriveFont(Font.ITALIC).getAttributes()));
}
frame = new JFrame();
// incorrect initial insets may ruin our size calculation
frame.setUndecorated(true); // TODO JDK-8244388
frame.setLayout(new FlowLayout());
frame.setSize(700, 400);
frame.setLocationRelativeTo(null);
if (addViaPopup) {
// doing our best to show lightweight or mediumweight popup
int x = frame.getX() + 50;
int y = frame.getY() + 200;
PopupFactory factory = PopupFactory.getSharedInstance();
popup = factory.getPopup(frame, component, x, y);
if (component instanceof JMenuItem) {
// TODO JDK-8244400
MenuItemLayoutHelper.clearUsedParentClientProperties((JMenuItem)component);
}
} else {
frame.add(new JScrollPane(component));
}
frame.setVisible(true);
if (popup != null) {
popup.show();
}
});
EventQueue.invokeAndWait(() -> {
if (!component.isValid()) {
dispose();
throw new RuntimeException("Component must be valid");
}
// After the frame was shown we change nothing, so current layout
// should be optimal and updateComponentTreeUI() should be no-op
Dimension before = component.getPreferredSize();
SwingUtilities.updateComponentTreeUI(frame);
Dimension after = component.getPreferredSize();
// We change the font size to some big value, as a result the
// layout and preferredSize of the component should be changed
component.setFont(component.getFont().deriveFont(35f));
Dimension last = component.getPreferredSize();
dispose();
if (!Objects.equals(before, after)) {
System.err.println("Component: " + component);
System.err.println("Before: " + before);
System.err.println("After: " + after);
throw new RuntimeException("Wrong PreferredSize");
}
// TODO JDK-8206024
// if (Objects.equals(after, last)) {
// System.err.println("Component: " + component);
// System.err.println("After: " + after);
// System.err.println("Last: " + last);
// throw new RuntimeException("Wrong PreferredSize");
// }
});
}
private static void dispose() {
if (popup != null) {
popup.hide();
popup = null;
}
frame.dispose();
}
private static void setLookAndFeel(final UIManager.LookAndFeelInfo laf) {
try {
UIManager.setLookAndFeel(laf.getClassName());
System.err.println("LookAndFeel: " + laf.getClassName());
} catch (final UnsupportedLookAndFeelException ignored) {
System.err.println(
"Unsupported LookAndFeel: " + laf.getClassName());
} catch (ClassNotFoundException | InstantiationException |
IllegalAccessException e) {
throw new RuntimeException(e);
}
}
}