我在我的程序中使用了一些JFormattedTextFields。由于某些原因,当单击文本字段后文本字段获得焦点时,插入符号位置始终跳转到左侧(位置0)。我希望插入符号最终位于用户点击的位置。因此,如果我在两个数字之间点击,则插入符应该在这两个数字之间。
所以我实现了一个FocusListener,它将获得点击位置并在那里设置插入位置。
FocusListener focusListener = new FocusListener(){
public void focusGained(FocusEvent evt) {
JFormettedTextField jftf = (JFormattedTextField) evt.getSource();
//This is where the caret needs to be.
int dot = jftf.getCaret().getDot();
SwingUtilities.invokeLater( new Runnable() {
public void run() {
'the textField that has focus'.setCaretPosition('Some how get the evt or dot');
}
});
}
public void focusLost (FocusEvent evt) {}
});
我尝试了许多让他上班的事情。我尝试过使用final关键字,但是只能用于单个文本字段。
我在焦点侦听器中使用了set / get方法来分配当前对象,但我不确定如何使其“安全”(例如,它们是否需要同步?)。
也许我遗失了什么?
答案 0 :(得分:10)
您需要使用MouseListener:
MouseListener ml = new MouseAdapter()
{
public void mousePressed(final MouseEvent e)
{
SwingUtilities.invokeLater(new Runnable()
{
public void run()
{
JTextField tf = (JTextField)e.getSource();
int offset = tf.viewToModel(e.getPoint());
tf.setCaretPosition(offset);
}
});
}
};
formattedTextField.addMouseListener(ml);
答案 1 :(得分:7)
这实际上发生在AbstractFormatter.install(JFormattedTextField)
中,当字段获得焦点时会调用它。
我不确定为什么会这样设计,但你可以覆盖这种行为(只要你的格式化程序不改变字段中字符串的长度。)
示例(假设字段值为int
):
class IntFormatter extends AbstractFormatter {
@Override
public void install(final JFormattedTextField ftf) {
int prevLen = ftf.getDocument().getLength();
int savedCaretPos = ftf.getCaretPosition();
super.install(ftf);
if (ftf.getDocument().getLength() == prevLen) {
ftf.setCaretPosition(savedCaretPos);
}
}
public Object stringToValue(String text) throws ParseException {
return Integer.parseInt(text);
}
public String valueToString(Object value) throws ParseException {
return Integer.toString(((Number) value).intValue());
}
}
请注意,这与默认的Integer
格式化程序不同。默认格式化程序使用DecimalFormat
分隔数字组,例如"1,000,000"
。这使得任务变得更难,因为它改变了字符串的长度。
答案 2 :(得分:1)
我认为有点改进了finnw的解决方案。例如:
public static void main(String[] args) {
NumberFormat format = NumberFormat.getInstance();
NumberFormatter formatter = new NumberFormatter(format) {
@Override
public void install(JFormattedTextField pField) {
final JFormattedTextField oldField = getFormattedTextField();
final int oldLength = pField.getDocument().getLength();
final int oldPosition = pField.getCaretPosition();
super.install(pField);
if (oldField == pField && oldLength == pField.getDocument().getLength()) {
pField.setCaretPosition(oldPosition);
}
}
};
JFormattedTextField field = new JFormattedTextField(formatter);
field.setValue(1234567890);
JOptionPane.showMessageDialog(null, field);
}
答案 3 :(得分:0)
如果您不能或不想覆盖格式化程序:
import java.awt.event.FocusEvent;
import java.awt.event.FocusListener;
import java.text.NumberFormat;
import javax.swing.JFormattedTextField;
import javax.swing.JOptionPane;
import javax.swing.SwingUtilities;
import javax.swing.text.NumberFormatter;
public class TestJFormattedTextFieldFocus {
public static void main(String[] args) {
NumberFormat format = NumberFormat.getInstance();
NumberFormatter formatter = new NumberFormatter(format);
JFormattedTextField field = new JFormattedTextField(formatter);
field.addFocusListener(new FocusListener() {
@Override
public void focusLost(FocusEvent e) {
}
@Override
public void focusGained(FocusEvent e) {
restoreCaretPosition();
}
private void restoreCaretPosition() {
int caretPosition = field.getCaretPosition();
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
field.setCaretPosition(caretPosition);
}
});
}
});
field.setValue(1234567890);
JOptionPane.showMessageDialog(null, field);
}
}