简体   繁体   English

如何使嵌套的JScrollPanes正常工作?

[英]How to make nested JScrollPanes work correctly?

在此处输入图片说明

As the image shows, I have two JScrollPanes, A and B. What I want to achieve is, when the cursor is over B, and you scroll the mouse wheel: 如图所示,我有两个JScrollPanes,A和B。我要实现的是,当光标移到B上时,您滚动鼠标滚轮:

1) If B has scrollable contents, just scroll B. 1)如果B具有可滚动的内容,则只需滚动B。

2) If B is scrolled to the end already, or B has very few contents that it's not scrollable, scroll A. 2)如果B已经滚动到末尾,或者B的内容很少,则无法滚动,请滚动A。

What happens now is that the mouse wheel event is always consumed by B no matter what, so A could never be scrolled. 现在发生的是,无论如何,鼠标滚轮事件始终被B占用,因此A永远不会滚动。 Any advice? 有什么建议吗? Thanks in advance. 提前致谢。

Here is one possible implementation using JLayer : 这是使用JLayer一种可能的实现:

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.plaf.LayerUI;
import javax.swing.table.*;
import javax.swing.text.*;

public class WheelOverNestedScrollPaneTest {
  private static final String TEXT = "aaa\na\na\na\na\naaaa\na\na\na\naaaa\n";

  private JComponent makeUI() {
    JTextArea textArea = new JTextArea(TEXT + TEXT + TEXT);
    JTable table = new JTable(50, 3);
    JTree tree = new JTree();
    tree.setVisibleRowCount(5);

    JTextPane textPane = new JTextPane();
    textPane.setEditable(false);
    textPane.setMargin(new Insets(5, 10, 5, 5));
    Document doc = textPane.getDocument();
    try {
      doc.insertString(doc.getLength(), TEXT, null);
      textPane.insertComponent(createChildScrollPane(textArea));
      doc.insertString(doc.getLength(), "\n", null);
      doc.insertString(doc.getLength(), TEXT, null);
      textPane.insertComponent(createChildScrollPane(table));
      doc.insertString(doc.getLength(), "\n", null);
      doc.insertString(doc.getLength(), TEXT, null);
      textPane.insertComponent(new JScrollPane(tree));
      doc.insertString(doc.getLength(), "\n", null);
      doc.insertString(doc.getLength(), TEXT, null);
    } catch (BadLocationException ex) {
      ex.printStackTrace();
    }
    return new JLayer<JScrollPane>(
        new JScrollPane(textPane), new WheelScrollLayerUI());
  }
  protected static JScrollPane createChildScrollPane(Component view) {
    return new JScrollPane(view) {
      @Override public Dimension getPreferredSize() {
        Dimension d = super.getPreferredSize();
        d.height = 120;
        return d;
      }
    };
  }
  public static void main(String... args) {
    EventQueue.invokeLater(() -> {
      JFrame f = new JFrame();
      f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
      f.getContentPane().add(new WheelOverNestedScrollPaneTest().makeUI());
      f.setSize(320, 240);
      f.setLocationRelativeTo(null);
      f.setVisible(true);
    });
  }
}

// http://java-swing-tips.blogspot.jp/2014/09/forward-mouse-wheel-scroll-event-in.html
class WheelScrollLayerUI extends LayerUI<JScrollPane> {
  @Override public void installUI(JComponent c) {
    super.installUI(c);
    if (c instanceof JLayer) {
      ((JLayer) c).setLayerEventMask(AWTEvent.MOUSE_WHEEL_EVENT_MASK);
    }
  }
  @Override public void uninstallUI(JComponent c) {
    if (c instanceof JLayer) {
      ((JLayer) c).setLayerEventMask(0);
    }
    super.uninstallUI(c);
  }
  @Override protected void processMouseWheelEvent(
      MouseWheelEvent e, JLayer<? extends JScrollPane> l) {
    Component c = e.getComponent();
    int dir = e.getWheelRotation();
    JScrollPane main = l.getView();
    if (c instanceof JScrollPane && !c.equals(main)) {
      JScrollPane child = (JScrollPane) c;
      BoundedRangeModel m = child.getVerticalScrollBar().getModel();
      int extent  = m.getExtent();
      int minimum = m.getMinimum();
      int maximum = m.getMaximum();
      int value   = m.getValue();
      if (value + extent >= maximum && dir > 0 || value <= minimum && dir < 0) {
        main.dispatchEvent(SwingUtilities.convertMouseEvent(c, e, main));
      }
    }
  }
}

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM