aRe*_*ess 15 java swing nimbus jpopupmenu
Nimbus通常看起来很棒,但对于某些颜色组合,结果是非最佳的.在我的情况下,a的背景JPopupMenu
不适合,这就是我想手动设置的原因.
我在Java 7上,有趣的是,Nimbus完全忽略了UIManager
(像PopupMenu.background
)中某些属性的设置.所以我唯一的选择是创建JPopupMenu
该覆盖的子类paintComponent(...)
.我知道,那很讨厌,但至少它起作用了.
但是,如果你添加一个JMenu
到另一个菜单,它嵌入它自己的实例,JPopupMenu
我无法弄清楚如何用我自己的子类替换它.
即使PopupMenuUI
为嵌入式实例分配自己也没有带来任何结果.如果直接从JPopupMenu
overriden paint(...)
方法继承而被调用,但是,无论我做了什么,都没有绘制.如果继承的javax.swing.plaf.synth.SynthPopupMenuUI
paint
甚至没有被调用,结果就是我根本没有设置自己PopupMenuUI
.
所以简单的问题是:如何JPopupMenu
使用Nimbus作为L&F,在Java 7上调整一个或者(如果这更容易)的背景颜色?
编辑:代码示例
看一下下面的代码和结果:
public static void main(final String[] args) {
try {
UIManager.setLookAndFeel(NimbusLookAndFeel.class.getCanonicalName());
UIManager.getLookAndFeelDefaults().put("PopupMenu.background", Color.GREEN);
UIManager.getLookAndFeelDefaults().put("Panel.background", Color.RED);
UIManager.getLookAndFeelDefaults().put("List.background", Color.BLUE);
} catch (ClassNotFoundException | InstantiationException
| IllegalAccessException | UnsupportedLookAndFeelException e) {
e.printStackTrace();
}
JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setSize(200,200);
JPanel panel = new JPanel(new BorderLayout());
panel.setBorder(BorderFactory.createEmptyBorder(10,10,10,10));
JList list = new JList();
panel.add(list);
frame.getContentPane().add(panel);
JPopupMenu menu = new JPopupMenu();
menu.add(new JMenuItem("A"));
menu.add(new JMenuItem("B"));
menu.add(new JMenuItem("C"));
frame.setVisible(true);
menu.show(frame, 50, 50);
}
Run Code Online (Sandbox Code Playgroud)
我知道,有人说你应该使用UIManager.put(key, value)
或UIManager.getLookAndFeelDefautls().put(key,value)
在设置L&F之前,但对我来说这并没有带来任何结果(意思是:根本没有改变默认颜色).上面的代码至少带来:
如果你用同样的事(没啥意思)发生JPopupMenu.setBackground(...)
.这是因为Nimbus使用内部画家,它根据Nimbus的原色计算颜色并忽略组件的属性.在此示例中,您可以使用以下方法作为解决方法:
JPopupMenu menu = new JPopupMenu() {
@Override
public void paintComponent(final Graphics g) {
g.setColor(Color.GREEN);
g.fillRect(0,0,getWidth(), getHeight());
}
};
Run Code Online (Sandbox Code Playgroud)
这带来了
但是,如果您插入一个JMenu
自己包装的JPopupMenu
无法覆盖,则此解决方法不起作用:
JMenu jmenu = new JMenu("D");
jmenu.add(new JMenuItem("E"));
menu.add(jmenu);
Run Code Online (Sandbox Code Playgroud)
按预期给出:
您可以JPopupMenu
使用JMenu.getPopupMenu()
但不能设置它来检索它.即使在自己的子类中重写此方法JMenu
也不会带来任何结果,因为JMenu
似乎在JPopupMenu
不使用getter 的情况下访问它的包装实例.
小智 8
一种方法是为各个JMenuItems的背景着色并使它们不透明:
JMenuItem a = new JMenuItem("A");
a.setOpaque(true);
a.setBackground(Color.GREEN);
Run Code Online (Sandbox Code Playgroud)
然后给菜单本身一个绿色边框填充其余部分:
menu.setBorder(BorderFactory.createLineBorder(Color.GREEN));
Run Code Online (Sandbox Code Playgroud)
可能有一个简单/更直接的方式,但这对我有用.
两个答案都有一些错误
和上面提到的方法来覆盖大多数影响另一个JComponents及其颜色的UIDeafaults
Nimbus拥有自己的Painter,其中一个例子就是......
来自代码
import com.sun.java.swing.Painter;
import java.awt.*;
import javax.swing.*;
public class MyPopupWithNimbus {
public MyPopupWithNimbus() {
JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setSize(200, 200);
JPanel panel = new JPanel(new BorderLayout());
panel.setBorder(BorderFactory.createEmptyBorder(10, 10, 10, 10));
JList list = new JList();
panel.add(list);
frame.getContentPane().add(panel);
JPopupMenu menu = new JPopupMenu();
menu.add(new JMenuItem("A"));
menu.add(new JMenuItem("B"));
menu.add(new JMenuItem("C"));
JMenu jmenu = new JMenu("D");
jmenu.add(new JMenuItem("E"));
menu.add(jmenu);
frame.setVisible(true);
menu.show(frame, 50, 50);
}
public static void main(String[] args) {
try {
for (UIManager.LookAndFeelInfo laf : UIManager.getInstalledLookAndFeels()) {
if ("Nimbus".equals(laf.getName())) {
UIManager.setLookAndFeel(laf.getClassName());
UIManager.getLookAndFeelDefaults().put("PopupMenu[Enabled].backgroundPainter",
new FillPainter(new Color(127, 255, 191)));
}
}
} catch (Exception e) {
e.printStackTrace();
}
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
MyPopupWithNimbus aa = new MyPopupWithNimbus();
}
});
}
}
class FillPainter implements Painter<JComponent> {
private final Color color;
FillPainter(Color c) {
color = c;
}
@Override
public void paint(Graphics2D g, JComponent object, int width, int height) {
g.setColor(color);
g.fillRect(0, 0, width - 1, height - 1);
}
}
Run Code Online (Sandbox Code Playgroud)
不是整个故事 - 但看起来像将菜单/项目的不透明度设置为true部分解决了它(正如@Derek Richard已经为在完全应用程序控制下创建的项目所做的那样):
UIDefaults ui = UIManager.getLookAndFeelDefaults();
ui.put("PopupMenu.background", GREEN);
ui.put("Menu.background", GREEN);
ui.put("Menu.opaque", true);
ui.put("MenuItem.background", GREEN);
ui.put("MenuItem.opaque", true);
Run Code Online (Sandbox Code Playgroud)
等等,适用于所有类型的项目,如radioButton/checkBox.
这仍然留下上/下灰色区域 - 不知道哪个部分负责该绘图.取消保证金有助于以挤压为代价
// this looks not so good, but without the margins above/below are still grey
ui.put("PopupMenu.contentMargins", null);
Run Code Online (Sandbox Code Playgroud)
本教程中有一个属性键列表.
归档时间: |
|
查看次数: |
6815 次 |
最近记录: |