java - 如何强制显示JScrollPane中的组件

标签 java swing jpanel jscrollpane

我在显示 JScrollPane 中的特定组件时遇到问题。我有带 GridLayout(1,0) 的水平 JScrollPane,它包含可变数量的 JPanels - 每个都包含图像。这就像 GIF 图像中帧的预览。我使用按钮在这些 JPanel 之间移动(通过更改边框并保留所选之一的索引),但我不知道如何强制 JScrollPane 在选择 JPanel 时向我显示它(如果可能的话将其居中)。

所以我想要这个 enter image description here

强制这样做: enter image description here

提前致谢!

编辑:几乎可以使用 scrollRectToVisible() 方法运行代码

public class MiniatursPanel extends JPanel{


private int indexOfChosenFrame = 0;
private ArrayList<JPanel> frames;
private JScrollPane scrollPane;
private JPanel innerPanel;



public MiniatursPanel(){
    setBorder(BorderFactory.createCompoundBorder(BorderFactory.createRaisedBevelBorder(),BorderFactory.createLoweredBevelBorder()));
    setPreferredSize(new Dimension(1200,170));
    setLayout(null);     
}

public void initialize(){
    int width = GifImageStats.getInstance().getWidth();
    int height = GifImageStats.getInstance().getHeight();
    int numberOfFrames = GifImageStats.getInstance().getNumberOfFrames();

    frames = new ArrayList(numberOfFrames);

    for (int i = 0; i < numberOfFrames; i++) {
        JPanel frameBox = new JPanel();
        frameBox.setLayout(new FlowLayout(FlowLayout.CENTER));
        JButton button = new JButton(String.valueOf(i+1));
        button.setPreferredSize(new Dimension(2*width,2*height));
        button.setBackground(Color.white);
        button.setFocusable(false);
        frameBox.add(button);
        frames.add(frameBox);
    }

    innerPanel = new JPanel();
    innerPanel.setLayout(new GridLayout(1,0,10,10));


    for (JPanel button : frames) {
        innerPanel.add(button);
    }

    scrollPane = new JScrollPane(innerPanel);
    scrollPane.setBounds(10, 10, 1180, 145);
    scrollPane.setHorizontalScrollBarPolicy(JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);
    scrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_NEVER);

    highlightFrame(frames.get(0));

    add(scrollPane);
}


public void nextFrame(){
    if (indexOfChosenFrame == frames.size() - 1) {           
        unhighlightFrame(frames.get(indexOfChosenFrame));
        indexOfChosenFrame = 0;
        highlightFrame(frames.get(0));
    }else{
        unhighlightFrame(frames.get(indexOfChosenFrame));
        indexOfChosenFrame++;
        highlightFrame(frames.get(indexOfChosenFrame));
    }
}

public void previousFrame(){
    if (indexOfChosenFrame == 0) {           
        unhighlightFrame(frames.get(0));
        indexOfChosenFrame = frames.size()-1;
        highlightFrame(frames.get(indexOfChosenFrame));
    }else{
        unhighlightFrame(frames.get(indexOfChosenFrame));
        indexOfChosenFrame--;
        highlightFrame(frames.get(indexOfChosenFrame));
    }
}

private void highlightFrame(JPanel frame){
    Rectangle rect = frame.getBounds();
    rect.setBounds(frame.getX()-550, frame.getY(), frame.getWidth()+1050, frame.getHeight());
    innerPanel.scrollRectToVisible(rect);     
    frame.setBorder(BorderFactory.createLineBorder(Color.red,2));
}          

private void unhighlightFrame(JPanel frame){
    frame.setBorder(null);
}

最佳答案

这里的相关方法是JComponent#scrollRectToVisible(Rectangle)。必须在滚动 Pane 视口(viewport)中的组件上调用它。 (在您的情况下,这是具有网格布局的面板,其中包含其他子面板)。

传递给这个方法的矩形可以是一个子面板的边界。在这种情况下,scoll Pane 将执行使给定矩形可见所需的“最小”滚动。如果您想确保相应的子面板位于中心,则可以增加此矩形的大小 - 也就是说,您以所需子面板位于中心的方式定义矩形。

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.GridLayout;
import java.awt.Rectangle;
import java.util.ArrayList;
import java.util.List;

import javax.swing.BorderFactory;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JSpinner;
import javax.swing.SpinnerNumberModel;
import javax.swing.SwingConstants;
import javax.swing.SwingUtilities;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;


public class ScrollToVisible
{
    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);

        int n = 20;
        final JPanel panel = new JPanel(new GridLayout(1,0));
        final List<JComponent> components = new ArrayList<JComponent>();
        for (int i=0; i<n; i++)
        {
            JComponent component = new JLabel(String.valueOf(i), SwingConstants.CENTER);
            component.setPreferredSize(new Dimension(100,100));
            component.setBorder(BorderFactory.createLineBorder(Color.BLACK));
            components.add(component);
            panel.add(component);
        }
        final JScrollPane scrollPane = new JScrollPane(panel);

        final JSpinner spinner = new JSpinner(new SpinnerNumberModel(0, 0, n-1, 1));
        spinner.addChangeListener(new ChangeListener()
        {
            JComponent selectedComponent = components.get(0);

            @Override
            public void stateChanged(ChangeEvent e)
            {
                selectedComponent.setBorder(BorderFactory.createLineBorder(Color.BLACK));

                int index = (Integer)spinner.getValue();
                JComponent component = components.get(index);
                Rectangle bounds = component.getBounds();

                // This would make the component "just" visible:
                //panel.scrollRectToVisible(bounds);

                // This will center the component:
                int cx = bounds.x + bounds.width / 2;
                int w = scrollPane.getViewport().getWidth();
                Rectangle r = new Rectangle(cx-w/2, bounds.y, w, bounds.height);
                panel.scrollRectToVisible(r);


                selectedComponent = component;
                selectedComponent.setBorder(BorderFactory.createLineBorder(Color.RED));

            }
        });

        f.getContentPane().setLayout(new BorderLayout());
        f.getContentPane().add(scrollPane, BorderLayout.CENTER);
        f.getContentPane().add(spinner, BorderLayout.NORTH);


        f.setSize(800, 300);
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

}

编辑:您应该使用setLayout(null),并且您应该手动调用setBounds,您应该很少使用setPreferredSize。而且...当您发布已经非常接近 https://stackoverflow.com/help/mcve 的代码时(或者甚至是从另一篇文章的可运行示例创建的)那么您应该使其真正可运行。重新插入样板代码并浪费一些时间进行调试很烦人,直到您意识到根本没有调用 initialize()...

但是,按照这个改代码:

private void highlightFrame(JPanel frame){
    Rectangle rect = frame.getBounds();

    int c = rect.x + rect.width / 2; 
    int w = scrollPane.getViewport().getWidth();
    int x = c-w/2;
    rect.setBounds(x, rect.y, w, rect.height);

    innerPanel.scrollRectToVisible(rect);     
    frame.setBorder(BorderFactory.createLineBorder(Color.red,2));
}          

private void unhighlightFrame(JPanel frame){
    frame.setBorder(BorderFactory.createEmptyBorder(2, 2, 2, 2));
}

最重要的是确保组件的大小正确,方法是设置一个与“突出显示”边框大小相同的空边框。

关于java - 如何强制显示JScrollPane中的组件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22504355/

相关文章:

java - Exchange Web 服务 Java API + RESTful 推送通知监听器

java - 如何获取java swing项目的项目路径

java - Netbeans:如何指定 JLabel 的宽度?

java - 字符串数组拆分而不删除

java - sql 查询在 Hibernate 中很慢,在 mysql 上很快

java - JSwing 拆分 Pane 不会显示

java - 在另一个 JFrame 上显示 JPanel

java - JPanels 的负坐标(并且缺乏与 Timer 的配合)

java - 选择 GQLQueries (JDO) Google app Engine JAVA 中的特定行

java - 如何获取 JTable 中显示的单元格值(不是完整的 TableModel 值)?