Добавление вертикальной прокрутки в JPopupMenu?

Я хотел бы добавить способ прокрутки пунктов меню в JPopupMenuочень похоже на прокрутку списка элементов в JComboBox,

Допустим, у меня есть 10 пунктов меню. Я хотел бы отображать только 5 пунктов меню одновременно, и я бы использовал вертикальную кнопку прокрутки внизу или вверху JPopupMenu показать пункты меню, которых нет в списке, и скрыть те, которые я только что видел.

Является ли это возможным? Я использую программное обеспечение JIDE JideSplitButton, который отображает JPopupMenu при нажатии. Я пытаюсь сохранить внешний вид командной строки, на которой я разместил JideSplitButtonпоэтому я не хочу заменять его JComboBox если я действительно не должен.

9 ответов

Решение

Вот версия, которую я создал с помощью полосы прокрутки. Это простой пример, поэтому адаптируйтесь так, как считаете нужным:

import java.awt.Component;
import java.awt.Container;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.Insets;
import java.awt.LayoutManager;
import java.awt.event.AdjustmentEvent;
import java.awt.event.AdjustmentListener;
import java.awt.event.MouseWheelEvent;
import java.awt.event.MouseWheelListener;
import javax.swing.JPopupMenu;
import javax.swing.JScrollBar;

public class JScrollPopupMenu extends JPopupMenu {
    protected int maximumVisibleRows = 10;

    public JScrollPopupMenu() {
        this(null);
    }


    public JScrollPopupMenu(String label) {
        super(label);
        setLayout(new ScrollPopupMenuLayout());

        super.add(getScrollBar());
        addMouseWheelListener(new MouseWheelListener() {
            @Override public void mouseWheelMoved(MouseWheelEvent event) {
                JScrollBar scrollBar = getScrollBar();
                int amount = (event.getScrollType() == MouseWheelEvent.WHEEL_UNIT_SCROLL)
                             ? event.getUnitsToScroll() * scrollBar.getUnitIncrement()
                             : (event.getWheelRotation() < 0 ? -1 : 1) * scrollBar.getBlockIncrement();

                scrollBar.setValue(scrollBar.getValue() + amount);
                event.consume();
            }
        });
    }

    private JScrollBar popupScrollBar;
    protected JScrollBar getScrollBar() {
        if(popupScrollBar == null) {
            popupScrollBar = new JScrollBar(JScrollBar.VERTICAL);
            popupScrollBar.addAdjustmentListener(new AdjustmentListener() {
                @Override public void adjustmentValueChanged(AdjustmentEvent e) {
                    doLayout();
                    repaint();
                }
            });

            popupScrollBar.setVisible(false);
        }

        return popupScrollBar;
    }

    public int getMaximumVisibleRows() {
        return maximumVisibleRows;
    }

    public void setMaximumVisibleRows(int maximumVisibleRows) {
        this.maximumVisibleRows = maximumVisibleRows;
    }

    public void paintChildren(Graphics g){
        Insets insets = getInsets();
        g.clipRect(insets.left, insets.top, getWidth(), getHeight() - insets.top - insets.bottom);
        super.paintChildren(g);
    }

    protected void addImpl(Component comp, Object constraints, int index) {
        super.addImpl(comp, constraints, index);

        if(maximumVisibleRows < getComponentCount()-1) {
            getScrollBar().setVisible(true);
        }
    }

    public void remove(int index) {
        // can't remove the scrollbar
        ++index;

        super.remove(index);

        if(maximumVisibleRows >= getComponentCount()-1) {
            getScrollBar().setVisible(false);
        }
    }

    public void show(Component invoker, int x, int y){
        JScrollBar scrollBar = getScrollBar();
        if(scrollBar.isVisible()){
            int extent = 0;
            int max = 0;
            int i = 0;
            int unit = -1;
            int width = 0;
            for(Component comp : getComponents()) {
                if(!(comp instanceof JScrollBar)) {
                    Dimension preferredSize = comp.getPreferredSize();
                    width = Math.max(width, preferredSize.width);
                    if(unit < 0){
                        unit = preferredSize.height;
                    }
                    if(i++ < maximumVisibleRows) {
                        extent += preferredSize.height;
                    }
                    max += preferredSize.height;
                }
            }

            Insets insets = getInsets();
            int widthMargin = insets.left + insets.right;
            int heightMargin = insets.top + insets.bottom;
            scrollBar.setUnitIncrement(unit);
            scrollBar.setBlockIncrement(extent);
            scrollBar.setValues(0, heightMargin + extent, 0, heightMargin + max);

            width += scrollBar.getPreferredSize().width + widthMargin;
            int height = heightMargin + extent;

            setPopupSize(new Dimension(width, height));
        }

        super.show(invoker, x, y);
    }

    protected static class ScrollPopupMenuLayout implements LayoutManager{
        @Override public void addLayoutComponent(String name, Component comp) {}
        @Override public void removeLayoutComponent(Component comp) {}

        @Override public Dimension preferredLayoutSize(Container parent) {
            int visibleAmount = Integer.MAX_VALUE;
            Dimension dim = new Dimension();
            for(Component comp :parent.getComponents()){
                if(comp.isVisible()) {
                    if(comp instanceof JScrollBar){
                        JScrollBar scrollBar = (JScrollBar) comp;
                        visibleAmount = scrollBar.getVisibleAmount();
                    }
                    else {
                        Dimension pref = comp.getPreferredSize();
                        dim.width = Math.max(dim.width, pref.width);
                        dim.height += pref.height;
                    }
                }
            }

            Insets insets = parent.getInsets();
            dim.height = Math.min(dim.height + insets.top + insets.bottom, visibleAmount);

            return dim;
        }

        @Override public Dimension minimumLayoutSize(Container parent) {
            int visibleAmount = Integer.MAX_VALUE;
            Dimension dim = new Dimension();
            for(Component comp : parent.getComponents()) {
                if(comp.isVisible()){
                    if(comp instanceof JScrollBar) {
                        JScrollBar scrollBar = (JScrollBar) comp;
                        visibleAmount = scrollBar.getVisibleAmount();
                    }
                    else {
                        Dimension min = comp.getMinimumSize();
                        dim.width = Math.max(dim.width, min.width);
                        dim.height += min.height;
                    }
                }
            }

            Insets insets = parent.getInsets();
            dim.height = Math.min(dim.height + insets.top + insets.bottom, visibleAmount);

            return dim;
        }

        @Override public void layoutContainer(Container parent) {
            Insets insets = parent.getInsets();

            int width = parent.getWidth() - insets.left - insets.right;
            int height = parent.getHeight() - insets.top - insets.bottom;

            int x = insets.left;
            int y = insets.top;
            int position = 0;

            for(Component comp : parent.getComponents()) {
                if((comp instanceof JScrollBar) && comp.isVisible()) {
                    JScrollBar scrollBar = (JScrollBar) comp;
                    Dimension dim = scrollBar.getPreferredSize();
                    scrollBar.setBounds(x + width-dim.width, y, dim.width, height);
                    width -= dim.width;
                    position = scrollBar.getValue();
                }
            }

            y -= position;
            for(Component comp : parent.getComponents()) {
                if(!(comp instanceof JScrollBar) && comp.isVisible()) {
                    Dimension pref = comp.getPreferredSize();
                    comp.setBounds(x, y, width, pref.height);
                    y += pref.height;
                }
            }
        }
    }
}

В дополнение к JScrollPopupMenu, описанному выше, мне также потребовалась полоса прокрутки в подменю (также известное как "Вытянуть правое меню"). Это, кажется, более распространенный случай. Поэтому я адаптировал JMenu для использования JScrollPopupMenu с именем JScrollMenu:

import javax.swing.Action;
import javax.swing.JButton;
import javax.swing.JMenu;
import javax.swing.JMenuItem;
import javax.swing.JPopupMenu;
import javax.swing.MenuElement;
import javax.swing.UIManager;
import javax.swing.plaf.MenuItemUI;
import javax.swing.plaf.PopupMenuUI;
import java.awt.Component;
import java.awt.ComponentOrientation;



public class JScrollMenu extends JMenu {
    // Covers the one in the JMenu because the method that creates it in JMenu is private
    /** The popup menu portion of the menu.*/
    private JPopupMenu popupMenu;


    /**
     * Constructs a new <code>JMenu</code> with no text.
     */
    public JScrollMenu() {
        this("");
    }

    /**
     * Constructs a new <code>JMenu</code> with the supplied string as its text.
     *
     * @param s the text for the menu label
     */
    public JScrollMenu(String s) {
        super(s);
    }

    /**
     * Constructs a menu whose properties are taken from the <code>Action</code> supplied.
     *
     * @param a an <code>Action</code>
     */
    public JScrollMenu(Action a) {
        this();
        setAction(a);
    }


    /**
     * Lazily creates the popup menu. This method will create the popup using the <code>JScrollPopupMenu</code> class. 
     */
    protected void ensurePopupMenuCreated() {
        if(popupMenu == null) {
            this.popupMenu = new JScrollPopupMenu();
            popupMenu.setInvoker(this);
            popupListener = createWinListener(popupMenu);
        }
    }

//////////////////////////////
//// All of these methods are necessary because ensurePopupMenuCreated() is private in JMenu
//////////////////////////////
    @Override
    public void updateUI() {
        setUI((MenuItemUI) UIManager.getUI(this));

        if(popupMenu != null) {
            popupMenu.setUI((PopupMenuUI) UIManager.getUI(popupMenu));
        }
    }


    @Override
    public boolean isPopupMenuVisible() {
        ensurePopupMenuCreated();
        return popupMenu.isVisible();
    }


    @Override
    public void setMenuLocation(int x, int y) {
        super.setMenuLocation(x, y);
        if(popupMenu != null) {
            popupMenu.setLocation(x, y);
        }
    }

    @Override
    public JMenuItem add(JMenuItem menuItem) {
        ensurePopupMenuCreated();
        return popupMenu.add(menuItem);
    }

    @Override
    public Component add(Component c) {
        ensurePopupMenuCreated();
        popupMenu.add(c);
        return c;
    }

    @Override
    public Component add(Component c, int index) {
        ensurePopupMenuCreated();
        popupMenu.add(c, index);
        return c;
    }


    @Override
    public void addSeparator() {
        ensurePopupMenuCreated();
        popupMenu.addSeparator();
    }

    @Override
    public void insert(String s, int pos) {
        if(pos < 0) {
            throw new IllegalArgumentException("index less than zero.");
        }

        ensurePopupMenuCreated();
        popupMenu.insert(new JMenuItem(s), pos);
    }

    @Override
    public JMenuItem insert(JMenuItem mi, int pos) {
        if(pos < 0) {
            throw new IllegalArgumentException("index less than zero.");
        }
        ensurePopupMenuCreated();
        popupMenu.insert(mi, pos);
        return mi;
    }

    @Override
    public JMenuItem insert(Action a, int pos) {
        if(pos < 0) {
            throw new IllegalArgumentException("index less than zero.");
        }

        ensurePopupMenuCreated();
        JMenuItem mi = new JMenuItem(a);
        mi.setHorizontalTextPosition(JButton.TRAILING);
        mi.setVerticalTextPosition(JButton.CENTER);
        popupMenu.insert(mi, pos);
        return mi;
    }

    @Override
    public void insertSeparator(int index) {
        if(index < 0) {
            throw new IllegalArgumentException("index less than zero.");
        }

        ensurePopupMenuCreated();
        popupMenu.insert(new JPopupMenu.Separator(), index);
    }


    @Override
    public void remove(JMenuItem item) {
        if(popupMenu != null){
            popupMenu.remove(item);
        }
    }

    @Override
    public void remove(int pos) {
        if(pos < 0) {
            throw new IllegalArgumentException("index less than zero.");
        }
        if(pos > getItemCount()) {
            throw new IllegalArgumentException("index greater than the number of items.");
        }
        if(popupMenu != null){
            popupMenu.remove(pos);
        }
    }

    @Override
    public void remove(Component c) {
        if(popupMenu != null){
            popupMenu.remove(c);
        }
    }

    @Override
    public void removeAll() {
        if(popupMenu != null){
            popupMenu.removeAll();
        }
    }

    @Override
    public int getMenuComponentCount() {
        return (popupMenu == null) ? 0 : popupMenu.getComponentCount();
    }

    @Override
    public Component getMenuComponent(int n) {
        return (popupMenu == null) ? null : popupMenu.getComponent(n);
    }

    @Override
    public Component[] getMenuComponents() {
        return (popupMenu == null) ? new Component[0] : popupMenu.getComponents();
    }

    @Override
    public JPopupMenu getPopupMenu() {
        ensurePopupMenuCreated();
        return popupMenu;
    }

    @Override
    public MenuElement[] getSubElements() {
        return popupMenu == null ? new MenuElement[0] : new MenuElement[]{popupMenu};
    }


    @Override
    public void applyComponentOrientation(ComponentOrientation o) {
        super.applyComponentOrientation(o);

        if(popupMenu != null) {
            int ncomponents = getMenuComponentCount();
            for(int i = 0; i < ncomponents; ++i) {
                getMenuComponent(i).applyComponentOrientation(o);
            }
            popupMenu.setComponentOrientation(o);
        }
    }

    @Override
    public void setComponentOrientation(ComponentOrientation o) {
        super.setComponentOrientation(o);
        if(popupMenu != null) {
            popupMenu.setComponentOrientation(o);
        }
    }
}

Вот еще один, который я нашел очень полезным: https://tips4java.wordpress.com/2009/02/01/menu-scroller/

Он может быть вызван в JMenu или JPopupMenu следующим образом:

MenuScroller.setScrollerFor(menuInstance, 8, 125, 3, 1);

скроллер меню

По сути, вы можете добавить любые JComponents в JPopupMenu, вы можете добавить JScrollpane в JPopup, вложив JPanel / JList с другим JComponents,

Заметьте, но есть правило, что графический интерфейс Swing не позволяет использовать два легких всплывающих окна одновременно, лучший пример - распространенная ошибка в Swing о JComboBox в JPopup.

Вы взглянули на JWindow, один раз создаете и повторно используете его для другого действия, нет ничего лучше, чтобы проверить, как всплывающее окно JWindow действительно работает для JCalendar от Kai Toedter

Добавление к ответу JScrollPopupMenu выше (я не могу его редактировать).

Чтобы он прокручивался при навигации по стрелкам и т. Д., Я добавил это:

      @Override public void scrollRectToVisible(Rectangle aRect) {
    final Insets insets = getInsets();
    final int scrollY = popupScrollBar.getValue();
    int y = aRect.y;
    if (y - insets.top < 0)
        popupScrollBar.setValue(scrollY + y - insets.top);
    else {
        y += aRect.height;
        final int bottom = getHeight() - insets.bottom;
        if (y > bottom)
            popupScrollBar.setValue(scrollY + y - bottom);
    }
}

Который затем будет называться так:

      popupMenu.scrollRectToVisible(menuItem.getBounds());

Для точной прокрутки (трекпад, скопируйте из com.formdev.flatlaf.ui.FlatScrollPaneUI.mouseWheelMovedSmooth):

      public JScrollPopupMenu(String label) {
    super(label);
    setLayout(new ScrollPopupMenuLayout());

    super.add(getScrollBar());
    addMouseWheelListener(e -> {
        if (e.getScrollType() == MouseWheelEvent.WHEEL_UNIT_SCROLL &&
                e.getPreciseWheelRotation() != 0 &&
                e.getPreciseWheelRotation() != e.getWheelRotation()) {
            mouseWheelMovedSmooth(e);
        } else {
            JScrollBar scrollBar = getScrollBar();
            int amount = (e.getScrollType() == MouseWheelEvent.WHEEL_UNIT_SCROLL)
                    ? e.getUnitsToScroll() * scrollBar.getUnitIncrement()
                    : (e.getWheelRotation() < 0 ? -1 : 1) * scrollBar.getBlockIncrement();

            scrollBar.setValue(scrollBar.getValue() + amount);
            e.consume();
        }
    });
}

...

// com.formdev.flatlaf.ui.FlatScrollPaneUI.mouseWheelMovedSmooth
private void mouseWheelMovedSmooth(MouseWheelEvent e) {
    // find scrollbar to scroll
    JScrollBar scrollbar = popupScrollBar;

    // consume event
    e.consume();

    // get precise wheel rotation
    double rotation = e.getPreciseWheelRotation();

    // get unit increment
    int orientation = scrollbar.getOrientation();
    int direction = rotation < 0 ? -1 : 1;
    int unitIncrement = scrollbar.getUnitIncrement(direction);

    // get viewport width/height (the visible width/height)
    int viewportWH = (orientation == SwingConstants.VERTICAL)
            ? this.getHeight()
            : this.getWidth();

    // limit scroll increment to viewport width/height
    // - if scroll amount is set to a large value in OS settings
    // - for large unit increments in small viewports (e.g. horizontal scrolling in file chooser)
    int scrollIncrement = Math.min(unitIncrement * e.getScrollAmount(), viewportWH);

    // compute relative delta
    double delta = rotation * scrollIncrement;
    int idelta = (int) Math.round(delta);

    // scroll at least one pixel to avoid "hanging"
    // - for "super-low-speed" scrolling (move fingers very slowly on trackpad)
    // - if unit increment is very small (e.g. 1 if scroll view does not implement
    //   javax.swing.Scrollable interface)
    if (idelta == 0) {
        if (rotation > 0) {
            idelta = 1;
        } else if (rotation < 0) {
            idelta = -1;
        }
    }

    // compute new value
    int value = scrollbar.getValue();
    int minValue = scrollbar.getMinimum();
    int maxValue = scrollbar.getMaximum() - scrollbar.getModel().getExtent();
    int newValue = Math.max(minValue, Math.min(value + idelta, maxValue));

    // set new value
    if (newValue != value) {
        scrollbar.setValue(newValue);
    }
}

С другой стороны, вы можете рассмотреть JidePopupMenu: прокручиваемое JPopupMenu

Поскольку мне нужно было всплывающее меню с полосой прокрутки, я просто повторно использовал всплывающее меню из JComboBox. Хитрость заключалась в том, чтобы поместить JComboBox в JViewport, чтобы была видна только кнопка со стрелкой. Вы можете сделать его всего на один пиксель маленьким или даже меньше и использовать событие из JideSplitButton, чтобы открыть всплывающее окно.

Вы можете найти код на GitHub.

Другие вопросы по тегам