简单;我在JScrollPane
中有一个JPanel
;
不出所料;JScrollPane
默认情况下侦听MouseWheelEvent
,以便在滚轮旋转时和光标悬停在JPanel
上时滚动工作良好。
但是在那之后;我刚刚更新了JPanel
,使其实现了MouseWheelListener
,并且我为JPanel
本身添加了这个鼠标滚轮侦听器。
@Override
public void mouseWheelMoved(MouseWheelEvent e) {
if (e.isControlDown()) {
if (e.getWheelRotation() < 0) {
System.out.println("mouse wheel Up");
} else {
System.out.println("mouse wheel Down");
}
}
}
当JPanel
同时响应此实现;按下Ctrl并且轮子正在旋转并且光标悬停在JPanel
上时。但是JScrollPane
的默认行为意外丢失!!!
当我在光标悬停在JPanel
上时旋转滚轮时,JScrollPane
的滚动没有响应!!!
似乎;MouseWheelListener
的这种实现打破了JPanel
的默认值。
所以;如何为JPanel
实现MouseWheelListener
而不破坏其默认实现?
MouseWheelEvents
和滚动行为有一些微妙的警告。
例如,当您打开此页面(我指的是您正在阅读的此页面),将鼠标放在中间,并开始随着滚轮向下滚动时,您将滚动代码片段。请注意,尽管代码片段包含在具有滚动条的代码块中,但持续旋转鼠标滚轮不会触发代码块中的滚动,而只会触发整个页面的滚动。但是,当您在代码块内移动鼠标并随后滚动鼠标滚轮时,您将仅在代码块中滚动,而不是整个页面。
同样,旋转鼠标滚轮可能不会影响悬停的可滚动。我认为这取决于窗口管理器和查找
但是,其中一些机制和微妙之处也可以在Swing组件中找到。例如,如果组件本身没有处理MouseWheelEvents
,则将其传递给祖先的重新调度机制。
遵循这种模式,一个解决方案(在概念上类似于LuxxMiner提出的解决方案,但可能更通用)可能是简单地将MouseWheelEvent
重新分派到父组件:
package stackoverflow;
import java.awt.Dimension;
import java.awt.GridLayout;
import java.awt.event.MouseWheelEvent;
import java.awt.event.MouseWheelListener;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.SwingUtilities;
public class MouseWheelListenerForPanelInScrollpane
{
public static void main(String[] args)
{
SwingUtilities.invokeLater(new Runnable()
{
@Override
public void run()
{
createAndShowGUI();
}
});
}
private static void createAndShowGUI()
{
JFrame f = new JFrame();
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.getContentPane().setLayout(new GridLayout(1,2));
MouseWheelListenerPanel m = new MouseWheelListenerPanel();
m.setPreferredSize(new Dimension(100,4000));
JScrollPane scrollPane = new JScrollPane(m);
f.getContentPane().add(scrollPane);
f.setSize(500,500);
f.setLocationRelativeTo(null);
f.setVisible(true);
}
}
class MouseWheelListenerPanel extends JPanel implements MouseWheelListener
{
MouseWheelListenerPanel()
{
addMouseWheelListener(this);
}
@Override
public void mouseWheelMoved(MouseWheelEvent e)
{
if (e.isControlDown())
{
if (e.getWheelRotation() < 0)
{
System.out.println("mouse wheel Up");
}
else
{
System.out.println("mouse wheel Down");
}
}
else
{
getParent().dispatchEvent(e);
}
}
}
如果ctrl未关闭,添加else
以将事件直接重新分派到滚动窗格:
@Override
public void mouseWheelMoved(MouseWheelEvent e) {
if (e.isControlDown()) {
if (e.getWheelRotation() < 0) {
System.out.println("mouse wheel Up");
} else {
System.out.println("mouse wheel Down");
}
} else {
// pass the event on to the scroll pane
getParent().dispatchEvent(e);
}
}
我不知道这是否真的符合正确的答案,因为它是一种解决方法,但我想出了以下解决方案:仅当Ctrl未被按下时,才调用scrollPane
的mouseWheelMove
方法:
if (e.isControlDown()) {
if (e.getWheelRotation() < 0) {
infoLabel.setText("Mouse Wheel Up");
} else {
infoLabel.setText("Mouse Wheel Down");
}
} else {
scrollPane.getListeners(MouseWheelListener.class)[0].mouseWheelMoved(e);
}
完整示例:
import java.awt.BorderLayout;
import java.awt.EventQueue;
import java.awt.GridLayout;
import java.awt.event.MouseWheelEvent;
import java.awt.event.MouseWheelListener;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.border.TitledBorder;
public class Example {
public Example() {
JFrame frame = new JFrame();
frame.setLayout(new BorderLayout());
frame.add(new ScrollPanePanel());
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setSize(400, 400);
frame.setVisible(true);
}
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
new Example();
}
});
}
}
class ScrollPanePanel extends JPanel implements MouseWheelListener {
private JLabel infoLabel;
private JScrollPane scrollPane;
public ScrollPanePanel() {
JPanel panel = new JPanel(new GridLayout(0, 1));
for (int i = 1; i <= 100; i++) {
panel.add(new JLabel("Label " + i));
}
panel.addMouseWheelListener(this);
scrollPane = new JScrollPane(panel);
infoLabel = new JLabel(" ");
JPanel infoPanel = new JPanel();
infoPanel.add(infoLabel);
setLayout(new BorderLayout());
add(scrollPane);
add(infoPanel, BorderLayout.SOUTH);
}
@Override
public void mouseWheelMoved(MouseWheelEvent e) {
if (e.isControlDown()) {
if (e.getWheelRotation() < 0) {
infoLabel.setText("Mouse Wheel Up");
} else {
infoLabel.setText("Mouse Wheel Down");
}
} else {
scrollPane.getListeners(MouseWheelListener.class)[0].mouseWheelMoved(e);
}
}
}