我希望消息框在用户更改文本字段中的值后立即出现。目前,我需要按回车键来弹出消息框。我的代码有什么问题吗?
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);
}
}
}
任何帮助都将不胜感激!
我对WindowBuilder是全新的,事实上,几年之后才回到Java,但我实现了“一些东西”,然后我想我应该查一下,遇到了这个线程。
我正在测试这个,所以,基于对这一切的新手,我确信我一定遗漏了一些东西。
下面是我所做的,其中“runTxt”是一个文本框,“runName”是类的数据成员:
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);
}
}
似乎比这里到目前为止简单得多,而且似乎是有效的,但是,因为我正在写这篇文章,我很高兴听到任何被忽视的陷阱。这是一个问题,用户可以进入和离开文本框w/o作出改变?我认为你所做的一切都是不必要的任务。
这里是@Boann的答案的Kotlin移植,这是一个很好的解决方案,对我来说一直很有效。
import java.beans.*
import javax.swing.*
import javax.swing.event.*
import javax.swing.text.*
/**
* Installs a listener to receive notification when the text of this
* [JTextComponent] is changed. Internally, it installs a [DocumentListener] on the
* text component's [Document], and a [PropertyChangeListener] on the text component
* to detect if the `Document` itself is replaced.
*
* @param changeListener a listener to receive [ChangeEvent]s when the text is changed;
* the source object for the events will be the text component
*/
fun JTextComponent.addChangeListener(changeListener: ChangeListener) {
val dl: DocumentListener = object : DocumentListener {
private var lastChange = 0
private var lastNotifiedChange = 0
override fun insertUpdate(e: DocumentEvent) = changedUpdate(e)
override fun removeUpdate(e: DocumentEvent) = changedUpdate(e)
override fun changedUpdate(e: DocumentEvent) {
lastChange++
SwingUtilities.invokeLater {
if (lastNotifiedChange != lastChange) {
lastNotifiedChange = lastChange
changeListener.stateChanged(ChangeEvent(this))
}
}
}
}
addPropertyChangeListener("document") { e: PropertyChangeEvent ->
(e.oldValue as? Document)?.removeDocumentListener(dl)
(e.newValue as? Document)?.addDocumentListener(dl)
dl.changedUpdate(null)
}
document?.addDocumentListener(dl)
}
你可以在任何文本组件上使用它,如下所示:
myTextField.addChangeListener { event -> myEventHandler(event) }
就像他的代码一样,也是公有领域。
我知道这涉及到一个非常老的问题,然而,它也给我带来了一些问题。正如kleopatra在上面的评论中回应的那样,我用JFormattedTextField解决了这个问题。然而,解决方案需要更多的工作,但更整洁。
在默认情况下,JFormattedTextField不会在字段中的每个文本更改后触发属性更改。JFormattedTextField的默认构造函数不创建格式化程序。
但是,要执行OP建议的操作,需要使用格式化程序,它将在字段的每次有效编辑之后调用commitEdit()方法。commitEdit()方法触发了我所看到的属性更改,如果没有格式化器,这将在焦点更改或按下enter键时默认触发。
详情见http://docs.oracle.com/javase/tutorial/uiswing/components/formattedtextfield.html#value。
创建一个默认的formatter (DefaultFormatter)对象,通过它的构造函数或setter方法传递给JFormattedTextField。默认格式化器的一个方法是setCommitsOnValidEdit(布尔提交),它设置格式化器在每次修改文本时触发commitEdit()方法。然后可以使用PropertyChangeListener和propertyChange()方法来获取。