我正在做这个任务,制作一个解决数独的程序。我有一个带有SudokuTextBox网格的面板,扩展了JFormattedTextField。我有一个MaskFormatter,因此每个文本框只接受一个整数。 然后在我的面板中,当密钥被关闭时,我有这个代码。
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 extends JTextField”时,更令人困惑的是,它就像一个魅力。但是我无法设置JTextField的大小以使其为正方形,并且每个文本框只能强制执行一个整数。
我错过了一些非常明显的东西吗?
答案 0 :(得分:6)
这是一个可调整大小的组件的示例,可能适用于此类游戏。虽然它不包含游戏逻辑,但它可以很好地处理输入。单击鼠标或按空格键弹出一个菜单,选项卡和数字键按预期工作。特别是,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;
}
}
答案 1 :(得分:2)
好的,现在我发现它,“掩码格式化程序的一个缺点是,从当前实现(Java 5)开始,它不支持让用户将字段恢复为空白值(初始值)在任何用户输入之前的字段,一旦他们在任何一点离开了字段。“
因为我使用的是MaskFormatter,所以无法清除该字段。
答案 2 :(得分:0)
虽然不是同一个问题,但我正在搜索像这样的(太多)问题。在我的情况下,我希望能够通过直接查找索引或使用一些微调器并创建一个字符串然后查找该字符串来填充另外两个字段(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);
}
}
}
}