首页 > Java > java教程 > 为什么在 Swing 中添加并关注 JTextField 而不是双缓冲动画 JPanel 会导致在使用 Java 1.6 的 Mac OS X 10.5 上出现闪烁问题?

为什么在 Swing 中添加并关注 JTextField 而不是双缓冲动画 JPanel 会导致在使用 Java 1.6 的 Mac OS X 10.5 上出现闪烁问题?

Mary-Kate Olsen
发布: 2024-12-28 05:52:10
原创
211 人浏览过

Why does adding and focusing on a JTextField over a double-buffered animated JPanel in Swing cause flickering issues on Mac OS X 10.5 with Java 1.6?

JPanel 上活动绘图之上的 JTextFields,线程问题

Swing 可用于在顶部创建适当的多缓冲渲染环境其中可以添加 Swing 用户界面元素。在本例中,我们在背景上绘制了一个动画红色矩形。背景不需要每帧都更新,因此我们将其渲染到 BufferedImage 上,并仅重绘清除矩形先前位置所需的部分。

到目前为止一切顺利;动画流畅,CPU占用率低,无闪烁。然后我们向 Jpanel 添加一个 JTextField(通过单击屏幕上的任意位置),并通过单击文本框内部来聚焦它。现在,每次光标闪烁时,清除矩形的先前位置都会失败,请参见下图。

我们很好奇是否有人知道为什么会发生这种情况(Swing 不是线程安全的?正在绘制的图像异步?)以及在什么方向寻找可能的解决方案。

这是在 Mac OS 10.5、Java 1.6 上。

NewTest 扩展J面板;但是因为每次调用paintComponent()时你并没有绘制每个像素,所以你需要调用超类的方法并删除旧的绘图:

@Override
protected void paintComponent(Graphics g) {
    super.paintComponent(g);
    int width = this.getWidth();
    int height = this.getHeight();
    g.setColor(Color.black);
    g.fillRect(0, 0, width, height);
    ...
}
登录后复制

附录:正如你所注意到的,设置背景构造函数中的 color 不需要在 PaintComponent() 中填充面板,而 super.paintComponent() 允许文本字段正确运行。正如您所观察到的,提议的解决方法很脆弱。相反,应简化代码并根据需要进行优化。例如,您可能不需要复杂的插图、额外的缓冲区和组件侦听器。

附录 2:请注意 super.paintComponent() 调用 UI 委托的 update() 方法,“该方法填充指定的组件”及其背景颜色(如果其不透明属性为真)。”您可以使用 setOpaque(false) 来排除这种情况。

代码的进一步改进版本,具有一些附加功能和优化:

import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GraphicsConfiguration;
import java.awt.GraphicsEnvironment;
import java.awt.Rectangle;
import java.awt.Transparency;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.ComponentAdapter;
import java.awt.event.ComponentEvent;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.image.BufferedImage;
import java.util.Random;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.Timer;

/** @see http://stackoverflow.com/questions/3256941 */
public class AnimationTest extends JPanel implements ActionListener {

    private static final int WIDE = 640;
    private static final int HIGH = 480;
    private static final int RADIUS = 25;
    private static final int FRAMES = 24;
    private final Timer timer = new Timer(20, this);
    private final Rectangle rect = new Rectangle();
    private BufferedImage background;
    private int index;
    private long totalTime;
    private long averageTime;
    private int frameCount;

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

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

    private void create() {
        JFrame f = new JFrame("AnimationTest");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.add(this);
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
        timer.start();
    }

    public AnimationTest() {
        super(true);
        this.setOpaque(false);
        this.setPreferredSize(new Dimension(WIDE, HIGH));
        this.addMouseListener(new MouseHandler());
        this.addComponentListener(new ComponentHandler());
    }

    @Override
    protected void paintComponent(Graphics g) {
        long start = System.nanoTime();
        super.paintComponent(g);
        int w = this.getWidth();
        int h = this.getHeight();
        g.drawImage(background, 0, 0, this);
        double theta = 2 * Math.PI * index++ / 64;
        g.setColor(Color.blue);
        rect.setRect(
            (int) (Math.sin(theta) * w / 3 + w / 2 - RADIUS),
            (int) (Math.cos(theta) * h / 3 + h / 2 - RADIUS),
            2 * RADIUS, 2 * RADIUS);
        g.fillOval(rect.x, rect.y, rect.width, rect.height);
        g.setColor(Color.white);
        if (frameCount == FRAMES) {
            averageTime = totalTime / FRAMES;
            totalTime = 0; frameCount = 0;
        } else {
            totalTime += System.nanoTime() - start;
            frameCount++;
        }
        String s = String.format("%1.3f", averageTime / 1000000d);
        g.drawString(s, 5, 16);
    }

    @Override
    public void actionPerformed(ActionEvent e) {
        this.repaint();
    }

    private class MouseHandler extends MouseAdapter {

        @Override
        public void mousePressed(MouseEvent e) {
            super.mousePressed(e);
            JTextField field = new JTextField("test");
            Dimension d = field.getPreferredSize();
            field.setBounds(e.getX(), e.getY(), d.width, d.height);
            add(field);
        }
    }

    private class ComponentHandler extends ComponentAdapter {

        private final GraphicsEnvironment ge =
            GraphicsEnvironment.getLocalGraphicsEnvironment();
        private final GraphicsConfiguration gc =
            ge.getDefaultScreenDevice().getDefaultConfiguration();
        private final Random r = new Random();

        @Override
        public void componentResized(ComponentEvent e) {
            super.componentResized(e);
            int w = getWidth();
            int h = getHeight();
            background = gc.createCompatibleImage(w, h, Transparency.OPAQUE);
            Graphics2D g = background.createGraphics();
            g.clearRect(0, 0, w, h);
            g.setColor(Color.green.darker());
            for (int i = 0; i < 128; i++) {
                g.drawLine(w / 2, h / 2, r.nextInt(w), r.nextInt(h));
            }
            g.dispose();
            System.out.println("Resized to " + w + " x " + h);
        }
    }
}
登录后复制

以上是为什么在 Swing 中添加并关注 JTextField 而不是双缓冲动画 JPanel 会导致在使用 Java 1.6 的 Mac OS X 10.5 上出现闪烁问题?的详细内容。更多信息请关注PHP中文网其他相关文章!

来源:php.cn
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
作者最新文章
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板