mirror of
https://github.com/openjdk/jdk8u.git
synced 2025-12-10 00:09:35 -06:00
8348760: RadioButton is not shown if JRadioButtonMenuItem is rendered with ImageIcon in WindowsLookAndFeel
8365375: Method SU3.setAcceleratorSelectionForeground assigns to acceleratorForeground Reviewed-by: sgehwolf, andrew Backport-of: 31fc05c4721aa1aaf12e5f326a52c1db48006abb
This commit is contained in:
parent
3f2b6ffee4
commit
333a55103e
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2002, 2010, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2002, 2025, 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
|
||||
@ -36,11 +36,20 @@ import java.util.concurrent.Callable;
|
||||
import java.applet.Applet;
|
||||
import java.awt.AWTEvent;
|
||||
import java.awt.EventQueue;
|
||||
import java.awt.Color;
|
||||
import java.awt.Component;
|
||||
import java.awt.Container;
|
||||
import java.awt.Graphics;
|
||||
import java.awt.Insets;
|
||||
import java.awt.Rectangle;
|
||||
import java.awt.Window;
|
||||
import javax.swing.ButtonModel;
|
||||
import javax.swing.Icon;
|
||||
import javax.swing.JComponent;
|
||||
import javax.swing.JMenu;
|
||||
import javax.swing.RepaintManager;
|
||||
import sun.swing.MenuItemLayoutHelper;
|
||||
import sun.swing.SwingUtilities2;
|
||||
|
||||
/**
|
||||
* A collection of utility methods for Swing.
|
||||
@ -61,6 +70,10 @@ public class SwingUtilities3 {
|
||||
private static final Object DELEGATE_REPAINT_MANAGER_KEY =
|
||||
new StringBuilder("DelegateRepaintManagerKey");
|
||||
|
||||
private static Color disabledForeground;
|
||||
private static Color acceleratorSelectionForeground;
|
||||
private static Color acceleratorForeground;
|
||||
|
||||
/**
|
||||
* Registers delegate RepaintManager for {@code JComponent}.
|
||||
*/
|
||||
@ -113,6 +126,128 @@ public class SwingUtilities3 {
|
||||
return Boolean.TRUE == vsyncedMap.get(rootContainer);
|
||||
}
|
||||
|
||||
public static void applyInsets(Rectangle rect, Insets insets) {
|
||||
if (insets != null) {
|
||||
rect.x += insets.left;
|
||||
rect.y += insets.top;
|
||||
rect.width -= (insets.right + rect.x);
|
||||
rect.height -= (insets.bottom + rect.y);
|
||||
}
|
||||
}
|
||||
|
||||
public static void paintCheckIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr,
|
||||
Color holdc, Color foreground) {
|
||||
if (lh.getCheckIcon() != null) {
|
||||
ButtonModel model = lh.getMenuItem().getModel();
|
||||
if (model.isArmed() || (lh.getMenuItem() instanceof JMenu
|
||||
&& model.isSelected())) {
|
||||
g.setColor(foreground);
|
||||
} else {
|
||||
g.setColor(holdc);
|
||||
}
|
||||
if (lh.useCheckAndArrow()) {
|
||||
lh.getCheckIcon().paintIcon(lh.getMenuItem(), g,
|
||||
lr.getCheckRect().x, lr.getCheckRect().y);
|
||||
}
|
||||
g.setColor(holdc);
|
||||
}
|
||||
}
|
||||
|
||||
public static void paintIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr, Color holdc) {
|
||||
if (lh.getIcon() != null) {
|
||||
Icon icon;
|
||||
ButtonModel model = lh.getMenuItem().getModel();
|
||||
if (!model.isEnabled()) {
|
||||
icon = lh.getMenuItem().getDisabledIcon();
|
||||
} else if (model.isPressed() && model.isArmed()) {
|
||||
icon = lh.getMenuItem().getPressedIcon();
|
||||
if (icon == null) {
|
||||
// Use default icon
|
||||
icon = lh.getMenuItem().getIcon();
|
||||
}
|
||||
} else {
|
||||
icon = lh.getMenuItem().getIcon();
|
||||
}
|
||||
|
||||
if (icon != null) {
|
||||
icon.paintIcon(lh.getMenuItem(), g, lr.getIconRect().x,
|
||||
lr.getIconRect().y);
|
||||
g.setColor(holdc);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public static void paintAccText(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr) {
|
||||
if (!lh.getAccText().isEmpty()) {
|
||||
ButtonModel model = lh.getMenuItem().getModel();
|
||||
g.setFont(lh.getAccFontMetrics().getFont());
|
||||
if (!model.isEnabled()) {
|
||||
|
||||
// paint the accText disabled
|
||||
if (disabledForeground != null) {
|
||||
g.setColor(disabledForeground);
|
||||
SwingUtilities2.drawString(lh.getMenuItem(), g,
|
||||
lh.getAccText(), lr.getAccRect().x,
|
||||
lr.getAccRect().y + lh.getAccFontMetrics().getAscent());
|
||||
} else {
|
||||
g.setColor(lh.getMenuItem().getBackground().brighter());
|
||||
SwingUtilities2.drawString(lh.getMenuItem(), g,
|
||||
lh.getAccText(), lr.getAccRect().x,
|
||||
lr.getAccRect().y + lh.getAccFontMetrics().getAscent());
|
||||
g.setColor(lh.getMenuItem().getBackground().darker());
|
||||
SwingUtilities2.drawString(lh.getMenuItem(), g,
|
||||
lh.getAccText(), lr.getAccRect().x - 1,
|
||||
lr.getAccRect().y + lh.getFontMetrics().getAscent() - 1);
|
||||
}
|
||||
} else {
|
||||
|
||||
// paint the accText normally
|
||||
if (model.isArmed()
|
||||
|| (lh.getMenuItem() instanceof JMenu
|
||||
&& model.isSelected())) {
|
||||
g.setColor(acceleratorSelectionForeground);
|
||||
} else {
|
||||
g.setColor(acceleratorForeground);
|
||||
}
|
||||
SwingUtilities2.drawString(lh.getMenuItem(), g, lh.getAccText(),
|
||||
lr.getAccRect().x, lr.getAccRect().y +
|
||||
lh.getAccFontMetrics().getAscent());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static void setDisabledForeground(Color disabledFg) {
|
||||
disabledForeground = disabledFg;
|
||||
}
|
||||
|
||||
public static void setAcceleratorSelectionForeground(Color acceleratorSelectionFg) {
|
||||
acceleratorSelectionForeground = acceleratorSelectionFg;
|
||||
}
|
||||
|
||||
public static void setAcceleratorForeground(Color acceleratorFg) {
|
||||
acceleratorForeground = acceleratorFg;
|
||||
}
|
||||
|
||||
public static void paintArrowIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr,
|
||||
Color foreground) {
|
||||
if (lh.getArrowIcon() != null) {
|
||||
ButtonModel model = lh.getMenuItem().getModel();
|
||||
if (model.isArmed() || (lh.getMenuItem() instanceof JMenu
|
||||
&& model.isSelected())) {
|
||||
g.setColor(foreground);
|
||||
}
|
||||
if (lh.useCheckAndArrow()) {
|
||||
lh.getArrowIcon().paintIcon(lh.getMenuItem(), g,
|
||||
lr.getArrowRect().x, lr.getArrowRect().y);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns delegate {@code RepaintManager} for {@code component} hierarchy.
|
||||
*/
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 1997, 2006, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1997, 2025, 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
|
||||
@ -74,6 +74,24 @@ public class WindowsCheckBoxMenuItemUI extends BasicCheckBoxMenuItemUI {
|
||||
}
|
||||
super.paintBackground(g, menuItem, bgColor);
|
||||
}
|
||||
|
||||
/**
|
||||
* Paint MenuItem.
|
||||
*/
|
||||
protected void paintMenuItem(Graphics g, JComponent c,
|
||||
Icon checkIcon, Icon arrowIcon,
|
||||
Color background, Color foreground,
|
||||
int defaultTextIconGap) {
|
||||
if (WindowsMenuItemUI.isVistaPainting()) {
|
||||
WindowsMenuItemUI.paintMenuItem(accessor, g, c, checkIcon,
|
||||
arrowIcon, background, foreground, defaultTextIconGap,
|
||||
menuItem, getPropertyPrefix());
|
||||
return;
|
||||
}
|
||||
super.paintMenuItem(g, c, checkIcon, arrowIcon, background,
|
||||
foreground, defaultTextIconGap);
|
||||
}
|
||||
|
||||
/**
|
||||
* Method which renders the text of the current menu item.
|
||||
* <p>
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 1998, 2013, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1998, 2025, 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
|
||||
@ -785,6 +785,7 @@ public class WindowsIconFactory implements Serializable
|
||||
}
|
||||
assert menuItem == null || c == menuItem;
|
||||
Icon icon = getIcon();
|
||||
|
||||
if (type == JCheckBoxMenuItem.class
|
||||
|| type == JRadioButtonMenuItem.class) {
|
||||
AbstractButton b = (AbstractButton) c;
|
||||
@ -808,19 +809,18 @@ public class WindowsIconFactory implements Serializable
|
||||
}
|
||||
XPStyle xp = XPStyle.getXP();
|
||||
if (xp != null) {
|
||||
Skin skin;
|
||||
skin = xp.getSkin(c, backgroundPart);
|
||||
skin.paintSkin(g, x, y,
|
||||
getIconWidth(), getIconHeight(), backgroundState);
|
||||
if (icon == null) {
|
||||
skin = xp.getSkin(c, part);
|
||||
Skin skin = xp.getSkin(c, part);
|
||||
if (icon == null || icon.getIconHeight() <= 16) {
|
||||
skin.paintSkin(g, x + OFFSET, y + OFFSET, state);
|
||||
} else {
|
||||
skin.paintSkin(g, x + OFFSET, y + icon.getIconHeight() / 2, state);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (icon != null) {
|
||||
icon.paintIcon(c, g, x + OFFSET, y + OFFSET);
|
||||
icon.paintIcon(c, g, x + VistaMenuItemCheckIconFactory.getIconWidth(),
|
||||
y + OFFSET);
|
||||
}
|
||||
}
|
||||
private static WindowsMenuItemUIAccessor getAccessor(
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1997, 2025, 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
|
||||
@ -31,9 +31,11 @@ import javax.swing.plaf.*;
|
||||
import javax.swing.plaf.basic.*;
|
||||
|
||||
import sun.swing.SwingUtilities2;
|
||||
import sun.swing.MenuItemLayoutHelper;
|
||||
|
||||
import com.sun.java.swing.plaf.windows.TMSchema.*;
|
||||
import com.sun.java.swing.plaf.windows.XPStyle.*;
|
||||
import com.sun.java.swing.SwingUtilities3;
|
||||
|
||||
/**
|
||||
* Windows rendition of the component.
|
||||
@ -47,8 +49,12 @@ import com.sun.java.swing.plaf.windows.XPStyle.*;
|
||||
*
|
||||
* @author Igor Kushnirskiy
|
||||
*/
|
||||
|
||||
public class WindowsMenuItemUI extends BasicMenuItemUI {
|
||||
|
||||
private static Color disabledForeground;
|
||||
private static Color acceleratorSelectionForeground;
|
||||
private static Color acceleratorForeground;
|
||||
|
||||
final WindowsMenuItemUIAccessor accessor =
|
||||
new WindowsMenuItemUIAccessor() {
|
||||
|
||||
@ -68,6 +74,136 @@ public class WindowsMenuItemUI extends BasicMenuItemUI {
|
||||
return new WindowsMenuItemUI();
|
||||
}
|
||||
|
||||
protected void installDefaults() {
|
||||
super.installDefaults();
|
||||
String prefix = getPropertyPrefix();
|
||||
|
||||
if (acceleratorSelectionForeground == null ||
|
||||
acceleratorSelectionForeground instanceof UIResource) {
|
||||
acceleratorSelectionForeground =
|
||||
UIManager.getColor(prefix + ".acceleratorSelectionForeground");
|
||||
}
|
||||
if (acceleratorForeground == null ||
|
||||
acceleratorForeground instanceof UIResource) {
|
||||
acceleratorForeground =
|
||||
UIManager.getColor(prefix + ".acceleratorForeground");
|
||||
}
|
||||
if (disabledForeground == null ||
|
||||
disabledForeground instanceof UIResource) {
|
||||
disabledForeground =
|
||||
UIManager.getColor(prefix + ".disabledForeground");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private static void applyInsets(Rectangle rect, Insets insets) {
|
||||
SwingUtilities3.applyInsets(rect, insets);
|
||||
}
|
||||
|
||||
private static void paintCheckIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr,
|
||||
Color holdc, Color foreground) {
|
||||
SwingUtilities3.paintCheckIcon(g, lh, lr, holdc, foreground);
|
||||
}
|
||||
|
||||
private static void paintIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr, Color holdc) {
|
||||
SwingUtilities3.paintIcon(g, lh, lr, holdc);
|
||||
}
|
||||
|
||||
private static void paintAccText(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr) {
|
||||
SwingUtilities3.setDisabledForeground(disabledForeground);
|
||||
SwingUtilities3.setAcceleratorSelectionForeground(
|
||||
acceleratorSelectionForeground);
|
||||
SwingUtilities3.setAcceleratorForeground(acceleratorForeground);
|
||||
SwingUtilities3.paintAccText(g, lh, lr);
|
||||
}
|
||||
|
||||
private static void paintArrowIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr,
|
||||
Color foreground) {
|
||||
SwingUtilities3.paintArrowIcon(g, lh, lr, foreground);
|
||||
}
|
||||
|
||||
protected void paintMenuItem(Graphics g, JComponent c,
|
||||
Icon checkIcon, Icon arrowIcon,
|
||||
Color background, Color foreground,
|
||||
int defaultTextIconGap) {
|
||||
if (WindowsMenuItemUI.isVistaPainting()) {
|
||||
WindowsMenuItemUI.paintMenuItem(accessor, g, c, checkIcon,
|
||||
arrowIcon, background, foreground,
|
||||
defaultTextIconGap, menuItem,
|
||||
getPropertyPrefix());
|
||||
return;
|
||||
}
|
||||
super.paintMenuItem(g, c, checkIcon, arrowIcon, background,
|
||||
foreground, defaultTextIconGap);
|
||||
}
|
||||
|
||||
static void paintMenuItem(WindowsMenuItemUIAccessor accessor, Graphics g,
|
||||
JComponent c, Icon checkIcon, Icon arrowIcon,
|
||||
Color background, Color foreground,
|
||||
int defaultTextIconGap, JMenuItem menuItem, String prefix) {
|
||||
// Save original graphics font and color
|
||||
Font holdf = g.getFont();
|
||||
Color holdc = g.getColor();
|
||||
|
||||
JMenuItem mi = (JMenuItem) c;
|
||||
g.setFont(mi.getFont());
|
||||
|
||||
Rectangle viewRect = new Rectangle(0, 0, mi.getWidth(), mi.getHeight());
|
||||
applyInsets(viewRect, mi.getInsets());
|
||||
|
||||
String acceleratorDelimiter =
|
||||
UIManager.getString("MenuItem.acceleratorDelimiter");
|
||||
if (acceleratorDelimiter == null) { acceleratorDelimiter = "+"; }
|
||||
Font acceleratorFont = UIManager.getFont("MenuItem.acceleratorFont");
|
||||
if (acceleratorFont == null) {
|
||||
acceleratorFont = UIManager.getFont("MenuItem.font");
|
||||
}
|
||||
|
||||
MenuItemLayoutHelper lh = new MenuItemLayoutHelper(mi, checkIcon,
|
||||
arrowIcon, viewRect, defaultTextIconGap, acceleratorDelimiter,
|
||||
mi.getComponentOrientation().isLeftToRight(), mi.getFont(),
|
||||
acceleratorFont, MenuItemLayoutHelper.useCheckAndArrow(menuItem),
|
||||
prefix);
|
||||
MenuItemLayoutHelper.LayoutResult lr = lh.layoutMenuItem();
|
||||
|
||||
paintBackground(accessor, g, mi, background);
|
||||
paintCheckIcon(g, lh, lr, holdc, foreground);
|
||||
paintIcon(g, lh, lr, holdc);
|
||||
|
||||
if (lh.getCheckIcon() != null && lh.useCheckAndArrow()) {
|
||||
Rectangle rect = lr.getTextRect();
|
||||
|
||||
rect.x += lh.getAfterCheckIconGap();
|
||||
|
||||
lr.setTextRect(rect);
|
||||
}
|
||||
if (!lh.getText().isEmpty()) {
|
||||
if (lh.getHtmlView() != null) {
|
||||
// Text is HTML
|
||||
lh.getHtmlView().paint(g, lr.getTextRect());
|
||||
} else {
|
||||
// Text isn't HTML
|
||||
paintText(accessor, g, lh.getMenuItem(),
|
||||
lr.getTextRect(), lh.getText());
|
||||
}
|
||||
}
|
||||
if (lh.getCheckIcon() != null && lh.useCheckAndArrow()) {
|
||||
Rectangle rect = lr.getAccRect();
|
||||
rect.x += lh.getAfterCheckIconGap();
|
||||
lr.setAccRect(rect);
|
||||
}
|
||||
paintAccText(g, lh, lr);
|
||||
paintArrowIcon(g, lh, lr, foreground);
|
||||
|
||||
// Restore original graphics font and color
|
||||
g.setColor(holdc);
|
||||
g.setFont(holdf);
|
||||
}
|
||||
|
||||
/**
|
||||
* Method which renders the text of the current menu item.
|
||||
* <p>
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 1997, 2007, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1997, 2025, 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
|
||||
@ -123,6 +123,25 @@ public class WindowsMenuUI extends BasicMenuUI {
|
||||
hotTrackingOn = (obj instanceof Boolean) ? (Boolean)obj : true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Paint MenuItem.
|
||||
*/
|
||||
protected void paintMenuItem(Graphics g, JComponent c,
|
||||
Icon checkIcon, Icon arrowIcon,
|
||||
Color background, Color foreground,
|
||||
int defaultTextIconGap) {
|
||||
if (WindowsMenuItemUI.isVistaPainting()) {
|
||||
WindowsMenuItemUI.paintMenuItem(accessor, g, c, checkIcon, arrowIcon,
|
||||
background, foreground,
|
||||
defaultTextIconGap, menuItem,
|
||||
getPropertyPrefix());
|
||||
return;
|
||||
}
|
||||
super.paintMenuItem(g, c, checkIcon, arrowIcon, background,
|
||||
foreground, defaultTextIconGap);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Draws the background of the menu.
|
||||
* @since 1.4
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 1997, 2006, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1997, 2025, 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
|
||||
@ -74,6 +74,23 @@ public class WindowsRadioButtonMenuItemUI extends BasicRadioButtonMenuItemUI {
|
||||
super.paintBackground(g, menuItem, bgColor);
|
||||
}
|
||||
|
||||
/**
|
||||
* Paint MenuItem.
|
||||
*/
|
||||
protected void paintMenuItem(Graphics g, JComponent c,
|
||||
Icon checkIcon, Icon arrowIcon,
|
||||
Color background, Color foreground,
|
||||
int defaultTextIconGap) {
|
||||
if (WindowsMenuItemUI.isVistaPainting()) {
|
||||
WindowsMenuItemUI.paintMenuItem(accessor, g, c, checkIcon,
|
||||
arrowIcon, background, foreground, defaultTextIconGap,
|
||||
menuItem, getPropertyPrefix());
|
||||
return;
|
||||
}
|
||||
super.paintMenuItem(g, c, checkIcon, arrowIcon, background,
|
||||
foreground, defaultTextIconGap);
|
||||
}
|
||||
|
||||
/**
|
||||
* Method which renders the text of the current menu item.
|
||||
* <p>
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 1997, 2025, 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
|
||||
@ -25,17 +25,52 @@
|
||||
|
||||
package javax.swing.plaf.basic;
|
||||
|
||||
import java.awt.*;
|
||||
import java.awt.event.*;
|
||||
import java.awt.Color;
|
||||
import java.awt.Component;
|
||||
import java.awt.Dimension;
|
||||
import java.awt.Font;
|
||||
import java.awt.FontMetrics;
|
||||
import java.awt.Graphics;
|
||||
import java.awt.Insets;
|
||||
import java.awt.Point;
|
||||
import java.awt.Rectangle;
|
||||
import java.awt.event.ActionEvent;
|
||||
import java.awt.event.InputEvent;
|
||||
import java.awt.event.MouseEvent;
|
||||
import java.beans.PropertyChangeEvent;
|
||||
import java.beans.PropertyChangeListener;
|
||||
|
||||
import javax.swing.*;
|
||||
import javax.swing.event.*;
|
||||
import javax.swing.plaf.*;
|
||||
import javax.swing.ButtonModel;
|
||||
import javax.swing.Icon;
|
||||
import javax.swing.InputMap;
|
||||
import javax.swing.JCheckBoxMenuItem;
|
||||
import javax.swing.JComponent;
|
||||
import javax.swing.JMenu;
|
||||
import javax.swing.JMenuItem;
|
||||
import javax.swing.JRadioButtonMenuItem;
|
||||
import javax.swing.KeyStroke;
|
||||
import javax.swing.LookAndFeel;
|
||||
import javax.swing.MenuElement;
|
||||
import javax.swing.MenuSelectionManager;
|
||||
import javax.swing.SwingUtilities;
|
||||
import javax.swing.UIManager;
|
||||
import javax.swing.event.MenuDragMouseEvent;
|
||||
import javax.swing.event.MenuDragMouseListener;
|
||||
import javax.swing.event.MenuKeyListener;
|
||||
|
||||
import javax.swing.event.MouseInputListener;
|
||||
import javax.swing.plaf.ComponentInputMapUIResource;
|
||||
import javax.swing.plaf.ComponentUI;
|
||||
import javax.swing.plaf.MenuItemUI;
|
||||
import javax.swing.plaf.UIResource;
|
||||
import javax.swing.text.View;
|
||||
|
||||
import sun.swing.*;
|
||||
import com.sun.java.swing.SwingUtilities3;
|
||||
import sun.swing.MenuItemCheckIconFactory;
|
||||
import sun.swing.MenuItemLayoutHelper;
|
||||
import sun.swing.SwingUtilities2;
|
||||
import sun.swing.UIAction;
|
||||
|
||||
|
||||
/**
|
||||
* BasicMenuItem implementation
|
||||
@ -561,84 +596,22 @@ public class BasicMenuItemUI extends MenuItemUI
|
||||
|
||||
private void paintIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr, Color holdc) {
|
||||
if (lh.getIcon() != null) {
|
||||
Icon icon;
|
||||
ButtonModel model = lh.getMenuItem().getModel();
|
||||
if (!model.isEnabled()) {
|
||||
icon = lh.getMenuItem().getDisabledIcon();
|
||||
} else if (model.isPressed() && model.isArmed()) {
|
||||
icon = lh.getMenuItem().getPressedIcon();
|
||||
if (icon == null) {
|
||||
// Use default icon
|
||||
icon = lh.getMenuItem().getIcon();
|
||||
}
|
||||
} else {
|
||||
icon = lh.getMenuItem().getIcon();
|
||||
}
|
||||
|
||||
if (icon != null) {
|
||||
icon.paintIcon(lh.getMenuItem(), g, lr.getIconRect().x,
|
||||
lr.getIconRect().y);
|
||||
g.setColor(holdc);
|
||||
}
|
||||
}
|
||||
SwingUtilities3.paintIcon(g, lh, lr, holdc);
|
||||
}
|
||||
|
||||
private void paintCheckIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr,
|
||||
Color holdc, Color foreground) {
|
||||
if (lh.getCheckIcon() != null) {
|
||||
ButtonModel model = lh.getMenuItem().getModel();
|
||||
if (model.isArmed() || (lh.getMenuItem() instanceof JMenu
|
||||
&& model.isSelected())) {
|
||||
g.setColor(foreground);
|
||||
} else {
|
||||
g.setColor(holdc);
|
||||
}
|
||||
if (lh.useCheckAndArrow()) {
|
||||
lh.getCheckIcon().paintIcon(lh.getMenuItem(), g,
|
||||
lr.getCheckRect().x, lr.getCheckRect().y);
|
||||
}
|
||||
g.setColor(holdc);
|
||||
}
|
||||
SwingUtilities3.paintCheckIcon(g, lh, lr, holdc, foreground);
|
||||
}
|
||||
|
||||
private void paintAccText(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr) {
|
||||
if (!lh.getAccText().equals("")) {
|
||||
ButtonModel model = lh.getMenuItem().getModel();
|
||||
g.setFont(lh.getAccFontMetrics().getFont());
|
||||
if (!model.isEnabled()) {
|
||||
// *** paint the accText disabled
|
||||
if (disabledForeground != null) {
|
||||
g.setColor(disabledForeground);
|
||||
SwingUtilities2.drawString(lh.getMenuItem(), g,
|
||||
lh.getAccText(), lr.getAccRect().x,
|
||||
lr.getAccRect().y + lh.getAccFontMetrics().getAscent());
|
||||
} else {
|
||||
g.setColor(lh.getMenuItem().getBackground().brighter());
|
||||
SwingUtilities2.drawString(lh.getMenuItem(), g,
|
||||
lh.getAccText(), lr.getAccRect().x,
|
||||
lr.getAccRect().y + lh.getAccFontMetrics().getAscent());
|
||||
g.setColor(lh.getMenuItem().getBackground().darker());
|
||||
SwingUtilities2.drawString(lh.getMenuItem(), g,
|
||||
lh.getAccText(), lr.getAccRect().x - 1,
|
||||
lr.getAccRect().y + lh.getFontMetrics().getAscent() - 1);
|
||||
}
|
||||
} else {
|
||||
// *** paint the accText normally
|
||||
if (model.isArmed()
|
||||
|| (lh.getMenuItem() instanceof JMenu
|
||||
&& model.isSelected())) {
|
||||
g.setColor(acceleratorSelectionForeground);
|
||||
} else {
|
||||
g.setColor(acceleratorForeground);
|
||||
}
|
||||
SwingUtilities2.drawString(lh.getMenuItem(), g, lh.getAccText(),
|
||||
lr.getAccRect().x, lr.getAccRect().y +
|
||||
lh.getAccFontMetrics().getAscent());
|
||||
}
|
||||
}
|
||||
SwingUtilities3.setDisabledForeground(disabledForeground);
|
||||
SwingUtilities3.setAcceleratorSelectionForeground(
|
||||
acceleratorSelectionForeground);
|
||||
SwingUtilities3.setAcceleratorForeground(acceleratorForeground);
|
||||
SwingUtilities3.paintAccText(g, lh, lr);
|
||||
}
|
||||
|
||||
private void paintText(Graphics g, MenuItemLayoutHelper lh,
|
||||
@ -657,26 +630,11 @@ public class BasicMenuItemUI extends MenuItemUI
|
||||
private void paintArrowIcon(Graphics g, MenuItemLayoutHelper lh,
|
||||
MenuItemLayoutHelper.LayoutResult lr,
|
||||
Color foreground) {
|
||||
if (lh.getArrowIcon() != null) {
|
||||
ButtonModel model = lh.getMenuItem().getModel();
|
||||
if (model.isArmed() || (lh.getMenuItem() instanceof JMenu
|
||||
&& model.isSelected())) {
|
||||
g.setColor(foreground);
|
||||
}
|
||||
if (lh.useCheckAndArrow()) {
|
||||
lh.getArrowIcon().paintIcon(lh.getMenuItem(), g,
|
||||
lr.getArrowRect().x, lr.getArrowRect().y);
|
||||
}
|
||||
}
|
||||
SwingUtilities3.paintArrowIcon(g, lh, lr, foreground);
|
||||
}
|
||||
|
||||
private void applyInsets(Rectangle rect, Insets insets) {
|
||||
if(insets != null) {
|
||||
rect.x += insets.left;
|
||||
rect.y += insets.top;
|
||||
rect.width -= (insets.right + rect.x);
|
||||
rect.height -= (insets.bottom + rect.y);
|
||||
}
|
||||
SwingUtilities3.applyInsets(rect, insets);
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@ -0,0 +1,228 @@
|
||||
/*
|
||||
* Copyright (c) 2025, 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 8348760
|
||||
* @summary Verify if RadioButtonMenuItem bullet and
|
||||
* JCheckboxMenuItem checkmark is shown if
|
||||
* JRadioButtonMenuItem and JCheckboxMenuItem
|
||||
* is rendered with ImageIcon in WindowsLookAndFeel
|
||||
* @requires (os.family == "windows")
|
||||
* @run main/manual TestRadioAndCheckMenuItemWithIcon
|
||||
*/
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.awt.Color;
|
||||
import java.awt.FlowLayout;
|
||||
import java.awt.Graphics;
|
||||
import java.awt.GridLayout;
|
||||
import java.awt.event.ActionEvent;
|
||||
import java.awt.event.ActionListener;
|
||||
import java.awt.event.KeyEvent;
|
||||
import java.awt.event.WindowAdapter;
|
||||
import java.awt.event.WindowEvent;
|
||||
import java.awt.image.BufferedImage;
|
||||
import java.util.concurrent.LinkedBlockingQueue;
|
||||
|
||||
import javax.swing.AbstractButton;
|
||||
import javax.swing.ButtonGroup;
|
||||
import javax.swing.ImageIcon;
|
||||
import javax.swing.JButton;
|
||||
import javax.swing.JCheckBoxMenuItem;
|
||||
import javax.swing.JFrame;
|
||||
import javax.swing.JMenu;
|
||||
import javax.swing.JMenuBar;
|
||||
import javax.swing.JMenuItem;
|
||||
import javax.swing.JPanel;
|
||||
import javax.swing.JRadioButtonMenuItem;
|
||||
import javax.swing.JScrollPane;
|
||||
import javax.swing.JTextArea;
|
||||
import javax.swing.KeyStroke;
|
||||
import javax.swing.SwingUtilities;
|
||||
import javax.swing.UIManager;
|
||||
|
||||
public class TestRadioAndCheckMenuItemWithIcon {
|
||||
|
||||
private static final String INSTRUCTIONS =
|
||||
"Clicking on the Menu above will show a\n"
|
||||
+ "JRadioButtonMenuItem group with 3 radiobutton menuitems\n"
|
||||
+ "and a JCheckBoxMenuItem group with 3 checkbox menuitems.\n"
|
||||
+ "\n"
|
||||
+ "First radiobutton menuitem is selected with imageicon of a red square.\n"
|
||||
+ "Second radiobutton menuitem is unselected with imageicon.\n"
|
||||
+ "Third radiobutton menuItem is unselected without imageicon.\n"
|
||||
+ "\n"
|
||||
+ "First checkbox menuitem is selected with imageicon.\n"
|
||||
+ "Second checkbox menuitem is unselected with imageicon.\n"
|
||||
+ "Third checkbox menuItem is unselected without imageicon.\n"
|
||||
+ "\n"
|
||||
+ "Verify that for first JRadioButtonMenuItem with imageicon,\n"
|
||||
+ "a bullet is shown alongside the imageicon and\n"
|
||||
+ "for first JCheckBoxMenuItem with imageicon\n"
|
||||
+ "a checkmark is shown alongside the imageicon.\n"
|
||||
+ "\n"
|
||||
+ "If bullet and checkmark is shown, test passes else fails.";
|
||||
|
||||
private static final LinkedBlockingQueue<Boolean> resultQueue = new LinkedBlockingQueue<Boolean>(1);
|
||||
|
||||
private static void endTest(boolean result) {
|
||||
SwingUtilities.invokeLater(
|
||||
new Runnable() {
|
||||
public void run() {
|
||||
try {
|
||||
resultQueue.put(result);
|
||||
} catch (Exception ignored) {
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
|
||||
|
||||
SwingUtilities.invokeLater(new Runnable() {
|
||||
public void run() {
|
||||
TestRadioAndCheckMenuItemWithIcon.doTest();
|
||||
}
|
||||
});
|
||||
|
||||
Boolean testPasses = resultQueue.take();
|
||||
|
||||
if (!testPasses) {
|
||||
throw new Exception("Test failed!");
|
||||
}
|
||||
}
|
||||
|
||||
public static void doTest() {
|
||||
BufferedImage img = new BufferedImage(16, 16, BufferedImage.TYPE_INT_ARGB);
|
||||
Graphics g = img.getGraphics();
|
||||
g.setColor(Color.red);
|
||||
g.fillRect(0, 0, img.getWidth(), img.getHeight());
|
||||
g.dispose();
|
||||
|
||||
BufferedImage img2 = new BufferedImage(64, 64, BufferedImage.TYPE_INT_ARGB);
|
||||
Graphics g2 = img2.getGraphics();
|
||||
g2.setColor(Color.red);
|
||||
g2.fillRect(0, 0, img2.getWidth(), img2.getHeight());
|
||||
g2.dispose();
|
||||
|
||||
JFrame frame = new JFrame("RadioButtonWithImageIcon");
|
||||
ImageIcon imageIcon1 = new ImageIcon(img);
|
||||
ImageIcon imageIcon2 = new ImageIcon(img2);
|
||||
AbstractButton button1;
|
||||
JRadioButtonMenuItem m1 = new JRadioButtonMenuItem("JRadioButtonMenuItem 1",
|
||||
imageIcon1);
|
||||
m1.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_F4, ActionEvent.ALT_MASK|ActionEvent.CTRL_MASK|ActionEvent.SHIFT_MASK));
|
||||
button1 = m1;
|
||||
button1.setSelected(true);
|
||||
AbstractButton button2 = new JRadioButtonMenuItem("JRadioButtonMenuItem 2", imageIcon2);
|
||||
AbstractButton button3 = new JRadioButtonMenuItem("JRadioButtonMenuItem 3");
|
||||
|
||||
ButtonGroup buttonGroup = new ButtonGroup();
|
||||
buttonGroup.add(button1);
|
||||
buttonGroup.add(button2);
|
||||
buttonGroup.add(button3);
|
||||
|
||||
AbstractButton check1 = new JCheckBoxMenuItem("JCheckBoxMenuItem 1",
|
||||
imageIcon1);
|
||||
check1.setSelected(true);
|
||||
AbstractButton check2 = new JCheckBoxMenuItem("JCheckBoxMenuItem 2", imageIcon1);
|
||||
JCheckBoxMenuItem c3;
|
||||
AbstractButton check3 = c3 = new JCheckBoxMenuItem("JCheckBoxMenuItem 3");
|
||||
c3.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_F5, ActionEvent.ALT_MASK|ActionEvent.CTRL_MASK|ActionEvent.SHIFT_MASK));
|
||||
|
||||
JMenu topLevel = new JMenu("Menu");
|
||||
|
||||
topLevel.add(button1);
|
||||
topLevel.add(button2);
|
||||
topLevel.add(button3);
|
||||
|
||||
topLevel.addSeparator();
|
||||
|
||||
topLevel.add(check1);
|
||||
topLevel.add(check2);
|
||||
topLevel.add(check3);
|
||||
|
||||
AbstractButton menuitem1 = new JMenuItem("MenuItem1");
|
||||
AbstractButton menuitem2 = new JMenuItem("MenuItem2", imageIcon1);
|
||||
topLevel.addSeparator();
|
||||
topLevel.add(menuitem1);
|
||||
topLevel.add(menuitem2);
|
||||
|
||||
JMenuBar menuBar = new JMenuBar();
|
||||
menuBar.add(topLevel);
|
||||
|
||||
frame.setJMenuBar(menuBar);
|
||||
|
||||
JTextArea instructions = new JTextArea();
|
||||
instructions.setText(INSTRUCTIONS);
|
||||
instructions.setEditable(false);
|
||||
instructions.setColumns(80);
|
||||
instructions.setRows(24);
|
||||
|
||||
JScrollPane scrInstructions = new JScrollPane(instructions);
|
||||
frame.getContentPane().setLayout(new BorderLayout());
|
||||
|
||||
frame.getContentPane().add(scrInstructions, BorderLayout.CENTER);
|
||||
|
||||
JPanel yesno = new JPanel();
|
||||
yesno.setLayout(new GridLayout(1, 2, 40, 40));
|
||||
|
||||
JButton yes = new JButton("Passes");
|
||||
JButton no = new JButton("Fails");
|
||||
yesno.add(yes);
|
||||
yesno.add(no);
|
||||
|
||||
yes.addActionListener(new ActionListener() {
|
||||
@Override
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
frame.dispose();
|
||||
endTest(true);
|
||||
}
|
||||
});
|
||||
|
||||
no.addActionListener(new ActionListener() {
|
||||
@Override
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
frame.dispose();
|
||||
endTest(false);
|
||||
}
|
||||
});
|
||||
|
||||
frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
|
||||
frame.addWindowListener(new WindowAdapter() {
|
||||
@Override
|
||||
public void windowClosed(WindowEvent e) {
|
||||
frame.dispose();
|
||||
endTest(false);
|
||||
}
|
||||
});
|
||||
|
||||
frame.getContentPane().add(yesno, BorderLayout.SOUTH);
|
||||
|
||||
frame.pack();
|
||||
frame.setLocationRelativeTo(null);
|
||||
frame.setVisible(true);
|
||||
}
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user