1

MVC モデルがあり、元に戻す/やり直し機能を追加しようとしています。ユーザーは JTextField に数値を配置でき、UNDO ボタンをクリックすると、数値が消去されます。これを実装するクラスのコードを提供しました。私が得るエラーは一番下にあります。

コントロール パッケージ:

ButtonnGUIControl.java

パッケージ管理;

/** * このクラスは、GUI のボタン アクションを制御します。actionListener() メソッドを実装し、 * クリックされたボタンに基づいて、モデル クラス内の対応するコンポーネントを * 初期化します。* */

public class ButtonGUIControl implements ActionListener{
private SudokuModel model;
private ButtonGUI bgui;
final UndoManager manager = new UndoManager();



public ButtonGUIControl(SudokuModel model) {
    this.model = model;
}

/*This method will listen for user events, and act accordingly
 * */

public void actionPerformed(ActionEvent e) {
    try {   
         if(e.getActionCommand().equals("Check My Answer!")){

            model.checkGame();
        }


         else if (e.getActionCommand().equals("Undo")){

             try {

                    manager.undo();
                } catch (CannotUndoException ex) { ex.printStackTrace(); }
                finally {
                    //updateButtons();
                }                   
         }
         else if (e.getActionCommand().equals("Redo")){

             manager.redo();
             System.out.println("e.getActionCommand().equals redo");
         }

        //otherwise, it simply places the selected number in the grid
        else{

            model.setNumber(Integer.parseInt(e.getActionCommand()));
        }

    } catch (NumberFormatException e1) {}

    catch (IOException e1) {}
    catch (CannotUndoException ex) {
        System.out.println("Unable to undo: " + ex);
        ex.printStackTrace();
        }
    catch (CannotRedoException ex) {
        System.out.println("Unable to redo: " + ex);
        ex.printStackTrace();
    }    

}

public void undoableEditHappened(UndoableEditEvent evt) {

    manager.addEdit(evt.getEdit());

    //bgui.updateButtons();

}

}


ビュー パッケージ内: ButtonGUI.java

ボタンに actionListner を追加します。

RedoBttn.addActionListener(buttonController);

UndoBttn.addActionListener(buttonController);


JTextField を拡張する Grid.java では、次のようになります。

public class Grid extends JTextField {
private int x;       
private int y; 

final UndoManager manager = new UndoManager();


public Grid(int x, int y) {
    super("");
    this.x = x;
    this.y = y;
    getDocument().addUndoableEditListener(manager);



    setPreferredSize(new Dimension(40, 40));
    setBorder(BorderFactory.createLineBorder(Color.GREEN));
    setOpaque(true);
}


public void setNumber(int number, boolean userInput)  {


    if(number > 0 & !userInput){
        setText(""+number);
        setEditable(false);


    }

    //These set of numbers are the only ones that the user can undo/redo
    else if (number > 0 & userInput){
        setText(""+number);
        setEditable(true);
}
    else
    setText("");

また、ビュー パッケージの別のクラスにある各 JTextField にマウス リスナを追加しました。したがって、元に戻す/やり直しの構造は正しいと思います。プログラムを実行すると、数値を選択して JTextField(グリッド) に配置できますが、「元に戻す」を押すとすぐに次の例外が発生します。

    javax.swing.undo.CannotUndoException
at javax.swing.undo.UndoManager.undo(UndoManager.java:411)
at control.ButtonGUIControl.actionPerformed(ButtonGUIControl.java:76)
at javax.swing.AbstractButton.fireActionPerformed(AbstractButton.java:2018)
at javax.swing.AbstractButton$Handler.actionPerformed(AbstractButton.java:2341)
at javax.swing.DefaultButtonModel.fireActionPerformed(DefaultButtonModel.java:402)
at javax.swing.DefaultButtonModel.setPressed(DefaultButtonModel.java:259)
at javax.swing.plaf.basic.BasicButtonListener.mouseReleased(BasicButtonListener.java:252)
at java.awt.Component.processMouseEvent(Component.java:6505)
at javax.swing.JComponent.processMouseEvent(JComponent.java:3321)
at java.awt.Component.processEvent(Component.java:6270)
at java.awt.Container.processEvent(Container.java:2229)
at java.awt.Component.dispatchEventImpl(Component.java:4861)
at java.awt.Container.dispatchEventImpl(Container.java:2287)
at java.awt.Component.dispatchEvent(Component.java:4687)
at java.awt.LightweightDispatcher.retargetMouseEvent(Container.java:4832)
at java.awt.LightweightDispatcher.processMouseEvent(Container.java:4492)
at java.awt.LightweightDispatcher.dispatchEvent(Container.java:4422)
at java.awt.Container.dispatchEventImpl(Container.java:2273)
at java.awt.Window.dispatchEventImpl(Window.java:2713)
at java.awt.Component.dispatchEvent(Component.java:4687)
at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:707)
at java.awt.EventQueue.access$000(EventQueue.java:101)
at java.awt.EventQueue$3.run(EventQueue.java:666)
at java.awt.EventQueue$3.run(EventQueue.java:664)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDomain.java:76)
at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDomain.java:87)
at java.awt.EventQueue$4.run(EventQueue.java:680)
at java.awt.EventQueue$4.run(EventQueue.java:678)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDomain.java:76)
at java.awt.EventQueue.dispatchEvent(EventQueue.java:677)
at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:211)
at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:128)
at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:117)
at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:113)
at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:105)
at java.awt.EventDispatchThread.run(EventDispatchThread.java:90)

76行目は私が呼び出す場所です:

    manager.undo() 

try/catch ブロックで。

何が問題なのですか?それは私の元に戻す/やり直しの実装ですか、それとも何か他のものですか?

ありがとう!!!

4

2 に答える 2

3

manager.undo();次のようにラッピングしてみてください。

if(manager.canUndo())
    manager.undo();

のデフォルトのUndoListener実装はJTextField、システム更新を追加しない(つまりsetText(""+number);、UNDOキューへの呼び出し)-UIからの更新のみを追加するのではないかと思います。

編集

いくつかのテストを行った後、setText("")関数はに元に戻すを追加するUndoManagerようですので、それはあなたが遭遇している問題ではありません。したがって、元に戻すことができるアクションがあることを確認することで、エラーを防ぐことができます。ただし、で元にできるアクションが発生しない場合はUndoManager、コードの別の場所で問題が発生している可能性があります。

于 2012-10-10T19:29:12.333 に答える
1

別の方法として、 を検討してJFormattedTextFieldください。デフォルトでは、reset-field-editアクションはキーにバインドされていますが、アクションの使用方法 に示すように、ESC必要に応じてアクションを別のキーにバインドできます。クロスプラットフォームの互換性については、ここで説明されている を使用してください。KeyEvent.VK_ZgetMenuShortcutKeyMask()

于 2012-10-10T19:08:02.467 に答える