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