首頁 >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
Mary-Kate Olsen原創
2024-12-28 05:52:10254瀏覽

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中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn