当它(表单)包含JFormattedTextField
时,我使用表单默认按钮遇到问题。
在具有此类字段的表单上,如果它恰好具有焦点(并且已更改),则必须按确定两次以使“默认”按钮“按下”。我想我知道它为什么会发生 - 这是因为在Commit处理中首先使用了Enter。
我还能够解决方法 - 如果你改变Formatter
来提交每个有效的编辑,那么你将得到正确的行为,但这a)迫使你指定格式化器的明确性,并且b)它不是可以恢复为“旧”值(例如,使用Escape或以编程方式)。
下面的代码演示了:当你在每个编辑的顶部提交时运行它字段并使用单个Enter(但你无法恢复),底部的字段允许恢复,但需要两个输入如果编辑。
import java.awt.BorderLayout;
import java.awt.EventQueue;
import java.util.Date;
import javax.swing.JButton;
import javax.swing.JFormattedTextField;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.border.EmptyBorder;
import javax.swing.text.DateFormatter;
public class ExFrame extends JFrame {
private JPanel contentPane;
/**
* Launch the application.
*/
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
public void run() {
try {
ExFrame frame = new ExFrame();
frame.setVisible(true);
} catch (Exception e) {
e.printStackTrace();
}
}
});
}
/**
* Create the frame.
*/
public ExFrame() {
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setBounds(100, 100, 450, 300);
contentPane = new JPanel();
contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
contentPane.setLayout(new BorderLayout(0, 0));
setContentPane(contentPane);
JFormattedTextField ff_1, ff_2;
//ff_1 has modified behavior of commit on each (valid) edit
DateFormatter f=new DateFormatter();
f.setCommitsOnValidEdit(true);
ff_1 = new JFormattedTextField(f);
ff_1.setValue(new Date());
//ff_2 has default behavior
ff_2 = new JFormattedTextField(new Date());
contentPane.add(ff_1, BorderLayout.NORTH);
contentPane.add(ff_2, BorderLayout.SOUTH);
JButton btnDefault = new JButton("I am default button");
contentPane.add(btnDefault, BorderLayout.CENTER);
getRootPane().setDefaultButton(btnDefault);
}
}
所以问题是:是否有办法让JFormattedTextField
提交提交(因此输入 已验证但只有一次)和如果成功验证,激活默认按钮(单击)?
答案 0 :(得分:2)
基本技巧是愚弄keyBinding机制,相信keyStroke不是由textField处理的。为了实现这一点,我们需要子类化JFormattedTextField并覆盖其processKeyBindings以返回false(意思是:“无法使用”)。
类似的东西:
JFormattedTextField multiplex = new JFormattedTextField() {
KeyStroke enter = KeyStroke.getKeyStroke("ENTER");
@Override
protected boolean processKeyBinding(KeyStroke ks, KeyEvent e,
int condition, boolean pressed) {
boolean processed = super.processKeyBinding(ks, e, condition,
pressed);
if (processed && condition != JComponent.WHEN_IN_FOCUSED_WINDOW
&& enter.equals(ks)) {
// Returning false will allow further processing
// of the bindings, eg our parent Containers will get a
// crack at them and f.i. route to a default button.
return !isEditValid();
}
return processed;
}
};
multiplex.setValue(new Date());
答案 1 :(得分:1)
对于日期/数字实例(在大多数情况下)更好地使用JSpinner而不是JFormattedTextField,那么您只能使用Locale
或{DateSpinnerModel进行设置{1}},
对于SimpleDateFormat
JSpinner
实例(适用于Number
),您必须添加Document for removing/filtering for unwanted chars
答案 2 :(得分:0)
Enter通常用于接受当前值,因此 输入JFTF也是如此。 你可以通过这样做来禁用它:
ff_1.getInputMap().put(KeyStroke.getKeyStroke("ENTER"), new Object());
ff_2.getInputMap().put(KeyStroke.getKeyStroke("ENTER"), new Object());
再见并且很酷;)
答案 3 :(得分:0)
您的答案都没有解决我的应用问题。但是,在我的情况下,似乎问题解决方案更简单:
private void jTextField1ActionPerformed(java.awt.event.ActionEvent evt) {
add(jTextField1); //this reacts after the "ENTER" gets pressed
jButton1.doClick(); //this activates the button with Enter
jTextField1.setText(""); //this removes the text from a text-field
jTextField1.grabFocus(); //this sets a cursor within a text-field
//Whenever the Enter is pressed, after the typed text, the action is performed again.
}