【问题标题】:Modify multiple JTextFields with a DocumentListener使用 DocumentListener 修改多个 JTextField
【发布时间】:2012-10-25 19:07:45
【问题描述】:

这是我在学校的第一个 java swing 应用程序。我们不得不做一个凯撒编码器。(http://en.wikipedia.org/wiki/Caesar_cipher)

它应该知道加密和解密,即使你不按下按钮它也应该自动完成。 (焦点监听器)

它就像一场梦,直到我只需要使用“代码!”按钮。然后我们必须添加一个我创建的 DocumentListener,如下所示。如果我仅将其用于加密或仅用于解密,它就可以工作。当我将两个 JTextField 添加到同一个 DocumentListener 时,就会出现问题。我得到一个带有“AWT-EventQueue-0”标签的 IllegalStateException。 它调用了 setText 函数两次,第二次是致命的。我认为这是因为如果我在第一个 JTextField 中键入一个字符,它就是一个插入,那么当我用 setText 设置第二个 JTextField 的内容时,它也是一个插入。我们的教授说它可以很容易地用一个标志来修复,但我失败了。你能帮帮我吗?

Application.java

    package javaswinglabor;

    public class Application {
        public static void main(String[] args) {
        CaesarFrame caesarFrame = new CaesarFrame();
        }
    }

CaesarProgram.java

package javaswinglabor;

public class CaesarProgram {

    /**
     * Caesar kodolo
     * @param input
     * @param offset
     * @return kod
     */
    static String caesarCoder(String input, char offset){
        input = input.toUpperCase();
        String kod="";
        for(int i=0; i < input.length(); i++){
        if(input.charAt(i) >= 'A' && input.charAt(i) <= 'Z' ) {
            kod += (char)(65 + ((input.charAt(i) - 'A' + offset - 'A')%26) );
        }
        }
        return kod;
    }

    /**
     * Caesar dekodolo
     * @param input
     * @param offset
     * @return kod
     */
    static String caesarDecoder(String input, char offset){
    input = input.toUpperCase();
    String kod="";
    for(int i=0; i < input.length(); i++){
        if(input.charAt(i) >= 'A' && input.charAt(i) <= 'Z' ) {
        kod += (char)(65 + (input.charAt(i) - offset + 26)%26 );
        }
    }
    return kod;
    }
}

CaesarFrame.java

 package javaswinglabor;

import java.awt.FlowLayout;
import java.awt.GridLayout;
import java.awt.HeadlessException;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.FocusEvent;
import java.awt.event.FocusListener;
import javax.swing.JButton;
import javax.swing.JComboBox;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.text.AbstractDocument;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.DocumentFilter;

public class CaesarFrame extends JFrame{



    //TOP JPanel
    private JComboBox cb;
    private JTextField t1;
    private JButton b;

    //DOWN JPanel
    private JTextField t2;

    //encrypt/decrypt
    private boolean encrypt;



    public CaesarFrame() throws HeadlessException {
    super("SwingLab");
    this.setDefaultCloseOperation(EXIT_ON_CLOSE);
    this.setSize(400, 110);
    this.setResizable(true);
    this.setLayout(new GridLayout(2, 3));

    JPanel pFelso = new JPanel(new FlowLayout(FlowLayout.LEFT));
    JPanel pAlso = new JPanel(new FlowLayout(FlowLayout.LEFT));
    this.add(pFelso);
    this.add(pAlso);

    //TOP Jpanel
    cb = new JComboBox(fillObject());
    t1 = new JTextField(20);
    b = new JButton("Code!");
    pFelso.add(cb);
    pFelso.add(t1);
    pFelso.add(b);

    //DOWN JPanel
    t2 = new JTextField(20);
    //t2.setEditable(false);
    pAlso.add(new JLabel("Output:"));
    pAlso.add(t2);

    //ButtonListener
    OkButtonActionListener obl = new OkButtonActionListener();
    b.addActionListener(obl);

    //DocumentListeners
    CoderDocumentFilter cdf = new CoderDocumentFilter();
    DecoderDocumentFilter ddf = new DecoderDocumentFilter();
    ((AbstractDocument)(t1.getDocument())).setDocumentFilter((cdf));
    ((AbstractDocument)(t2.getDocument())).setDocumentFilter((ddf));

    //InputFieldKeyListener ifkl = new InputFieldKeyListener();
    //t1.addKeyListener(ifkl);

    //SpecialDocumentListener t1_dl = new SpecialDocumentListener();
    //t1.getDocument().addDocumentListener(t1_dl);
    //t2.getDocument().addDocumentListener(t1_dl);

    //FocusListener
    JTextFieldFocusListener jtfl = new JTextFieldFocusListener();
    t1.addFocusListener(jtfl);
    t2.addFocusListener(jtfl);

    this.pack();
    this.setVisible(true);
    }



    private void Coder(){
    Character c = (Character)cb.getSelectedItem();
    t2.setText( CaesarProgram.caesarCoder(t1.getText(), c.charValue()) );
    }
    private void Decoder(){
    Character c = (Character)cb.getSelectedItem();
    t1.setText( CaesarProgram.caesarDecoder(t2.getText(), c.charValue()) );
    }



    private class CoderDocumentFilter extends DocumentFilter{
    @Override public void insertString(DocumentFilter.FilterBypass fb, int offset, String text, AttributeSet attr) throws BadLocationException {
        super.insertString(fb, offset, text.toUpperCase(), attr);
        if (encrypt) {Coder();}
    }
    @Override public void replace (DocumentFilter.FilterBypass fb, int offset, int length, String text, AttributeSet attr) throws BadLocationException {
        super.replace(fb, offset, length, text.toUpperCase(), attr);
        if (encrypt) {Coder();}
    }
    @Override public void remove(DocumentFilter.FilterBypass fb, int offset, int length) throws BadLocationException{
        super.remove(fb, offset, length);
        if (encrypt) {Coder();}
    }
    }



    private class DecoderDocumentFilter extends DocumentFilter{
    @Override public void insertString(DocumentFilter.FilterBypass fb, int offset, String text, AttributeSet attr) throws BadLocationException {
        super.insertString(fb, offset, text.toUpperCase(), attr);
        if (!encrypt) {Decoder();}
    }
    @Override public void replace (DocumentFilter.FilterBypass fb, int offset, int length, String text, AttributeSet attr) throws BadLocationException {
        super.replace(fb, offset, length, text.toUpperCase(), attr);
        if (!encrypt) {Decoder();}
    }
    @Override public void remove(DocumentFilter.FilterBypass fb, int offset, int length) throws BadLocationException{
        super.remove(fb, offset, length);
        if (!encrypt) {Decoder();}
    }
    }



    /**
     * Egy 26-elemu Object tombot tolt fel az angol ABC nagybetuivel.
     * @return object[]
     */
    private static Object[] fillObject() {
    Object[] object = new Object[26];
    for(int i=0; i < 26; i++){
        object[i] = (char)(i+65);
    }
    return object;
    }



    //Button listener
    private class OkButtonActionListener implements ActionListener{
    @Override
    public void actionPerformed(ActionEvent ae) {
        if(encrypt){Coder();}
        else{Decoder();}
    }
    }



    //FocusListener
    private class JTextFieldFocusListener implements FocusListener {

    @Override public void focusGained(FocusEvent fe) {
        if(t1.isFocusOwner()) {encrypt = true;}
        else if(t2.isFocusOwner()){encrypt = false;}
    }

    @Override public void focusLost(FocusEvent fe) {}
    }



    //Previous task
    /*
    private class InputFieldKeyListener extends KeyAdapter {
    @Override public void keyPressed(KeyEvent ke) {
        super.keyPressed(ke);
        Character c = (Character)cb.getSelectedItem();
        String s = String.valueOf(ke.getKeyChar());
        t2.setText(t2.getText() + CaesarProgram.caesarCoder(s, c.charValue()) );
    }
    }
    */
}

我解决了这个问题,谢谢大家的帮助。我确信它不是最优雅或最快的解决方案,但它最终可以工作。如果有人想让它更高效或更简单/更清洁,我会很高兴。

【问题讨论】:

  • 你真的不应该尝试从DocumentListener修改文本字段,你最好使用DocumentFilter
  • 另一个问题是,您从DocumentListener 中调用setText,这导致您处于非法状态,因为正在更新文档

标签: java swing actionlistener jtextfield documentlistener


【解决方案1】:

如前所述,您应该使用DocumentFilter 而不是尝试使用DocumentListener 修改字段

下面的示例演示了基本思想。让你的解码/编码方法正常工作让我很开心,所以我自己写了 ;)

查看MDP's Weblog 了解更多示例,查看Text Component Features 了解更多信息(有一个关于文档过滤器的部分)

public class TestCeaserCipher {

    public static final int OFFSET = 13;

    public static void main(String[] args) {
        new TestCeaserCipher();
    }

    public TestCeaserCipher() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (ClassNotFoundException ex) {
                } catch (InstantiationException ex) {
                } catch (IllegalAccessException ex) {
                } catch (UnsupportedLookAndFeelException ex) {
                }

                JFrame frame = new JFrame("Test");
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.setLayout(new BorderLayout());
                frame.add(new CeaserCiperPane());
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }

        });
    }

    protected class CeaserCiperPane extends JPanel {

        private JTextField encode;
        private JTextField decode;

        public CeaserCiperPane() {
            encode = new JTextField(12);
            decode = new JTextField(12);

            setLayout(new GridBagLayout());
            GridBagConstraints gbc = new GridBagConstraints();
            gbc.gridx = 0;
            gbc.gridy = 0;
            gbc.fill = GridBagConstraints.HORIZONTAL;
            gbc.weightx = 1;
            add(encode, gbc);
            gbc.gridy++;
            add(decode, gbc);

            ((AbstractDocument) encode.getDocument()).setDocumentFilter(new EncodeFilter());
            ((AbstractDocument) decode.getDocument()).setDocumentFilter(new DecodeFilter());
        }
    }

    public class EncodeFilter extends DocumentFilter {
        @Override
        public void replace(FilterBypass fb, int offset, int length, String text, AttributeSet attrs) throws BadLocationException {
            text = encode(text);
            super.replace(fb, offset, length, text, attrs);
        }

        @Override
        public void insertString(FilterBypass fb, int offset, String text, AttributeSet attr) throws BadLocationException {
            text = encode(text);
            super.insertString(fb, offset, text, attr);

        }
    }

    public class DecodeFilter extends DocumentFilter {
        @Override
        public void replace(FilterBypass fb, int offset, int length, String text, AttributeSet attrs) throws BadLocationException {
            text = decode(text);
            super.replace(fb, offset, length, text, attrs);
        }

        @Override
        public void insertString(FilterBypass fb, int offset, String text, AttributeSet attr) throws BadLocationException {
            text = decode(text);
            super.insertString(fb, offset, text, attr);
        }
    }

    public static String encode(String text) {
        StringBuilder ciper = new StringBuilder(text.length());
        for (int index = 0; index < text.length(); index++) {
            ciper.append(TestCeaserCipher.encode(text.charAt(index), OFFSET));
        }

        return ciper.toString();
    }

    public static String decode(String text) {
        StringBuilder ciper = new StringBuilder(text.length());
        for (int index = 0; index < text.length(); index++) {
            ciper.append(TestCeaserCipher.decdoe(text.charAt(index), OFFSET));
        }

        return ciper.toString();
    }

    public static char encode(char input, int offset) {
        char encode = Character.toUpperCase(input);
        if (encode >= 'A' && encode <= 'Z') {

            int min = (int)'A';
            int max = (int)'Z';

            int value = (encode - offset);

            if (value < min) {
                value++;
                value = min - value;
                value = max - value;
            }

            encode = (char)value;

        }
        return encode;
    }

    public static char decdoe(char input, int offset) {
        char decode = Character.toUpperCase(input);
        if (decode >= 'A' && decode <= 'Z') {

            int min = (int)'A';
            int max = (int)'Z';

            int value = decode + offset;
            if (value > max) {
                value--;
                value = value - max;
                value = min + value;
            }

            decode = (char)value;

        }
        return decode;
    }

}

【讨论】:

  • 哇,非常感谢@MadProgrammer,我自己完成后才看到这个,所以解决方案不是你的,但我很感激帮助,很高兴看到,很多人们互相帮助。 (但是我无法构建您的解决方案,Overrides 有一些东西,但现在我太累了,无法搜索问题。)
【解决方案2】:

来自How to Write a Document Listener

文档监听器不应该修改文档的内容;到侦听器收到更改通知时,更改已经完成。

要解决此问题,请使用DocumentFilter 而不是DocumentListener

见这个例子Implementing a Document Filter

【讨论】:

  • 谢谢,现在很明显不要在 DocumentListener 中修改它。 (所以它只对记录有好处?)我仍然无法弄清楚解决方案:(。我是否使用过 DoucementListener?如何解决这个问题?
  • 不,请改用DocumentFilter。见上面的链接。
  • 谢谢,我花了一些时间和谷歌,但似乎我解决了。
猜你喜欢
  • 2013-04-15
  • 2011-01-18
  • 2013-10-11
  • 2017-07-16
  • 2013-05-23
  • 2013-03-10
  • 2015-06-21
  • 2016-10-13
  • 2013-02-16
相关资源
最近更新 更多