Bunun olağan cevabı "kullan DocumentListener
" dır . Ancak, her zaman bu arayüzü hantal buluyorum. Gerçekten arayüz aşırı tasarlanmış. Yalnızca bir yönteme ihtiyaç duyduğunda metnin eklenmesi, çıkarılması ve değiştirilmesi için üç yöntemi vardır: değiştirme. (Ekleme, metin içermeyen bir metnin yerine geçmesi olarak, kaldırma ise metin içermeyen bir metnin yerine geçmesi olarak görülebilir.)
Genellikle bilmek istediğiniz tek şey kutudaki metnin ne zaman değiştiğidir , bu nedenle tipik bir DocumentListener
uygulamada bir yöntemi çağıran üç yöntem vardır.
Bu nedenle, a ChangeListener
yerine daha basit kullanmanızı sağlayan aşağıdaki yardımcı programı yaptım DocumentListener
. (Java 8'in lambda sözdizimini kullanır, ancak gerekirse eski Java için uyarlayabilirsiniz.)
/**
* 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);
}
Belgeye doğrudan bir dinleyici eklemenin aksine, bu, bir metin bileşenine yeni bir belge nesnesi yüklediğiniz (nadir) vakayı işler. Buna ek olarak, Jean-Marc Astesana'nın cevabında belirtilen sorunun etrafında çalışır ve belgenin bazen ihtiyaç duyduğundan daha fazla olayı tetikler .
Her neyse, bu yöntem aşağıdaki gibi can sıkıcı kodu değiştirmenizi sağlar:
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();
}
});
İle:
addChangeListener(someTextBox, e -> doSomething());
Kod kamu malı olarak yayınlandı. İyi eğlenceler!