JTextPane 或 JTextField 中的文本更改未更改

问题描述

好的,我遇到了一个非常具体的问题。我有一个带有 JTextField 和 JTextPane 的面板(实际上是 2 个 JTextPane,但我只有一个处于活动状态)并且需要一个事件来在用户完成文本编辑后更新调用方表单中的相关变量。

JTextPane 是使用 StyledDocument 设置的,因此,我无法在每次按下键时更新其变量,因此使用 DocumentListener 不可行。

并且使用 FocusListener 也不是解决方案,因为变量(对)是使用 JList 组件选择的,并且它在更改 FocusLost 之前更改了引用。简而言之,其他组件正在干扰变量实例。

我就是这样解决问题的。

首先我创建了一个 InputVerifier

InputVerifier myInputVerifier = new InputVerifier() {
    @Override
    public boolean verify(JComponent input) {
        if ((input instanceof JTextField source) && source == tfKeywords) {
            PropertyChangeEvent evt = new PropertyChangeEvent(source,"Keywords",source.getText(),null);
            for (Propertychangelistener l: pnlEditorFull.this.getPropertychangelisteners())
                l.propertyChange(evt);
        } else if (input instanceof JTextPane source && source == ActualEditor) {
            PropertyChangeEvent evt = new PropertyChangeEvent(source,"Description",null);
            for (Propertychangelistener l: pnlEditorFull.this.getPropertychangelisteners())
                l.propertyChange(evt);
        }
        return true;
    }
};

当验证器被询问时,它会触发一个事件 propertyChange 作为 JTextField 的“Keywords”或 JTextArea 的“Description”。

另一方面,我有一个 PropertyListener

pnlEditor.addPropertychangelistener((PropertyChangeEvent evt) - > {
    if ("Keywords".equals(evt.getPropertyName()))
        myKeywords = EditorPane.getKeywords(); // actually a little more complicated
    else if ("Description".equals(evt.getPropertyName()))
        myDescription = EditorPane.getText(); // actually a little more complicated
});

为了检索文本,我在面板上使用了一个名为 getText()函数

public String getText() {
    StringWriter sw = new StringWriter();
    try {
        ActualEditor.write(sw);
    } catch (IOException ex) {
        LOGGER.log(Level.SEVERE,null,ex);
    }
    return sw.toString();
}

需要使用 ActualEditor.write(sw);,因为我需要检索文本和格式。

它奏效了。它在 FocusLost 之前和 JList 更改其所选项目之前触发。当然,我可以在选择更改时使用 JList ListSelectionListener 来设置更改(我最初是这样做的),但这会留下一个不确定的窗口,如果其他组件干扰,我很可能会丢失所有更改或更改错误的变量。需要在更改完成时调用它(而不是在更改发生时),这是我唯一能想到的解决方案。

这是它应该如何工作的简化示例(它还包括 JList):

import java.beans.PropertyChangeEvent;
import java.beans.Propertychangelistener;
import java.io.IOException;
import java.io.StringReader;
import java.io.StringWriter;
import java.lang.System.Logger;
import java.lang.System.Logger.Level;
import javax.swing.DefaultListModel;
import javax.swing.InputVerifier;
import javax.swing.JComponent;
import javax.swing.JList;
import javax.swing.JTextArea;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.ListSelectionListener;

/**
 *
 * @author luca.scarcia
 */
public class frmMain extends javax.swing.JFrame {
    
    /**
     * 
     */
    private static final long serialVersionUID = 1L;

    static final Logger LOGGER  = System.getLogger(frmMain.class.getName());
    
    private JTextArea anEditor;
    private JList<String> aList;
    private DefaultListModel<String> Model;

    public frmMain() {
        
        setDefaultCloSEOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        getContentPane().setLayout(new java.awt.GridLayout());

        aList = new JList<>();
        this.add(aList);
        Model = new DefaultListModel<>();
        Model.addElement("Element 1");
        Model.addElement("Element 2");
        Model.addElement("Element 3");
        Model.addElement("Element 4");
        
        aList.setModel(Model);
        
        anEditor = new JTextArea();
        this.add(anEditor);

        anEditor.setInputVerifier(new InputVerifier() {
            @Override
            public boolean verify(JComponent input) {
                if(input == anEditor) {
                    PropertyChangeEvent evt = new PropertyChangeEvent(anEditor,anEditor.getText(),null);
                    for(Propertychangelistener l:anEditor.getPropertychangelisteners()) {
                        l.propertyChange(evt);
                    }
                }
                return true;
            }
        });
        
        anEditor.addPropertychangelistener(new Propertychangelistener() {
            @Override
            public void propertyChange(PropertyChangeEvent evt) {
            if(aList.getSelectedindex()>=0)
                if("Description".equals(evt.getPropertyName()) && evt.getSource()==anEditor) {
                    StringWriter sw = new StringWriter();
                    try {
                        anEditor.write(sw);
                    } catch (IOException ex) {
                        LOGGER.log(Level.ERROR,ex);
                    }
                    Model.setElementAt(sw.toString(),aList.getSelectedindex());
                    aList.validate();
                }
            }
        });
        
        
        aList.addListSelectionListener(new ListSelectionListener() {
            @Override
            public void valueChanged(ListSelectionEvent e) {
                if(e.getSource() == aList) {
                    StringReader sr = new StringReader((String)aList.getSelectedValue());
                    try {
                        anEditor.read(sr,null);
                    } catch (IOException ex) {
                        LOGGER.log(Level.ERROR,ex);
                    }
                }
            }
        });
        pack();
    }

    /**
     * @param args the command line arguments
     */
    public static void main(String args[]) {
        try {
            for (javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.getInstalledLookAndFeels()) {
                if ("Nimbus".equals(info.getName())) {
                    javax.swing.UIManager.setLookAndFeel(info.getClassName());
                    break;
                }
            }
        } catch (ClassNotFoundException | InstantiationException | illegalaccessexception | javax.swing.UnsupportedLookAndFeelException ex) {
            LOGGER.log(Level.ERROR,ex);
        }

        /* Create and display the form */
        java.awt.EventQueue.invokelater(() -> {
            new frmMain().setVisible(true);
        });
    }
}

问题:有人有更好的解决方案吗?

解决方法

并且使用 FocusListener 也不是解决方案,因为变量(对)是使用 JList 组件选择的,并且它在更改 FocusLost 之前更改了引用

当焦点位于文本区域时,您可以保存所选索引。

然后您还需要确保在选择侦听器完成调整之前不要重置文本:

import java.awt.event.*;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.io.IOException;
import java.io.StringReader;
import java.io.StringWriter;
import java.lang.System.Logger;
import java.lang.System.Logger.Level;
import javax.swing.*;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.ListSelectionListener;

/**
 *
 * @author luca.scarcia
 */
public class frmMain2 extends javax.swing.JFrame {

    private JTextArea anEditor;
    private JList<String> aList;
    private DefaultListModel<String> Model;

    public frmMain2() {

        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        getContentPane().setLayout(new java.awt.GridLayout());

        aList = new JList<>();
        this.add(aList);
        Model = new DefaultListModel<>();
        Model.addElement("Element 1");
        Model.addElement("Element 2");
        Model.addElement("Element 3");
        Model.addElement("Element 4");

        aList.setModel(Model);

        anEditor = new JTextArea(5,30);
        this.add(anEditor);

        anEditor.addFocusListener( new FocusAdapter()
        {
            int selected;

            @Override
            public void focusGained(FocusEvent e)
            {
                selected = aList.getSelectedIndex();
            }

            @Override
            public void focusLost(FocusEvent e)
            {
                Model.setElementAt(anEditor.getText(),selected);
            }
        });

        aList.addListSelectionListener(new ListSelectionListener() {
            @Override
            public void valueChanged(ListSelectionEvent e) {
                if(!e.getValueIsAdjusting()) {
                    StringReader sr = new StringReader((String)aList.getSelectedValue());
                    try {
                        anEditor.read(sr,null);
                    } catch (IOException ex) {
                        System.out.println(ex);
                    }
                }
            }
        });

        pack();
    }

    /**
     * @param args the command line arguments
     */
    public static void main(String args[]) {
        try {
            for (javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.getInstalledLookAndFeels()) {
                if ("Nimbus".equals(info.getName())) {
                    javax.swing.UIManager.setLookAndFeel(info.getClassName());
                    break;
                }
            }
        } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | javax.swing.UnsupportedLookAndFeelException ex) {
        }

        /* Create and display the form */
        java.awt.EventQueue.invokeLater(() -> {
            new frmMain2().setVisible(true);
        });
    }
}