Mordan
Mordan

Reputation: 337

AWT Canvas flickering on manual resize

Just out of intellectual interest can you make a Canvas not flicker when doing a manual resize.

public class FlickerAWT extends Canvas {

public static void main(String[] args) {
 Frame f = new Frame(str);
 //this line change nothing
 //JFrame f = new JFrame(str);
 f.add(new FlickerAWT());
 f.pack();

 int frameWidth = f.getWidth();
 int frameHeight = f.getHeight();
 Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize();
 f.setLocation(screenSize.width / 2 - frameWidth / 2, screenSize.height / 2 - frameHeight / 2);
 f.setVisible(true);

 f.addWindowListener(new WindowAdapter() {
    public void windowClosing(WindowEvent e) {
      System.exit(0);
    }
    public void windowDeiconified(WindowEvent e) {
    }
    public void windowIconified(WindowEvent e) {
    }
 });
}
private Color bgColor; private Color contentColor;
Font          f   = new Font("Georgia", Font.BOLD, 16);
static String str = "AWT Canvas Resize Flickering";
public FlickerAWT() {
 Random r = new Random();
 bgColor = new Color(r.nextInt(256), r.nextInt(256), r.nextInt(256));
 contentColor = new Color(r.nextInt(256), r.nextInt(256), r.nextInt(256));
}
public Dimension getPreferredSize() {
 FontMetrics fm = getFontMetrics(f);
 return new Dimension(fm.stringWidth(str) + 20, fm.getHeight() + 10);
}
public void paint(java.awt.Graphics g) {
 g.setColor(bgColor);
 g.fillRect(0, 0, getWidth(), getHeight());
 g.setColor(contentColor);
 g.setFont(f);
 FontMetrics fm = g.getFontMetrics(f);
 int dx = getWidth() / 2 - (fm.stringWidth(str) / 2);
 int dy = getHeight() / 2 + (fm.getHeight() / 2);
 g.drawString(str, dx, dy);
}
}

You can copy paste in a Java editor and run the example.

Upvotes: 0

Views: 3751

Answers (3)

Markus A.
Markus A.

Reputation: 12742

I know this question is ancient, but it came up during my search and I meanwhile found a solution:

There are two problems:

On the one hand, the update(...) method of java.awt.Container looks like the following:

public void update(Graphics g) {
    if (isShowing()) {
        if (! (peer instanceof LightweightPeer)) {
            g.clearRect(0, 0, width, height);
        }
        paint(g);
    }
}

I.e. it calls g.clearRect(...) to erase the current content before painting its children.

Therefore, you need to override update(...) in every descendant of java.awt.Container in your view-stack, that doesn't already do so, with something like:

public void update(Graphics g) {
    if (isShowing()) paint(g);
}

Also, it seems that AWT or the JVM or whoever (haven't figured this out yet) also clears the background of the main window, independent of any Container's update-methods. To prevent this, follow @WhiteFang34's suggestion and add the following line to your code somewhere:

System.setProperty("sun.awt.noerasebackground", "true");

Only doing both of these things finally solved my flicker issues...

Upvotes: 3

WhiteFang34
WhiteFang34

Reputation: 72039

You can add this to the beginning of your main method to avoid the background flicker:

System.setProperty("sun.awt.noerasebackground", "true");

Upvotes: 7

Hovercraft Full Of Eels
Hovercraft Full Of Eels

Reputation: 285403

The key I believe is to use double buffering, and one way to possibly solve this is to use Swing which double buffers by default:

import java.awt.*;
import java.util.Random;

import javax.swing.*;

public class FlickerSwing extends JPanel {

   public static void main(String[] args) {
      java.awt.EventQueue.invokeLater(new Runnable() {
         public void run() {
            createAndShowGui();
         }
      });
   }
   private static void createAndShowGui() {
      JFrame f = new JFrame(str);
      f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
      f.add(new FlickerSwing());
      f.setLocationRelativeTo(null);
      f.pack();
      f.setVisible(true);
   }

   private Color bgColor;
   private Color contentColor;
   Font f = new Font("Georgia", Font.BOLD, 16);
   static String str = "Swing Resize Flickering";

   public FlickerSwing() {
      Random r = new Random();
      bgColor = new Color(r.nextInt(256), r.nextInt(256), r.nextInt(256));
      contentColor = new Color(r.nextInt(256), r.nextInt(256), r.nextInt(256));

      setBackground(bgColor);
   }

   public Dimension getPreferredSize() {
      FontMetrics fm = getFontMetrics(f);
      return new Dimension(fm.stringWidth(str) + 20, fm.getHeight() + 10);
   }

   @Override
   protected void paintComponent(Graphics g) {
      super.paintComponent(g);
      g.setColor(contentColor);
      g.setFont(f);
      FontMetrics fm = g.getFontMetrics(f);
      int dx = getWidth() / 2 - (fm.stringWidth(str) / 2);
      int dy = getHeight() / 2 + (fm.getHeight() / 2);
      g.drawString(str, dx, dy);
   }
}

Upvotes: 1

Related Questions