2

コードの醜さを最小限に抑えながら、Swing プログラムにデバッグ フックを記述して、階層内のどのコンポーネントが実際に各 KeyStroke またはマウス クリックを処理し、コンポーネントのアクション マップでそれにマップされたアクションを実行しているかを知るにはどうすればよいでしょうか? 複雑な GUI を作成しているため、この情報を知っておくと非常に役立ちます。

4

2 に答える 2

0

AspectJ を使用して、ロギングの側面をコード ベースに織り込むことができます。actionPerformed(ActionEvent)以下は、コードベースにあるメソッドを使用してオブジェクトを実行する際のジョインポイントに関するアドバイスの例です。同様の構造を使用して、他のリスナーにアドバイスすることができます。

以下は、ボタンの押下や ActionListeners を持つその他のコンポーネントにアドバイスするアスペクトです。アクションのソースのクラス名と actionPerformed メソッドのシグネチャを出力するだけです。

import java.awt.event.ActionEvent;

import org.aspectj.lang.Signature;

public aspect Logger {
  before(ActionEvent e) : execution(* *.actionPerformed(ActionEvent)) && args(e) {
    Signature sig = thisJoinPoint.getSignature();
    System.out.println(e.getSource().getClass() + " lead to " + sig);
  }
}

異なるクラスの 2 つのボタンを生成するテスト クラス (ファイル StackTraceExample.java 内):

import java.awt.BorderLayout;
import java.awt.GridLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextArea;
import javax.swing.SwingUtilities;

class MyButton extends JButton {
  public MyButton(String string) {
    super(string);
  }
}

class MyOtherButton extends JButton {
  public MyOtherButton(String string) {
    super(string);
  }
}

class ButtonStackDisplay implements ActionListener {
  private final JTextArea stackTraceText;

  ButtonStackDisplay(JTextArea textArea) {
    this.stackTraceText = textArea;
  }

  public void actionPerformed(ActionEvent e) {
    String endl = System.getProperty("line.separator");
    StringBuilder b = new StringBuilder();

    // you can see the source of the event 
    b.append(e.getSource()).append(endl).append(endl);

    // the stack trace shows that events don't propagate through the components
    // originating them, but instead processed in a different thread
    for (StackTraceElement se : new Throwable().getStackTrace()) {
      b.append(se.toString());
      b.append(endl);
    }
    stackTraceText.setText(b.toString());
  }
}

public class StackTraceExample {
  public static void main(String[] args) {
    SwingUtilities.invokeLater(new Runnable() {
      public void run() {
        JFrame f = new JFrame();
        f.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
        f.setLayout(new BorderLayout());
        JPanel top = new JPanel();
        JButton button = new MyButton("Stack Trace");
        top.setLayout(new GridLayout(2, 1));
        top.add(button);
        JButton otherButton = new MyOtherButton("Stack Trace");
        top.add(otherButton);
        f.getContentPane().add(top, BorderLayout.NORTH);
        JTextArea stackTraceText = new JTextArea();
        f.add(stackTraceText, BorderLayout.CENTER);
        ButtonStackDisplay bsd = new ButtonStackDisplay(stackTraceText);
        button.addActionListener(bsd);
        otherButton.addActionListener(bsd);
        f.setSize(400, 400);
        f.setLocationRelativeTo(null);
        f.setVisible(true);
        f.toFront();
      }
    });
  }
}
于 2011-06-28T03:27:20.367 に答える
0

カスタム イベント ディスパッチャを配置: http://tips4java.wordpress.com/2009/09/06/global-event-dispatching/

また、API ドキュメントで AWTEvent.getID()、MouseEvent、KeyEvent を調べてください。

これは、私が取り組んでいるソフトウェアがマウスとキーボードを監視して、ユーザーが忙しいかどうかを確認し、そうでない場合はウィンドウをロックする方法です。

于 2011-06-23T20:59:28.653 に答える