Java 7 アプリケーションのメニュー バーを (Mac の場合) 画面の上部に表示し、キーボード ショートカットを正しく機能させるにはどうすればよいですか?
Swing ユーザー インターフェイスを備えた Java アプリケーションがあります。多くのメニューには、不可欠なキーボードと同等のものがあります。
システムに依存するものはほとんどありませんが、Mac OS X では、メニュー バーは各ウィンドウではなく画面の上部に表示されるはずなので、apple.laf.useScreenMenuBar
.
これは Java 6 では問題なく動作しますが、Java 7 (先週リリースされました!) で同じコードをコンパイルして実行すると、キーボード ショートカットでメニュー アクションが 2 回実行されます。たとえば、添付のコードでは、Command ⌘+Oは 1 つではなく 2 つのファイル ダイアログを開きます。(他のキーボード ショートカットも 2 回動作しますが、動作を確認するためにウィンドウを移動する必要がある場合があります。)
を設定しなければ、キーボードの問題はなくなります。必要に応じて設定apple.laf.useScreenMenuBar
しますが、Mac ユーザーは不満に思うでしょう。メニュー バーを適切な場所に配置し、キーボード ショートカットを機能させたいと思っています。
システム: 2010 年後半の MacBook Pro 上の Mac OS 10.7.3 (Lion)
Java 7:
Java バージョン "1.7.0_04"
Java(TM) SE ランタイム環境 (ビルド 1.7.0_04-b21)
Java HotSpot(TM) 64 ビット サーバー VM (ビルド 23.0-b21、混合モード)
Java 6:
Java バージョン "1.6.0_31"
Java(TM) SE ランタイム環境 (ビルド 1.6.0_31-b04-415-11M3635)
Java HotSpot(TM) 64 ビット サーバー VM (ビルド 20.6-b01-415、混合モード)
私が見た場所:
なぜapple.laf.useScreenMenuBar
取り除かなければならないかについての議論
-- 私は大賛成ですが、取り除かれなかったようです。
Mac を使用していることを検出するために使用しないmrj.version
ことについての議論
-- 直接の関連性はありませんが、有望に思えます。
添付のコードの長さ (148 行) については申し訳ありませんが、私の Swing コーディングは非常に古くなっています。特別なフラグや設定なしで、コマンド ラインからコンパイルして実行する必要があります。
import javax.swing.*;
import java.awt.Toolkit;
import java.awt.*;
import java.awt.event.*;
/**
* Shows that using the single screen-top menu bar on a Mac with Java 7
* causes keyboard shortcuts to act twice.
*
* To see the problem(on a Mac -- running OS X 10.7.3 in my case):
* 1) compile on either Java 6 or Java 7
* 2) run on Java 7
* 3) give the command-O shortcut
* You will see two file dialogues.
*
* -- J. Clarke, May 2012
*/
public class MenuBug {
private static void go(String[] args) {
// Comment out the following line to fix the problem;
// leave it active to see the problem.
// It doesn't help to ...
// ... put the line into a static block.
// ... put the line right after the setLookAndFeel call.
// ... put the line before after the setLookAndFeel call.
System.setProperty("apple.laf.useScreenMenuBar", "true");
MainWindow mainWindow = new MainWindow();
}
public static void main(final String[] args) {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
}
catch (Exception e) {
JOptionPane.showMessageDialog(null,
e + " while loading look and feel",
"MenuBug error", JOptionPane.ERROR_MESSAGE);
System.exit(1);
}
SwingUtilities.invokeLater(new Runnable() {
public void run() {
go(args);
}
});
}
}
class MainWindow extends JFrame {
MainWindow() {
super ("Main Window");
setDefaultCloseOperation(WindowConstants.DO_NOTHING_ON_CLOSE);
addWindowListener (new WindowAdapter() {
public void windowClosing(WindowEvent e) {
dispose();
System.exit(0);
}
});
JMenuBar menuBar = createMenuBar();
setJMenuBar(menuBar);
pack();
setSize(350,300);
setVisible(true);
}
private JMenuBar createMenuBar() {
JMenuBar mBar = new JMenuBar();
JMenu menu = new JMenu("File");
String[] menuItemNames = new String[] {"New", "Open...", "Other"};
for (int i = 0; i < menuItemNames.length; i++) {
String miName = menuItemNames[i];
JMenuItem mi = new JMenuItem(miName);
mi.setActionCommand(miName);
linkMenuItemToAction(mi);
menu.add(mi);
}
mBar.add(menu);
return mBar;
}
/**
* Create an Action for menuItem, and make sure the action and the menu
* item know about each other; where appropriate, add keyboard equivalents.
* @param menuItem The menu item to be linked to an action.
*/
private void linkMenuItemToAction(JMenuItem menuItem) {
final int META_MASK =
Toolkit.getDefaultToolkit().getMenuShortcutKeyMask();
Action a = null;
String miName = menuItem.getActionCommand();
if (miName.equals ("New")) {
a = new NewAction();
menuItem.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_N,
META_MASK));
}
else if (miName.equals ("Open...")) {
a = new OpenAction();
menuItem.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_O,
META_MASK));
}
else if (miName.equals ("Other")) {
a = new OtherAction();
menuItem.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_T,
META_MASK));
}
menuItem.setEnabled(a.isEnabled());
menuItem.addActionListener(a);
}
private class NewAction extends AbstractAction {
public void actionPerformed(ActionEvent e) {
new MainWindow();
}
}
private void makeDialog() {
String dialogTitle = "Please choose a file to open";
FileDialog fileDialog = new FileDialog(this, dialogTitle,
FileDialog.LOAD);
fileDialog.setVisible(true);
String fileName = fileDialog.getFile();
}
private class OpenAction extends AbstractAction {
public void actionPerformed(ActionEvent e) {
makeDialog();
}
}
private class OtherAction extends AbstractAction {
public void actionPerformed(ActionEvent e) {
JOptionPane.showMessageDialog(null,
"an example message",
"not really an error", JOptionPane.ERROR_MESSAGE);
}
}
}