मूल्य बदलें श्रोता JTextField के लिए


215

मैं चाहता हूं कि टेक्स्ट बॉक्स में उपयोगकर्ता द्वारा मान बदलने के तुरंत बाद संदेश बॉक्स दिखाई दे। वर्तमान में, मुझे संदेश बॉक्स को पॉप आउट करने के लिए एंटर कुंजी को हिट करना होगा। क्या मेरे कोड के साथ कुछ गलत है?

textField.addActionListener(new java.awt.event.ActionListener() {
    public void actionPerformed(java.awt.event.ActionEvent e) {

        if (Integer.parseInt(textField.getText())<=0){
            JOptionPane.showMessageDialog(null,
                    "Error: Please enter number bigger than 0", "Error Message",
                    JOptionPane.ERROR_MESSAGE);
        }       
    }
}

किसी भी सहायता की सराहना की जाएगी!

जवाबों:


373

अंतर्निहित दस्तावेज़ में एक श्रोता जोड़ें, जो स्वचालित रूप से आपके लिए बनाया गया है।

// Listen for changes in the text
textField.getDocument().addDocumentListener(new DocumentListener() {
  public void changedUpdate(DocumentEvent e) {
    warn();
  }
  public void removeUpdate(DocumentEvent e) {
    warn();
  }
  public void insertUpdate(DocumentEvent e) {
    warn();
  }

  public void warn() {
     if (Integer.parseInt(textField.getText())<=0){
       JOptionPane.showMessageDialog(null,
          "Error: Please enter number bigger than 0", "Error Message",
          JOptionPane.ERROR_MESSAGE);
     }
  }
});

चेतावनी / प्रकार कास्ट के लिए अच्छा प्रारूप। एक ही पैटर्न दोहरी मात्रा (बिक्री के आंकड़े / मूल्य दर्ज या प्रदर्शित) को
अधिकतम

यह ठीक काम कर रहा है, लेकिन मेरे पास एक क्वेरी है, जब मैं टेक्स्टफील्ड में कुछ पाठ सम्मिलित करता हूं तब मैं एक विधि को कॉल करना चाहता हूं। मुझे इस बारे में अधिक जानकारी नहीं है कि यह कैसे किया जाता है ..

मैं एक JTable के साथ एक अन्य टेबल सेल पर क्लिक करते समय संपादन योग्य JComboBox से टेक्स्ट बॉक्स अपडेट नहीं प्राप्त करने के मुद्दे पर चल रहा था, और यहाँ InsertUpdate फ़ंक्शन इसे ठीक से काम करने का एकमात्र तरीका था।
विंचेला

14
"मालिश करने में त्रुटि"
ungato

51

इसका सामान्य उत्तर "उपयोग DocumentListener" है। हालांकि, मुझे हमेशा वह इंटरफ़ेस बोझिल लगता है। सच में इंटरफ़ेस अति-इंजीनियर है। पाठ के सम्मिलन, हटाने और प्रतिस्थापन के लिए इसकी तीन विधियाँ हैं, जब इसे केवल एक विधि की आवश्यकता होती है: प्रतिस्थापन। (एक सम्मिलन को कुछ पाठ के साथ बिना पाठ के प्रतिस्थापन के रूप में देखा जा सकता है, और एक निष्कासन को इस पाठ के साथ कुछ पाठ के प्रतिस्थापन के रूप में देखा जा सकता है।)

आमतौर पर आप जो भी जानना चाहते हैं वह यह है कि जब बॉक्स में पाठ बदल गया है , तो एक विशिष्ट DocumentListenerकार्यान्वयन में तीन विधियों को एक विधि कहा जाता है।

इसलिए मैंने निम्नलिखित उपयोगिता विधि बनाई, जो आपको एक के ChangeListenerबजाय एक सरल उपयोग करने देता है DocumentListener। (यह जावा 8 के लैम्ब्डा सिंटैक्स का उपयोग करता है, लेकिन यदि आवश्यक हो तो आप इसे पुराने जावा के लिए अनुकूलित कर सकते हैं।)

/**
 * Installs a listener to receive notification when the text of any
 * {@code JTextComponent} is changed. Internally, it installs a
 * {@link DocumentListener} on the text component's {@link Document},
 * and a {@link PropertyChangeListener} on the text component to detect
 * if the {@code Document} itself is replaced.
 * 
 * @param text any text component, such as a {@link JTextField}
 *        or {@link JTextArea}
 * @param changeListener a listener to receieve {@link ChangeEvent}s
 *        when the text is changed; the source object for the events
 *        will be the text component
 * @throws NullPointerException if either parameter is null
 */
public static void addChangeListener(JTextComponent text, ChangeListener changeListener) {
    Objects.requireNonNull(text);
    Objects.requireNonNull(changeListener);
    DocumentListener dl = new DocumentListener() {
        private int lastChange = 0, lastNotifiedChange = 0;

        @Override
        public void insertUpdate(DocumentEvent e) {
            changedUpdate(e);
        }

        @Override
        public void removeUpdate(DocumentEvent e) {
            changedUpdate(e);
        }

        @Override
        public void changedUpdate(DocumentEvent e) {
            lastChange++;
            SwingUtilities.invokeLater(() -> {
                if (lastNotifiedChange != lastChange) {
                    lastNotifiedChange = lastChange;
                    changeListener.stateChanged(new ChangeEvent(text));
                }
            });
        }
    };
    text.addPropertyChangeListener("document", (PropertyChangeEvent e) -> {
        Document d1 = (Document)e.getOldValue();
        Document d2 = (Document)e.getNewValue();
        if (d1 != null) d1.removeDocumentListener(dl);
        if (d2 != null) d2.addDocumentListener(dl);
        dl.changedUpdate(null);
    });
    Document d = text.getDocument();
    if (d != null) d.addDocumentListener(dl);
}

दस्तावेज़ में सीधे श्रोता को जोड़ने के विपरीत, यह (असामान्य) मामले को संभालता है जो आप एक पाठ घटक पर एक नया दस्तावेज़ ऑब्जेक्ट स्थापित करते हैं। इसके अतिरिक्त, यह जीन-मार्क एस्टेसाना के उत्तर में वर्णित समस्या के आसपास काम करता है , जहां दस्तावेज़ कभी-कभी ज़रूरत से ज़्यादा घटनाओं को आग लगा देता है।

वैसे भी, यह विधि आपको कष्टप्रद कोड को बदलने देती है जो इस तरह दिखता है:

someTextBox.getDocument().addDocumentListener(new DocumentListener() {
    @Override
    public void insertUpdate(DocumentEvent e) {
        doSomething();
    }

    @Override
    public void removeUpdate(DocumentEvent e) {
        doSomething();
    }

    @Override
    public void changedUpdate(DocumentEvent e) {
        doSomething();
    }
});

साथ में:

addChangeListener(someTextBox, e -> doSomething());

सार्वजनिक डोमेन के लिए जारी किया गया कोड। मज़े करो!


5
इसी तरह का समाधान: abstract class DocumentChangeListener implements DocumentListenerएक अतिरिक्त सार विधि के साथ बनाएं change(DocumentEvent e)जिसे आप सभी 3 अन्य तरीकों से कॉल करते हैं। मेरे लिए अधिक स्पष्ट लगता है क्योंकि यह abstract *Adapterश्रोताओं के रूप में कमोबेश उसी तर्क का उपयोग करता है।
geronimo

+1 के रूप में changedUpdateविधि स्पष्ट रूप से प्रत्येक के भीतर एक कॉल के माध्यम से आह्वान किया जाएगा insertUpdateऔर removeUpdate, यह काम पाने के लिए ..
Kais

16

बस एक इंटरफ़ेस बनाएँ, जो डॉक्यूमेंटलिस्ट को बढ़ाता है और सभी डॉक्यूमेंटलिस्ट तरीकों को लागू करता है:

@FunctionalInterface
public interface SimpleDocumentListener extends DocumentListener {
    void update(DocumentEvent e);

    @Override
    default void insertUpdate(DocumentEvent e) {
        update(e);
    }
    @Override
    default void removeUpdate(DocumentEvent e) {
        update(e);
    }
    @Override
    default void changedUpdate(DocumentEvent e) {
        update(e);
    }
}

और फिर:

jTextField.getDocument().addDocumentListener(new SimpleDocumentListener() {
    @Override
    public void update(DocumentEvent e) {
        // Your code here
    }
});

या आप लैम्ब्डा अभिव्यक्ति का उपयोग भी कर सकते हैं:

jTextField.getDocument().addDocumentListener((SimpleDocumentListener) e -> {
    // Your code here
});

1
मत भूलना कि इस समाधान जावा 8. करने से पहले सभी संस्करणों में एक अंतरफलक के बजाय एक अमूर्त वर्ग की आवश्यकता है
klaar

15

ध्यान रखें कि जब उपयोगकर्ता फ़ील्ड को संशोधित करता है, तो DocumentListener, कभी-कभी, दो ईवेंट प्राप्त कर सकता है। उदाहरण के लिए, यदि उपयोगकर्ता संपूर्ण फ़ील्ड सामग्री का चयन करता है, तो एक कुंजी दबाएं, आपको एक removeUpdate (सभी सामग्री हटा दी जाएगी) और एक सम्मिलित करें। आपके मामले में, मुझे नहीं लगता कि यह एक समस्या है, लेकिन आम तौर पर बोलना, यह है। दुर्भाग्य से, ऐसा लगता है कि JTextField को उप-वर्गीकृत किए बिना टेक्स्टफीड की सामग्री को ट्रैक करने का कोई तरीका नहीं है। यहाँ एक वर्ग का कोड है जो "पाठ" संपत्ति प्रदान करता है:

package net.yapbam.gui.widget;

import javax.swing.JTextField;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.PlainDocument;

/** A JTextField with a property that maps its text.
 * <br>I've found no way to track efficiently the modifications of the text of a JTextField ... so I developed this widget.
 * <br>DocumentListeners are intended to do it, unfortunately, when a text is replace in a field, the listener receive two events:<ol>
 * <li>One when the replaced text is removed.</li>
 * <li>One when the replacing text is inserted</li>
 * </ul>
 * The first event is ... simply absolutely misleading, it corresponds to a value that the text never had.
 * <br>Anoter problem with DocumentListener is that you can't modify the text into it (it throws IllegalStateException).
 * <br><br>Another way was to use KeyListeners ... but some key events are throw a long time (probably the key auto-repeat interval)
 * after the key was released. And others events (for example a click on an OK button) may occurs before the listener is informed of the change.
 * <br><br>This widget guarantees that no "ghost" property change is thrown !
 * @author Jean-Marc Astesana
 * <BR>License : GPL v3
 */

public class CoolJTextField extends JTextField {
    private static final long serialVersionUID = 1L;

    public static final String TEXT_PROPERTY = "text";

    public CoolJTextField() {
        this(0);
    }

    public CoolJTextField(int nbColumns) {
        super("", nbColumns);
        this.setDocument(new MyDocument());
    }

    @SuppressWarnings("serial")
    private class MyDocument extends PlainDocument {
        private boolean ignoreEvents = false;

        @Override
        public void replace(int offset, int length, String text, AttributeSet attrs) throws BadLocationException {
            String oldValue = CoolJTextField.this.getText();
            this.ignoreEvents = true;
            super.replace(offset, length, text, attrs);
            this.ignoreEvents = false;
            String newValue = CoolJTextField.this.getText();
            if (!oldValue.equals(newValue)) CoolJTextField.this.firePropertyChange(TEXT_PROPERTY, oldValue, newValue);
        }

        @Override
        public void remove(int offs, int len) throws BadLocationException {
            String oldValue = CoolJTextField.this.getText();
            super.remove(offs, len);
            String newValue = CoolJTextField.this.getText();
            if (!ignoreEvents && !oldValue.equals(newValue)) CoolJTextField.this.firePropertyChange(TEXT_PROPERTY, oldValue, newValue);
        }
    }

3
स्विंग में पहले से ही एक प्रकार का टेक्स्टफिल्ड है जो दस्तावेज़ को संपत्ति में बदलता है - इसे JFormattedTextField :-) कहा जाता है
kleopatra

11

मुझे पता है कि यह वास्तव में एक पुरानी समस्या से संबंधित है, हालांकि, इससे मुझे कुछ समस्याएं भी हुईं। जैसा कि ऊपर एक टिप्पणी में क्लियोपेट्रा ने जवाब दिया, मैंने समस्या को हल कर दिया JFormattedTextField। हालांकि, समाधान के लिए थोड़ा और काम करने की आवश्यकता होती है, लेकिन यह बकवास है।

JFormattedTextFieldडिफ़ॉल्ट ट्रिगर द्वारा एक संपत्ति परिवर्तन हर पाठ के बाद क्षेत्र में आने वाले बदलाव नहीं करता है। डिफ़ॉल्ट कंस्ट्रक्टर JFormattedTextFieldफॉर्मेटर नहीं बनाता है।

हालांकि, ओपी ने जो सुझाव दिया है, उसे करने के लिए आपको एक फॉर्मेटर का उपयोग करने की आवश्यकता है जो commitEdit()क्षेत्र के प्रत्येक वैध संपादन के बाद विधि को लागू करेगा । commitEdit()विधि यह डिफ़ॉल्ट रूप से ध्यान देने के परिवर्तन या जब कुंजी दबाने में प्रवेश पर शुरू हो रहा है, क्या मैं क्या देख सकते हैं से और संरूपक बिना संपत्ति परिवर्तन से चलाता है।

अधिक विवरण के लिए http://docs.oracle.com/javase/tutorial/uiswing/compenders/formattedtextfield.html#value देखें ।

एक डिफ़ॉल्ट फ़ॉर्मैटर ( DefaultFormatter) ऑब्जेक्ट को JFormattedTextFieldउसके निर्माता या सेटर विधि के माध्यम से पारित किया जाना चाहिए । डिफ़ॉल्ट फॉर्मैटर की एक विधि है setCommitsOnValidEdit(boolean commit), जो commitEdit()पाठ को बदलने पर हर बार विधि को ट्रिगर करने के लिए फॉर्मेटर सेट करती है। यह तो एक PropertyChangeListenerऔर propertyChange()विधि का उपयोग करके उठाया जा सकता है ।


2
textBoxName.getDocument().addDocumentListener(new DocumentListener() {
   @Override
   public void insertUpdate(DocumentEvent e) {
       onChange();
   }

   @Override
   public void removeUpdate(DocumentEvent e) {
      onChange();
   }

   @Override
   public void changedUpdate(DocumentEvent e) {
      onChange();
   } 
});

लेकिन मैं सिर्फ उपयोगकर्ता (शायद दुर्घटना पर) कुछ भी अपने कीबोर्ड पर टच नहीं करता Integer। आपको किसी भी Exceptionएस को फेंक दिया जाना चाहिए और यह सुनिश्चित करना चाहिए कि JTextFieldखाली नहीं है।


2

यदि हम रनिंग करने योग्य विधि का उपयोग करते हैं तो SwingUtilities.invokeLater () का उपयोग करते हुए दस्तावेज़ श्रोता अनुप्रयोग कभी-कभी अटक रहा है और परिणाम को अपडेट करने के लिए समय ले रहा है (मेरे प्रयोग के अनुसार)। इसके बजाय हम यहां वर्णित के रूप में टेक्स्ट फ़ील्ड परिवर्तन श्रोता के लिए कीरेल्ड घटना का भी उपयोग कर सकते हैं

usernameTextField.addKeyListener(new KeyAdapter() {
    public void keyReleased(KeyEvent e) {
        JTextField textField = (JTextField) e.getSource();
        String text = textField.getText();
        textField.setText(text.toUpperCase());
    }
});

1

यह Codemwnci का अपडेट संस्करण था। उसका कोड काफी ठीक है और त्रुटि संदेश को छोड़कर महान काम करता है। त्रुटि से बचने के लिए आपको स्थिति विवरण बदलना होगा।

  // Listen for changes in the text
textField.getDocument().addDocumentListener(new DocumentListener() {
  public void changedUpdate(DocumentEvent e) {
    warn();
  }
  public void removeUpdate(DocumentEvent e) {
    warn();
  }
  public void insertUpdate(DocumentEvent e) {
    warn();
  }

  public void warn() {
     if (textField.getText().length()>0){
       JOptionPane.showMessageDialog(null,
          "Error: Please enter number bigger than 0", "Error Massage",
          JOptionPane.ERROR_MESSAGE);
     }
  }
});

आपका अनुकूलन त्रुटि संदेश संवाद को आग लगा देता है जब भी लंबाई = 0 से किसी भी स्ट्रिंग को टेक्स्टफील्ड में दर्ज किया जाता है। तो यह मूल रूप से एक खाली स्ट्रिंग के अलावा किसी भी स्ट्रिंग है। यह अनुरोधित समाधान नहीं है।
क्लेर

0

आप नियंत्रित करने के लिए "MouseExited" का भी उपयोग कर सकते हैं। उदाहरण:

 private void jtSoMauMouseExited(java.awt.event.MouseEvent evt) {                                    
        // TODO add your handling code here:
        try {
            if (Integer.parseInt(jtSoMau.getText()) > 1) {
                //auto update field
                SoMau = Integer.parseInt(jtSoMau.getText());
                int result = SoMau / 5;

                jtSoBlockQuan.setText(String.valueOf(result));
            }
        } catch (Exception e) {

        }

    }   

6
वास्तव में नहीं: आवश्यकता तब होती है जब पाठ को बदल दिया जाता है - यह माउसइवेंट्स से असंबंधित होता है ;-)
kleopatra

0

मैं WindowBuilder के लिए बिल्कुल नया हूं, और वास्तव में, कुछ वर्षों के बाद जावा में वापस आ रहा हूं, लेकिन मैंने "कुछ" लागू किया, फिर सोचा कि मैं इसे देखूंगा और इस धागे के पार आ गया।

मैं इस परीक्षण के बीच में हूं, इसलिए, यह सब नया होने के आधार पर, मुझे यकीन है कि मुझे कुछ याद आ रहा होगा।

यहाँ मैंने क्या किया, जहां "रनटैक्स" एक टेक्स्टबॉक्स है और "रननाम" क्लास का डेटा सदस्य है:

public void focusGained(FocusEvent e) {
    if (e.getSource() == runTxt) {
        System.out.println("runTxt got focus");
        runTxt.selectAll();
    }
}

public void focusLost(FocusEvent e) {
    if (e.getSource() == runTxt) {
        System.out.println("runTxt lost focus");
        if(!runTxt.getText().equals(runName))runName= runTxt.getText();
        System.out.println("runText.getText()= " + runTxt.getText() + "; runName= " + runName);
    }
}

लगता है कि यहाँ अभी तक क्या है, और काम कर रहा है की तुलना में बहुत आसान लगता है, लेकिन, जब से मैं यह लिखने के बीच में हूँ, मैं किसी भी अनदेखी gotchas के सुनने की सराहना करता हूँ। क्या यह एक समस्या है जिसे उपयोगकर्ता एक बदलाव करते हुए टेक्स्टबॉक्स w / o दर्ज कर सकता है और छोड़ सकता है? मुझे लगता है कि आपने जो कुछ भी किया है वह एक अनावश्यक असाइनमेंट है।


-1

ActionListener (जो प्रवेश पर ट्रिगर होती है) के बजाय एक KeyListener (जो किसी भी कुंजी पर ट्रिगर होती है) का उपयोग करें


यह काम नहीं करता क्योंकि फ़ील्ड का मान ठीक से कैप्चर नहीं किया गया है, field.getText()प्रारंभिक मान लौटाता है। और घटना ( arg0.getKeyChar()) प्रमुख दबाए गए त्रुटि की जांच करता है यह निर्धारित करने के लिए आवश्यक है कि क्या आपको फ़ील्ड पाठ के साथ संक्षिप्त करना चाहिए।
ग्लैंड 19'16

@glend, आप keyTyped ईवेंट के बजाय कीरेलड इवेंट का उपयोग कर सकते हैं। इसने मेरे लिए काम किया और पूरा मूल्य प्राप्त किया।
काकुमानु शिव कृष्ण

-1

DocumentFilter ? यह आपको हेरफेर करने की क्षमता देता है।

[ http://www.java2s.com/Tutorial/Java/0240__Swing/FormatJTextFieldstexttouppercase.htm ]

माफ़ करना। J, Jython (Java में Python) का उपयोग कर रहा है - लेकिन समझने में आसान है

# python style
# upper chars [ text.upper() ]

class myComboBoxEditorDocumentFilter( DocumentFilter ):
def __init__(self,jtext):
    self._jtext = jtext

def insertString(self,FilterBypass_fb, offset, text, AttributeSet_attrs):
    txt = self._jtext.getText()
    print('DocumentFilter-insertString:',offset,text,'old:',txt)
    FilterBypass_fb.insertString(offset, text.upper(), AttributeSet_attrs)

def replace(self,FilterBypass_fb, offset, length, text, AttributeSet_attrs):
    txt = self._jtext.getText()
    print('DocumentFilter-replace:',offset, length, text,'old:',txt)
    FilterBypass_fb.replace(offset, length, text.upper(), AttributeSet_attrs)

def remove(self,FilterBypass_fb, offset, length):
    txt = self._jtext.getText()
    print('DocumentFilter-remove:',offset, length, 'old:',txt)
    FilterBypass_fb.remove(offset, length)

// (java style ~example for ComboBox-jTextField)
cb = new ComboBox();
cb.setEditable( true );
cbEditor = cb.getEditor();
cbEditorComp = cbEditor.getEditorComponent();
cbEditorComp.getDocument().setDocumentFilter(new myComboBoxEditorDocumentFilter(cbEditorComp));
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.