0

我有一个非常基本的小 JFrame,它带有 JToggleButtons 和子类 JPanel,它们知道如何绘制我希望它们绘制的内容。选择一个按钮会导致一个椭圆出现在相应的面板中。取消选择按钮会使图纸消失。

按下按钮,出现一个形状

不幸的是,最小化(图标化)然后恢复(去图标化)会导致任何绘制的形状消失。所以我需要手动触发重绘。问题是,如果我先显示一个消息框,我只能完成重绘(也就是说,我只看到它)。

这是 JFrame 的 deiconify 事件:

private void formWindowDeiconified(java.awt.event.WindowEvent evt)
{                                       
    //having this message makes everything work
    JOptionPane.showMessageDialog(null, "Useless message this is.");
    //but if I skip it, I'm SOL
    //what's going on?
    drawAll();
}

此方法遍历我的所有按钮,并在必要时要求重绘:

public void drawAll()
{
    for (int i=0; i<channels; i++)
    {
        if (buttons[i].isSelected())
        {
            lightboxes[i].drawMe();            
        }
    }
}

这是我的子类 JPanel:

class MyJPanel extends JPanel {

    public void drawMe()
    {
        Graphics myGraphics = this.getGraphics();
        myGraphics.fillOval(0, 0, this.getWidth(), this.getHeight());    
    }

    public void unDraw()
    {
        this.invalidate();
        this.repaint();
    }
}
4

3 回答 3

4

一旦由RepaintManager. 问题是你没有像你应该的那样执行自定义绘画......

这不是如何进行自定义绘画...

public void drawMe()
{
    Graphics myGraphics = this.getGraphics();
    myGraphics.fillOval(0, 0, this.getWidth(), this.getHeight());    
}

getGraphics可以返回null并且充其量只是图形状态的快照。

Swing 中的绘画可能随时发生,原因有很多,其中大部分是您无法控制的(也不应该在意)。

您的工作只是响应这些重绘请求并更新您的组件状态。

Swing 有一个详细的绘制链,它会自动调用并且您可以使用。

您应该paintComponent在此方法中覆盖并执行所有绘画

查看在 AWT 和 Swing中执行自定义绘画和绘画以了解更多详细信息

于 2013-06-26T23:47:41.077 回答
1

首先,为了速度,我会使用双缓冲。最好将图形绘制到屏幕外,并在绘图完成后将它们显示到屏幕上。下面应该给你排序。

public class MyPanel extends JPanel {
    private BufferedImage buffer;
    private Graphics2D canvas;

    @Override
    public void paintComponent(Graphics g) {
        if(buffer == null) {
            buffer = new BufferedImage(getWidth(), getHeight(), BufferedImage.TYPE_INT_RGB);
            canvas = buffer.createGraphics();
        }
        canvas.fillOval(0, 0, this.getWidth(), this.getHeight());
        g.drawImage(buffer, 0, 0, this);
    }
}
于 2013-06-27T01:26:33.417 回答
1

我只是提供这个答案,以便人们可以看到我最终做了什么。大家指出的主要教训是使用组件的paintComponent。有关您自己可能遇到的问题,请参阅评论。

编辑:更新以反映来自 MadProgrammer 的评论。

//with help from Slihp and MadProgrammer on StackOverflow
//http://stackoverflow.com/q/17331986/1736461
class MyJPanel extends JPanel {

    private boolean buttonSelected = false;

    @Override
    public void paintComponent(Graphics g) {
        //make sure the background gets painted (wacky results otherwise)        
        super.paintComponent(g);

        //now if the corresponding toggle button is on, plop on the circle
        if (buttonSelected)
        {
            g.fillOval(0, 0, this.getWidth(), this.getHeight());
        }        
    }

    //an action listener for the button calls this
    public void setButtonSelected(boolean buttonStateIn)
    {
        buttonSelected = buttonStateIn;    
    }
}

我也对按钮进行了子类化,因此我可以从事件处理程序中获取它的“ID”:

class MyJToggleButton extends JToggleButton
{       
    private int whoAmI;

    public MyJToggleButton(int whoAmIn)
    {
        //the string given to the constructor becomes the button's label
        //("1", "2", "3", etc..)
        super(Integer.toString(whoAmIn + 1));
        whoAmI = whoAmIn;      
    }

    public int getWho()
    {
        return whoAmI;
    }
}

制作按钮的 JFrame 代码:

private void makeButtons(int howMany)
    {
        buttons = new MyJToggleButton[howMany];
        for (int i=0; i<howMany; i++)
        {
            buttons[i] = new MyJToggleButton(i);
            buttons[i].addActionListener(new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent evt) {   
                    //find out which button this is
                    MyJToggleButton button = (MyJToggleButton) evt.getSource();
                    int which = button.getWho();
                    //send the button state to the corresponding lightbox
                    lightboxes[which].setButtonSelected(button.isSelected());
                    //trigger its redrawing
                    lightboxes[which].invalidate();
                    lightboxes[which].repaint();
                }
            });
            this.add(buttons[i]);
        }
    }

这是我唯一需要做的手动重绘——调整大小和重新显示以及所有其他有趣的事情最终都会出现在paintComponent上,它只需要知道它的按钮是否被按下就知道要做什么。超级干净,正是我想要的。

于 2013-06-27T23:19:07.690 回答