当用户滚动事件时,我需要处理来自JScrollPane
的事件。在这里,建议将addAdjustmentListener
用于JViewport
或垂直/水平滚动条。
有什么不同吗?我应该选择哪种方法?
我也想将scrollRectToVisible
用于相同的JScrollPane
,我是否应该预期AdjustmentListener
适用于这种方法?
此外,我想知道scrollRectToVisible
是试图进行最小滚动以使所需的矩形可见,还是试图使其在JViewport
中间可见?
UPD:要求:
1) 有一个带有一个JPanel的JScrollPane,它有许多JLabel(ImageIcon),所以其中一些是不可见的。
2) 当网络事件发生时,我需要向用户显示一个JLabel(使其可见)。如果我最初不可见,那么JScrollPane应该自动滚动。这就是我提到scrollRectToVisible
的原因。
3) 在上面带有ImageIcon的JLabel中,我需要显示一条消息,解释这个元素发生了什么。由于JLayeredPane的层次结构更高,消息目前被实现为另一个带有浮动的JLabel。现在的问题是,如果用户滚动JScrollPane,浮动JLabel应该相应地移动到相应的JLabel(ImageIcon)之上。
UDP2:SSSCE
我还没有实现浮动JLabel,但我已经不喜欢索引为11的标签对scrollRectToVisible
的反应,因为它是半裁剪的。
import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.event.AdjustmentEvent;
import java.awt.event.AdjustmentListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.util.ArrayList;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.SwingUtilities;
public class JScrollPaneTest {
protected ArrayList<JLabel> labels = new ArrayList<JLabel>();
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
new JScrollPaneTest();
}
});
}
public JScrollPaneTest() {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new BorderLayout());
JPanel panel1 = new JPanel ();
panel1.setLayout(new GridBagLayout());
GridBagConstraints gbc = new GridBagConstraints();
for (int i = 0; i < 20; i++) {
JLabel label = new JLabel(" | Label" + i + " | ");
panel1.add(label, gbc);
labels.add(label);
}
panel1.addMouseListener(new MouseAdapter(){
public void mousePressed (MouseEvent me) {
JLabel label = labels.get(11);
label.scrollRectToVisible(label.getBounds());
}
});
JScrollPane pane = new JScrollPane(panel1) {
private static final long serialVersionUID = 1L;
@Override
public Dimension getPreferredSize() {
return new Dimension(600, 400);
}
};
pane.getHorizontalScrollBar().addAdjustmentListener(new AdjustmentListener() {
@Override
public void adjustmentValueChanged(AdjustmentEvent e) {
System.out.println("adjustmentValueChanged: " + e);
}
});
frame.getContentPane().add(pane);
frame.pack();
frame.setVisible(true);
}
});
}
}
部分答案(注释中的代码只是…并注意到我在注释中出错;-)
方法scrollRectToVisible在调用它的组件的坐标系中获取一个矩形。要滚动子项使其完全可见,基本上有两个选项(使用其中一个):
// either call on child with zero offset
child.scrollRectToVisible(new Rectangle(child.getSize());
// or call on child's parent with child bounds
child.getParent().scrollRectToVisible(child.getBounds());