改变JButton背景的最佳方法

时间:2011-05-24 23:07:51

标签: java swing background-color jbutton

现在我使用

更改按钮的背景颜色
button.setBackground(Color.WHITE);

这是一个例子。

但是当我使用jbuttons(1000+)的大网格时,只需运行for循环来更改每个按钮背景非常非常慢。你可以看到网格慢慢变白,一帧一帧。我真的不想要这个

是否有更好的方法可以同时将网格上的每个JButton更改为相同的颜色?

这就是我制作网格的方式,所使用的数字仅仅是例如......

grid = new JPanel(new GridLayout(64, 64, 0, 0));

这是4096个按钮,将每个按钮更改为相同颜色大约需要30秒以上。

编辑1:我需要按钮是可点击的,就像当我点击一个按钮时它会变成蓝色。单击所有按钮时,将每个按钮的颜色更改为白色。现在我的工作正常,但改变每个按钮的颜色只是很慢。

编辑2:这就是我改变按钮的方式:

    new javax.swing.Timer(300, new ActionListener() {
        int counter = 0;
        public void actionPerformed(ActionEvent e) {
            if (counter >= counterMax) {
                ((Timer) e.getSource()).stop();
            }
            Color bckgrndColor = (counter % 2 == 0) ? flashColor : Color.white;
            for (JButton button : gridButton) {
                button.setBackground(bckgrndColor);
            }
            counter++;
        }
    }).start();

2 个答案:

答案 0 :(得分:6)

您看到正在重新绘制的框的事实表明关闭了双缓冲,或者按钮UI中的绘图代码使用了paintImmediately()

我使用64x64 JButton测试了您的设置,确保所有UI操作都在EDT(事件调度线程)中执行。我可以确认你看到的效果,改变所有按钮的背景花了大约1200毫秒,每个盒子立即重新粉刷。 您可以通过将网格设置为不可见之前绕过立即重绘,并在更改背景后显示:

grid.setVisible(false);
for (Component comp : grid.getComponents()) {
   comp.setBackground(color);
}
grid.setVisible(true);

这导致网格只进行一次重绘,并将时间缩短到~300ms(因子4)。

这对于频繁更新来说仍然太慢,所以最好使用自定义组件绘制网格,或者一个flyweight容器(评论中为你的问题建议的trashgod)你想让网格单元成为任意组件。

答案 1 :(得分:2)

如果只需要重新绘制可见按钮,您就可以获得相当大的好处。在下面显示的MVC方法中,每个按钮都会侦听定义其当前状态的模型。与重新绘制相比,更新模型的速度非常快。虽然启动需要几秒钟,但我看到更新了< 10毫秒处于稳定状态。它不像JTable所使用的flyweight pattern那样具有可扩展性,如here所示,但它可以提供。

import java.awt.*;
import java.awt.event.*;
import java.util.Observable;
import java.util.Observer;
import java.util.Random;
import javax.swing.*;

/** @see https://stackoverflow.com/questions/6117908 */
public class UpdateTest {

    private static final int ROW = 64;
    private static final int COL = 64;
    private static final int MAX = COL * ROW;
    private final DataModel model = new DataModel(MAX);

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                new UpdateTest().create();
            }
        });
    }

    void create() {
        JPanel panel = new JPanel();
        panel.setLayout(new GridLayout(ROW, COL));
        for (int i = 0; i < MAX; i++) {
            panel.add(new ViewPanel(model, i));
        }
        Timer timer = new Timer(1000, new ActionListener() {

            @Override
            public void actionPerformed(ActionEvent e) {
                long start = System.nanoTime();
                model.update();
                System.out.println(
                    (System.nanoTime() - start) / (1000 * 1000));
            }
        });
        JFrame f = new JFrame("JTextTest");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.add(new JScrollPane(panel), BorderLayout.CENTER);
        f.setPreferredSize(new Dimension(800, 600));
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
        timer.start();
    }

    private static class ViewPanel extends JPanel implements Observer {

        private final JButton item = new JButton();
        private DataModel model;
        private int index;

        public ViewPanel(DataModel model, int i) {
            this.model = model;
            this.index = i;
            this.add(item);
            item.setText(String.valueOf(i));
            item.setOpaque(true);
            item.setBackground(new Color(model.get(index)));
            model.addObserver(this);
        }

        @Override
        public void update(Observable o, Object arg) {
            int value = model.get(index);
            item.setBackground(new Color(value));
        }
    }

    private static class DataModel extends Observable {

        private final Random rnd = new Random();
        private final int[] data;

        public DataModel(int n) {
            data = new int[n];
            fillData();
        }

        public void update() {
            fillData();
            this.setChanged();
            this.notifyObservers();
        }

        public int get(int i) {
            return data[i];
        }

        private void fillData() {
            for (int i = 0; i < data.length; i++) {
                data[i] = rnd.nextInt();
            }
        }
    }
}