将上下文菜单添加到应用程序中的所有Swing文本组件

时间:2013-10-17 10:45:43

标签: java swing contextmenu jtextcomponent

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();
        }
    }
}

3 个答案:

答案 0 :(得分:4)

我已经想出了如何在应用程序范围内执行此操作,包括JFileChoosers和showInputDialog等等!我不确定它是多么健康,但是它有效。它(ab)使用可插拔的外观和感觉系统。 JTextComponent在其构造函数期间调用updateUI,当L& F被要求提供其UI委托时,它提供了调用setComponentPopupMenu的机会。

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

净效果是它的行为就像每个JTextComponent本身在其构造函数中只调用一次setComponentPopupMenu一样。因此,对于不需要菜单或想要不同菜单的特殊文本框,可以轻松覆盖它:只需再次调用setComponentPopupMenu。例如,来自textfield子类构造函数或来自创建窗口及其小部件的调用代码。

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

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; }
});

答案 1 :(得分:1)

您有两种选择:

  1. 如果您的应用程序有自己定制的外观,您可以 只需在TextUI中的installUI(JComponent c)方法中执行此操作。

  2. 如果您没有自己的L& F,则必须遍历组件树, 找到JTextComponent的所有实例,并添加上下文菜单。一世 将它作为一个静态助手方法。对于对话框和框架你 需要一个基本类,它重新定义要添加的方法setVisible() 所有文本组件的上下文菜单。所有自定义对话框或 框架必须扩展基本对话框/框架。

答案 2 :(得分:1)

由于我遇到的JavaHelp冲突,我需要一个不同的解决方案。我想出的是选择的变体2.在第二个答案中,我在实例化类而不是调用setVisible()时进行工作。

public static void installDefaultTextContextMenus( Container aContainer )
    {
        if ( aContainer != null )
        {
            if ( aContainer instanceof JFrame )
            {
                aContainer = ((JFrame)aContainer).getContentPane();
            }
            else if ( aContainer instanceof JDialog )
            {
                aContainer = ((JDialog)aContainer).getContentPane();
            }
            Component[] lComponents = aContainer.getComponents();
            for ( int lCompNum = 0; lCompNum < lComponents.length; ++lCompNum )
            {
                lComponents[ lCompNum ].getClass();
                if ( ( lComponents[ lCompNum ] instanceof JPanel ) ||
                     ( lComponents[ lCompNum ] instanceof JInternalFrame ) ||
                     ( lComponents[ lCompNum ] instanceof JScrollPane ) ||
                     ( lComponents[ lCompNum ] instanceof JSplitPane ) ||
                     ( lComponents[ lCompNum ] instanceof JTabbedPane ) ||
                     ( lComponents[ lCompNum ] instanceof Panel ) ||
                     ( lComponents[ lCompNum ] instanceof ScrollPane ) ||
                     ( lComponents[ lCompNum ] instanceof JViewport ) ||
                     ( lComponents[ lCompNum ] instanceof JFrame ) ||
                     ( lComponents[ lCompNum ] instanceof JDialog ) )
                {
                    installDefaultTextContextMenus( (Container)lComponents[ lCompNum ] );
                }
                else if ( lComponents[ lCompNum ] instanceof JTextComponent )
                {
                    ((JTextComponent)lComponents[ lCompNum ]).setComponentPopupMenu( TextContextMenu.INSTANCE );
                }
                else if ( lComponents[ lCompNum ] instanceof JComboBox )
                {
                    Component lEditorComp = ((JComboBox)lComponents[ lCompNum ]).getEditor().getEditorComponent();
                    if ( lEditorComp instanceof JTextComponent )
                    {
                        ((JTextComponent)lEditorComp).setComponentPopupMenu( TextContextMenu.INSTANCE );
                    }
                }
            }
        }
    }

该方法几乎遍历任何Swing组件树。幸运的是,我已经有了一个常用的方法来装饰我的所有应用程序的顶级JFrame,我可以在其中添加对此方法的调用。但是,我必须为我的顶级菜单未处理的帧或对话框的每个实例化添加对此方法的调用。