我有一个带有大面板的Swing应用程序,它包含在JScrollPane
中。用户通常通过Tab键在面板的子组件之间移动,因此当他们选择显示视图时,我希望滚动窗格自动滚动,以便具有输入焦点的组件始终可见。
我尝试使用KeyboardFocusManager
来监听输入焦点更改,然后调用scrollRectToVisible
。
这是显示我当前策略的SSCCE(只需复制/粘贴并运行!):
import java.awt.KeyboardFocusManager;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import javax.swing.*;
public class FollowFocus {
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
final int ROWS = 100;
final JPanel content = new JPanel();
content.setLayout(new BoxLayout(content, BoxLayout.Y_AXIS));
content.add(new JLabel(
"Thanks for helping out. Use tab to move around."));
for (int i = 0; i < ROWS; i++) {
JTextField field = new JTextField("" + i);
field.setName("field#" + i);
content.add(field);
}
KeyboardFocusManager.getCurrentKeyboardFocusManager()
.addPropertyChangeListener("focusOwner",
new PropertyChangeListener() {
@Override
public void propertyChange(PropertyChangeEvent evt) {
if (!(evt.getNewValue() instanceof JComponent)) {
return;
}
JComponent focused = (JComponent) evt.getNewValue();
if (content.isAncestorOf(focused)) {
System.out.println("Scrolling to " + focused.getName());
focused.scrollRectToVisible(focused.getBounds());
}
}
});
JFrame window = new JFrame("Follow focus");
window.setContentPane(new JScrollPane(content));
window.setSize(200, 200);
window.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
window.setVisible(true);
}
});
}
}
如果你运行这个例子,你会发现它不能很好地工作。它确实获得焦点更改通知,但对scrollRectToVisible
的调用似乎没有任何效果。在我的应用程序中(这里显示起来太复杂了),scrollRectToVisible
大约有一半的时间是我在视口外面的东西。
有没有一种既定方法可以解决这个问题?如果它有所不同,Swing应用程序是基于Netbeans RCP(我们的大多数客户运行Windows)。
答案 0 :(得分:13)
我对其他答案的评论:
组件本身的scrollRectToVisible是其中的重点 方法;-)它传递给层次结构,直到父母做了 找到滚动
...除非组件本身处理它 - 正如JTextField所做的那样:它实现为水平滚动以使插入符号可见。解决方法是在字段的父级上调用方法。
修改强>
为了清楚起见,替换的行是
content.scrollRectToVisible(focused.getBounds());
答案 1 :(得分:3)
您必须从Rectangle
和JPanel
获取JViewPort
,然后进行比较,例如
通知(反对向下投票)需要对JViewPort
import java.awt.KeyboardFocusManager;
import java.awt.Rectangle;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import javax.swing.*;
//http://stackoverflow.com/questions/8245328/how-do-i-make-jscrollpane-scroll-to-follow-input-focus
public class FollowFocus {
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
final int ROWS = 100;
final JPanel content = new JPanel();
content.setLayout(new BoxLayout(content, BoxLayout.Y_AXIS));
content.add(new JLabel(
"Thanks for helping out. Use tab to move around."));
for (int i = 0; i < ROWS; i++) {
JTextField field = new JTextField("" + i);
field.setName("field#" + i);
content.add(field);
}
final JScrollPane scroll = new JScrollPane(content);
KeyboardFocusManager.getCurrentKeyboardFocusManager().
addPropertyChangeListener("focusOwner", new PropertyChangeListener() {
@Override
public void propertyChange(PropertyChangeEvent evt) {
if (!(evt.getNewValue() instanceof JComponent)) {
return;
}
JViewport viewport = (JViewport) content.getParent();
JComponent focused = (JComponent) evt.getNewValue();
if (content.isAncestorOf(focused)) {
System.out.println("Scrolling to " + focused.getName());
Rectangle rect = focused.getBounds();
Rectangle r2 = viewport.getVisibleRect();
content.scrollRectToVisible(new Rectangle(rect.x, rect.y, (int) r2.getWidth(), (int) r2.getHeight()));
}
}
});
JFrame window = new JFrame("Follow focus");
window.setContentPane(new JScrollPane(content));
window.setSize(200, 200);
window.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
window.setVisible(true);
}
});
}
}
答案 2 :(得分:0)
您的代码中的一个主要问题是:
focused.scrollRectToVisible(focused.getBounds());
您正在调用组件本身的scrollRectToVisible!大概是一个错字。 使您的JScrollPane成为最终变量并调用
scrollPane.getViewport().scrollRectToVisible(focused.getBounds());
答案 3 :(得分:0)
此处jtextbox是您要关注的组件,jscrollpane是您的滚动窗格:
jScrollpane.getVerticalScrollBar().setValue(jtextbox.getLocation().x);
答案 4 :(得分:0)
这是我的简短摘要。 将此添加到您的工具类:
public static void addOnEnter(Component c, Consumer<FocusEvent> onEnter) {
FocusListener fl = new FocusListener() {
@Override
public void focusGained(FocusEvent e) {
onEnter.accept(e);
}
@Override
public void focusLost(FocusEvent e) { }
};
c.addFocusListener(fl);
}
public static void scrollToFocus(FocusEvent e) {
((JComponent) e.getComponent().getParent()).scrollRectToVisible(
e.getComponent().getBounds());
}
并像这样使用它:
Tools.addOnEnter(component, Tools::scrollToFocus);
组件可以是JTextField,JButton等...