JFormattedTextField没有正确清除
我正在做这个任务,做一个解决数独的程序。我有一个SudokuTextBox网格扩展JFormattedTextField的面板。我有一个MaskFormatter,因此它只接受每个文本框一个整数。 然后在我的面板中,我有这个代码,当一个关键是相关的。JFormattedTextField没有正确清除
public void keyReleased(KeyEvent e) {
SudokuTextBox tb = (SudokuTextBox) e.getSource();
int row = tb.getRow();
int col = tb.getCol();
int value = toInteger(tb.getText());
//System.out.println(value);
if(sudoku.isValid(row, col, value)) {
sudoku.set(row, col, value);
}
else {
sudoku.set(row, col, 0);
tb.setText(null);
}
tb.setCaretPosition(0);
sudoku.print();
}
的事情是,如果我在文本框中放一个有效的值,然后我回去,并输入一个无效的值(由数独的规则)文本框被清除。但是当我向前看时,前面的有效值显示在文本框中。 包含所有已输入数字的我的sudokumatrix会清除它应该显示的值,因此它只在相应的文本框中显示。
当我将“SudokuTextBox扩展JFormattedTextField”改为“SudokuTextBox扩展JTextField”时,使事情更加令人困惑,它像一个魅力。但我不能设置JTextField的大小,以便它是方形的,我不能强制每个文本框只有一个整数。
我是否错过了一些非常明显的东西?
好,现在我发现它,“掩码格式化程序的一个缺点是,从当前实现(Java 5)开始,它不支持让用户将字段还原为空值(初始值任何用户输入之前的字段值)一旦他们在任何点离开了该字段。“
所以,因为我使用MaskFormatter我无法清除该字段。
下面是一个可调整组件的例子,可能适合这样的游戏。虽然它不包含游戏逻辑,但它可以很好地处理输入。单击鼠标或按下空格键弹出菜单,并且标签和数字键按预期工作。特别是,Digit.EMPTY
是有效的值。
import java.awt.*;
import java.awt.event.*;
import java.awt.image.BufferedImage;
import java.util.EnumSet;
import javax.swing.*;
/** @see http://stackoverflow.com/questions/4148336 */
public class CellTest extends JPanel {
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
//@Override
public void run() {
createGUI();
}
});
}
public static void createGUI() {
JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new GridLayout(3, 3));
for (Digit d : Digit.digits) {
frame.add(new CellTest(d));
}
frame.pack();
frame.setVisible(true);
}
CellTest(Digit digit) {
this.setLayout(new BorderLayout());
this.setBorder(BorderFactory.createLineBorder(Color.black, 1));
this.setBackground(new Color(0x00e0e0));
JLabel candidates = new JLabel("123456789");
candidates.setHorizontalAlignment(JLabel.CENTER);
this.add(candidates, BorderLayout.NORTH);
JDigit cellValue = new JDigit(digit);
add(cellValue, BorderLayout.CENTER);
}
}
class JDigit extends JButton {
private static final int SIZE = 128;
private static final int BASE = SIZE/32;
private static final Font FONT = new Font("Serif", Font.BOLD, SIZE);
private JPopupMenu popup = new JPopupMenu();
private Digit digit;
private Image image;
private int width, height;
public JDigit(Digit digit) {
this.digit = digit;
this.image = getImage(digit);
this.setPreferredSize(new Dimension(64, 64));
this.setBackground(new Color(0xe0e000));
this.setForeground(Color.black);
this.setBorderPainted(false);
this.setAction(new ButtonAction());
this.addFocusListener(new FocusHandler());
for (Digit d : Digit.values()) {
Action select = new SelectAction(d);
JMenuItem item = new JMenuItem(select);
getInputMap().put(KeyStroke.getKeyStroke(
KeyEvent.VK_0 + d.value(), 0), d.toString());
getInputMap().put(KeyStroke.getKeyStroke(
KeyEvent.VK_NUMPAD0 + d.value(), 0), d.toString());
getActionMap().put(d.toString(), select);
popup.add(item);
}
}
public Digit getDigit() {
return digit;
}
public void setDigit(Digit digit) {
this.digit = digit;
this.image = getImage(digit);
this.repaint();
}
@Override
protected void paintComponent(Graphics g) {
int w = this.getWidth();
int h = this.getHeight();
g.setColor(this.getBackground());
int dx1 = w * width/height/4;
int dx2 = w - dx1;
g.fillRect(dx1, 0, dx2 - dx1, h);
g.drawImage(image,
dx1, 0, dx2, h,
0, 0, width, height, null);
}
private Image getImage(Digit digit) {
BufferedImage bi = new BufferedImage(
SIZE, SIZE, BufferedImage.TYPE_INT_ARGB);
Graphics2D g2d = bi.createGraphics();
g2d.setRenderingHint(
RenderingHints.KEY_ANTIALIASING,
RenderingHints.VALUE_ANTIALIAS_ON);
g2d.setColor(this.getForeground());
g2d.setFont(FONT);
FontMetrics fm = g2d.getFontMetrics();
width = fm.stringWidth(digit.toString());
height = fm.getAscent();
g2d.drawString(digit.toString(), 0, height - BASE);
g2d.dispose();
return bi;
}
private class ButtonAction extends AbstractAction {
//@Override
public void actionPerformed(ActionEvent e) {
popup.show(JDigit.this, getWidth() - width, getHeight()/2);
}
}
private class SelectAction extends AbstractAction {
private Digit digit;
public SelectAction(Digit digit) {
this.digit = digit;
this.putValue(Action.NAME, digit.toString());
}
//@Override
public void actionPerformed(ActionEvent e) {
setDigit(digit);
}
}
private class FocusHandler implements FocusListener {
private Color background = getBackground();
//@Override
public void focusGained(FocusEvent e) {
setBackground(background.brighter());
}
//@Override
public void focusLost(FocusEvent e) {
setBackground(background);
}
}
}
enum Digit {
EMPTY(0, " "), ONE(1, "1"), TWO(2, "2"), THREE(3, "3"), FOUR(4, "4"),
FIVE(5, "5"), SIX(6, "6"), SEVEN(7, "7"), EIGHT(8, "8"), NINE(9, "9");
public static EnumSet<Digit> digits = EnumSet.range(Digit.ONE, Digit.NINE);
private int i;
private String s;
Digit(int i, String s) {
this.i = i;
this.s = s;
}
@Override
public String toString() {
return s;
}
public int value() {
return i;
}
}
虽然不是同一个问题,我是通过这样一个(太多)的问题进行搜索。在我的情况下,我希望能够通过直接查找索引或使用一些spinners并创建一个字符串然后查看该字符串来填充另外两个字段(jtfStarePatReq和jtfStarePatFound,它们是JTextFields)(好吧,也许这也太过了含糊不清,但我认为这是足够的上下文)。我想要的是,如果用户删除或清除了JFormattedTextField jftfStareOpsIndex中的值,那么其他两个字段也将被清除。我在JFormattedTextField上使用.isEmpty()方法来决定是使用该字段还是使用较长的计算搜索方法。所以我需要它是空的,如果有人从查找自己的索引到让软件搜索索引。无论如何,我试图捕获commitEdit()异常,并将该值设置为null,它似乎可以做到这一点。
public class stareOpsIndexListener extends KeyAdapter implements FocusListener {
public void focusGained(FocusEvent e) {
}
public void focusLost(FocusEvent e) {
try {
JFormattedTextField jftf = (JFormattedTextField) e.getComponent();
jftf.commitEdit();
updateStareOpsIndex(jftf);
} catch (ParseException ex) {
jtfStarePatReq.setText("");
jtfStarePatFound.setText("");
jftfStareOpsIndex.setValue(null);
}
}
public void keyPressed(KeyEvent e) {
int key = e.getKeyCode();
if (key == KeyEvent.VK_ENTER) {
try {
JFormattedTextField jftf = (JFormattedTextField) e.getComponent();
jftf.commitEdit();
updateStareOpsIndex(jftf);
} catch (ParseException ex) {
jtfStarePatReq.setText("");
jtfStarePatFound.setText("");
jftfStareOpsIndex.setValue(null);
}
}
}
}
对不起,我不接受你的答案。然而,我更感兴趣的是为什么我无法让JFormattedTextField做我想做的事,而不是如何推出我自己的组件。虽然谢谢! – evading 2010-11-13 12:03:53
@refuser:没问题;我正在探索关键绑定。 – trashgod 2010-11-13 14:46:18