gpt4 book ai didi

java - 向应用程序中的所有 Swing 文本组件添加上下文菜单

转载 作者:行者123 更新时间:2023-11-29 07:06:15 24 4
gpt4 key购买 nike

Swing 文本组件没有带有剪切/复制/粘贴/等功能的上下文菜单。我想添加一个,以便它表现得更流畅,更像一个本地应用程序。我为此写了一个菜单,它工作正常。我使用以下方法将它添加到每个文本框:

someTextBox.setComponentPopupMenu(TextContextMenu.INSTANCE);

问题是,到处添加这个很烦人。其次,如果我忘记了某处的文本框,则应用程序不一致。第三,我无法为我无法控制创建代码的文本框添加它,例如来自 JOptionPane.showInputDialogJFileChooser 对话框的文本框。

有什么方法可以在整个应用程序范围内覆盖 JTextComponent 的默认上下文菜单吗?我知道这将是 spooky action at a distance 的一种形式但我没意见。也欢迎对菜单本身发表评论。

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.text.JTextComponent;

public class TextContextMenu extends JPopupMenu implements ActionListener {
public static final TextContextMenu INSTANCE = new TextContextMenu();
private final JMenuItem itemCut;
private final JMenuItem itemCopy;
private final JMenuItem itemPaste;
private final JMenuItem itemDelete;
private final JMenuItem itemSelectAll;

private TextContextMenu() {
itemCut = newItem("Cut", 'T');
itemCopy = newItem("Copy", 'C');
itemPaste = newItem("Paste", 'P');
itemDelete = newItem("Delete", 'D');
addSeparator();
itemSelectAll = newItem("Select All", 'A');
}

private JMenuItem newItem(String text, char mnemonic) {
JMenuItem item = new JMenuItem(text, mnemonic);
item.addActionListener(this);
return add(item);
}

@Override
public void show(Component invoker, int x, int y) {
JTextComponent tc = (JTextComponent)invoker;
boolean changeable = tc.isEditable() && tc.isEnabled();
itemCut.setVisible(changeable);
itemPaste.setVisible(changeable);
itemDelete.setVisible(changeable);
super.show(invoker, x, y);
}

@Override
public void actionPerformed(ActionEvent e) {
JTextComponent tc = (JTextComponent)getInvoker();
tc.requestFocus();

boolean haveSelection = tc.getSelectionStart() != tc.getSelectionEnd();
if (e.getSource() == itemCut) {
if (!haveSelection) tc.selectAll();
tc.cut();
} else if (e.getSource() == itemCopy) {
if (!haveSelection) tc.selectAll();
tc.copy();
} else if (e.getSource() == itemPaste) {
tc.paste();
} else if (e.getSource() == itemDelete) {
if (!haveSelection) tc.selectAll();
tc.replaceSelection("");
} else if (e.getSource() == itemSelectAll) {
tc.selectAll();
}
}
}

最佳答案

我已经弄清楚如何在整个应用程序范围内执行此操作,包括 JFileChooser 和 showInputDialog 等等!我不确定它是多么理智和恰当,但它确实有效。它(ab)使用可插入的外观系统。 JTextComponent 在其构造函数期间调用 updateUI,这提供了在 L&F 被要求提供其 UI 委托(delegate)时调用 setComponentPopupMenu 的机会。

如果你change the look and feel for already-open windows ,每个组件的 updateUI 方法都会被再次调用。为了防止再次设置默认菜单,下面的代码使用 JComponent.putClientProperty 存储文本框是否已经初始化的属性。 .

最终效果是它的行为就像每个 JTextComponent 本身在其构造函数期间仅调用一次 setComponentPopupMenu 一样。因此,对于不需要菜单或需要不同菜单的特殊文本框,很容易覆盖它:只需再次调用 setComponentPopupMenu。例如,来自文本字段子类构造函数或来自创建窗口及其小部件的调用代码。

这是在应用程序启动时运行一次的代码:

UIManager.addAuxiliaryLookAndFeel(new LookAndFeel() {
private final UIDefaults defaults = new UIDefaults() {
@Override
public javax.swing.plaf.ComponentUI getUI(JComponent c) {
if (c instanceof javax.swing.text.JTextComponent) {
if (c.getClientProperty(this) == null) {
c.setComponentPopupMenu(TextContextMenu.INSTANCE);
c.putClientProperty(this, Boolean.TRUE);
}
}
return null;
}
};
@Override public UIDefaults getDefaults() { return defaults; };
@Override public String getID() { return "TextContextMenu"; }
@Override public String getName() { return getID(); }
@Override public String getDescription() { return getID(); }
@Override public boolean isNativeLookAndFeel() { return false; }
@Override public boolean isSupportedLookAndFeel() { return true; }
});

关于java - 向应用程序中的所有 Swing 文本组件添加上下文菜单,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19424574/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com