使用JLayer和JPanel时出现的绘图问题

5

当用户输入无效时,我希望能够绘制一个图标。我找到了 Oracle 的一个示例,并根据我的需求进行了修改。图标的绘制是正确的,但是当我更改值以使其正确时,图标并没有完全消失:在 JPanel 上绘制的部分仍然显示出来。

以下是我的代码:

import java.awt.AlphaComposite;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.Insets;
import java.awt.RenderingHints;
import java.awt.image.BufferedImage;
import java.text.NumberFormat;

import javax.swing.Icon;
import javax.swing.ImageIcon;
import javax.swing.JComponent;
import javax.swing.JFormattedTextField;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JLayer;
import javax.swing.JPanel;
import javax.swing.UIManager;
import javax.swing.plaf.LayerUI;

public class FieldValidator extends JPanel {
    private static final int ICON_SIZE = 12;
    private static final Icon ICON = createResizedIcon((ImageIcon) UIManager.getIcon("OptionPane.errorIcon"));
    public static void main(String[] args) {
        javax.swing.SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                createUI();
            }
        });
    }

    public static void createUI() {
        final JFrame f = new JFrame ("FieldValidator");

        final JComponent content = createContent();

        f.add (content);

        f.pack();
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.setLocationRelativeTo (null);
        f.setVisible (true);
    }

    private static JComponent createContent() {
        final LayerUI<JPanel> panelUI = new ValidationLayerUI();

        // Number field.
        final JLabel numberLabel = new JLabel("Number:");

        final NumberFormat numberFormat = NumberFormat.getInstance();
        final JFormattedTextField numberField = new JFormattedTextField(numberFormat) {
            /**
             * {@inheritDoc}
             */
            @Override
            public void replaceSelection(String content) {
                super.replaceSelection(content);
                getParent().repaint();
            }
        };
        numberField.setColumns(16);
        numberField.setFocusLostBehavior(JFormattedTextField.PERSIST);
        numberField.setValue(42);

        final int i = (ICON_SIZE / 2) + (ICON_SIZE % 2);
        final JPanel numberPanel = new JPanel();
        numberPanel.add(numberLabel);
        final JPanel panel = new JPanel(new GridBagLayout());
        final GridBagConstraints constr = new GridBagConstraints();
        constr.insets = new Insets(i, i, i, i);
        constr.weightx = 1;
        constr.weighty = 1;
        constr.fill = GridBagConstraints.BOTH;
        panel.add(numberField, constr);
        numberPanel.add(new JLayer<JPanel>(panel, panelUI));

        return numberPanel;
    }

    //Icon resized to 12x12
    private static Icon createResizedIcon(ImageIcon anIcon) {
        final BufferedImage result = new BufferedImage(ICON_SIZE, ICON_SIZE, BufferedImage.TYPE_INT_ARGB);
        final Graphics2D g = result.createGraphics();
        g.setComposite(AlphaComposite.Src);
        g.setRenderingHint(RenderingHints.KEY_INTERPOLATION,
                RenderingHints.VALUE_INTERPOLATION_BICUBIC);
        g.drawImage(anIcon.getImage(), 0, 0, ICON_SIZE, ICON_SIZE, null);
        g.dispose();
        return new ImageIcon(result);

    }
    static class ValidationLayerUI extends LayerUI<JPanel> {

        @Override
        public void paint (Graphics g, JComponent c) {
            super.paint (g, c);
            final JLayer jlayer = (JLayer) c;
            final JPanel panel = (JPanel) jlayer.getView();
            final JFormattedTextField ftf = (JFormattedTextField) panel.getComponent(0);
            if (!ftf.isEditValid()) {
                ICON.paintIcon(panel, g, 0, panel.getHeight() - ICON.getIconHeight());
            }
        }

    }
}

这里是屏幕截图: 初始状态都正确。
当我绘制无效的图标时,仍然一切正常。
但是当值变为正确时,只有文本字段会被重新绘制。
如何强制JPanel重新绘制?
附:我已经找到了使用JLayeredPane的方法,可以正常工作,但我想知道我的代码哪里出了问题?
1个回答

4
使用DocumentListener怎么样:
numberField.getDocument().addDocumentListener(new DocumentListener() {
  @Override public void insertUpdate(DocumentEvent e) {
    //Container c = numberField.getParent();
    Container c = SwingUtilities.getUnwrappedParent(numberField);
    if (c != null) {
      c.repaint();
    }
  }
  @Override public void removeUpdate(DocumentEvent e) {
    insertUpdate(e);
  }
  @Override public void changedUpdate(DocumentEvent e) {}
});

编辑

来自此链接的引用: AWT和Swing中的绘图

重绘管理器 Swing的RepaintManager类的目的是最大化Swing容器层次结构上的重绘处理的效率,还实现了Swing的“重新验证”机制(后者将成为单独一篇文章的主题)。它通过拦截Swing组件上的所有重绘请求(因此不再由AWT处理)并在其自己的状态下维护需要更新的内容(称为“脏区域”)来实现重绘机制。最后,它使用invokeLater()在事件分派线程上处理未决请求,如“重绘处理”一节所述(选项B)。

在这种情况下,当isEditValid()状态更改时,父级JPanel不会是脏区域,因此保留以前的Icon绘制。


1
@aterai,请问您为什么在关于JLayer的代码示例中调用updateUI/repaint()方法?实际上并不可能通过这种方式通知容器(画家)有绘画变化。 - mKorbel
谢谢您的解决方案。问题是:为什么它在文档监听器中有效,但在replaceSelection中无效? - Sergiy Medvynskyy
2
@SergiyMedvynskyy 当文本的一部分被移除时,JTextComponent#replaceSelection(...) 不会被调用。 - aterai
@aterai 奇怪但你是对的。我确信它也会在空字符串删除时被调用。我的错 :) - Sergiy Medvynskyy

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接