簡體   English   中英

JTextPane 或 JTextField 中的文本已更改(未更改)

[英]Text changeD (not changing) in a JTextPane or 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", source.getText(), null);
            for (PropertyChangeListener l: pnlEditorFull.this.getPropertyChangeListeners())
                l.propertyChange(evt);
        }
        return true;
    }
};

當詢問驗證程序時,它會觸發一個事件 propertyChange,作為 JTextField 的“關鍵字”或 JTextArea 的“描述”。

另一方面,我有一個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()的面板上使用 function :

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 來設置更改(我最初是這樣做的),但這會留下不確定性 window 在其中,如果其他組件干擾,我很可能會丟失所有更改或更改錯誤的變量。 需要在更改完成時調用它(而不是在更改發生時),這是我能想到的唯一解決方案。

這是它應該如何工作的簡化示例(它還包括 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, "Description", 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);
        });
    }
}

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM