Java JPanel内的JTextComponent内的JScrollPane内
我最近需要在一个JScrollPane的视口视图中放置几个组件,其中包括一个JTextPane。(如有兴趣,请参阅详细信息)。Java JPanel内的JTextComponent内的JScrollPane内,java,swing,jscrollpane,jtextcomponent,Java,Swing,Jscrollpane,Jtextcomponent,我最近需要在一个JScrollPane的视口视图中放置几个组件,其中包括一个JTextPane。(如有兴趣,请参阅详细信息)。 我将所有组件(两个JPanel和JTextPane)放在另一个JPanel中,这个JPanel有一个BorderLayout LayoutManager,并将该JPanel设置为滚动窗格的视口视图 我立刻注意到: JTextPane不再根据JScrollPane的宽度调整其宽度,即使JScrollPane设置为水平滚动条。相反,它只是将其宽度最大化,以便将内部的整个文本
我将所有组件(两个JPanel和JTextPane)放在另一个JPanel中,这个JPanel有一个BorderLayout LayoutManager,并将该JPanel设置为滚动窗格的视口视图 我立刻注意到:
JTextPane不再根据JScrollPane的宽度调整其宽度,即使JScrollPane设置为水平滚动条。相反,它只是将其宽度最大化,以便将内部的整个文本放入一行中(除了“\n”换行符) 下面是两张它的外观图片:
它应该是这样的:
现在,我找到了问题第一部分的答案。JPanel需要子类化,子类需要实现可滚动 然而,另一个问题由此产生。JPanel内部的JTextPane不再支持JPanel的BorderLayout.CENTER。如果文本未填充窗口中的剩余空间,JTextPane将不再拉伸自身 我正在寻找解决方案。
我希望有一个可靠的方法使JTextPane再次尊重JPanel的BorderLayout,并相应地进行扩展。我不是在寻找一些解决方法,比如将背景设置为白色,这样JTextPane看起来就像是在填充帧,而实际上不是 下面,我发布了一些代码,您可以在其中尝试不同的配置
import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.EventQueue;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.border.EmptyBorder;
import java.awt.GridLayout;
import java.awt.Rectangle;
import javax.swing.JLabel;
import javax.swing.JTextPane;
import javax.swing.JScrollPane;
import javax.swing.ScrollPaneConstants;
import javax.swing.Scrollable;
public class MinimalExample extends JFrame {
private JPanel contentPane;
private JPanel outerPanel;
private JPanel firstHeadlinePanel;
private JLabel lblSomeJlabel;
private JPanel secondHeadlinePanel;
private JLabel lblAnotherJlabel;
private JPanel headlinesPanel;
private JTextPane textPane;
private JScrollPane scrollPane;
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
public void run() {
try {
MinimalExample frame = new MinimalExample();
frame.setVisible(true);
}
catch (Exception e) {
e.printStackTrace();
}
}
});
}
public MinimalExample() {
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setBounds(100, 100, 230, 350);
contentPane = new JPanel();
contentPane.setBorder(null);
setContentPane(contentPane);
contentPane.setLayout(new GridLayout(1, 0, 0, 0));
outerPanel = new ScrollablePanel();
//outerPanel = new JPanel();
outerPanel.setLayout(new BorderLayout(0, 0));
headlinesPanel = new JPanel();
headlinesPanel.setBorder(new EmptyBorder(5, 0, 0, 0));
outerPanel.add(headlinesPanel, BorderLayout.NORTH);
headlinesPanel.setLayout(new BorderLayout(0, 0));
firstHeadlinePanel = new JPanel();
firstHeadlinePanel.setBorder(new EmptyBorder(0, 0, 5, 0));
headlinesPanel.add(firstHeadlinePanel, BorderLayout.NORTH);
firstHeadlinePanel.setLayout(new BorderLayout(0, 0));
lblSomeJlabel = new JLabel("Some JLabel");
firstHeadlinePanel.add(lblSomeJlabel, BorderLayout.CENTER);
secondHeadlinePanel = new JPanel();
headlinesPanel.add(secondHeadlinePanel, BorderLayout.SOUTH);
secondHeadlinePanel.setLayout(new BorderLayout(0, 0));
lblAnotherJlabel = new JLabel("Another JLabel");
lblAnotherJlabel.setBorder(new EmptyBorder(0, 0, 5, 0));
secondHeadlinePanel.add(lblAnotherJlabel, BorderLayout.NORTH);
textPane = new JTextPane();
textPane.setText(addSomeShortText());
//textPane.setText(addSomeLongText());
outerPanel.add(textPane);
scrollPane = new JScrollPane();
scrollPane.setHorizontalScrollBarPolicy(ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER);
scrollPane.setViewportView(outerPanel);
contentPane.add(scrollPane);
}
private String addSomeShortText() {
return ("Lorem ipsum dolor sit amet, consectetur adipiscing elit. Curabitur facilisis "
+ "dictum ullamcorper. Nulla sed diam sit amet ante pellentesque ultricies in non "
+ "est.");
}
private String addSomeLongText() {
return (" Lorem ipsum dolor sit amet, consectetur adipiscing elit. Curabitur facilisis "
+ "dictum ullamcorper. Nulla sed diam sit amet ante pellentesque ultricies in non "
+ "est. Suspendisse sit amet leo purus. Aenean lacinia ornare quam, a porttitor "
+ "tortor vulputate quis. Quisque suscipit porttitor nisl, mollis suscipit ligula "
+ "laoreet vitae. Nulla quis faucibus dui. Donec vestibulum quam sit amet "
+ "facilisis scelerisque. Pellentesque eu lacus eget nibh elementum venenatis in "
+ "congue magna. Donec et vestibulum magna, ut dapibus sapien.\n"
+ "Donec nec enim magna. Curabitur vitae est est. Etiam non tellus nec ipsum "
+ "dapibus ultricies eget ac lacus. Aliquam eu rhoncus leo. Suspendisse sodales "
+ "nibh eros, vel placerat velit bibendum ut. Vivamus feugiat purus at viverra "
+ "volutpat. Donec tincidunt quam quam, ut mattis enim rutrum sit amet. Integer "
+ "volutpat sit amet metus vitae elementum. Aenean ullamcorper iaculis vulputate. "
+ "Etiam venenatis mollis arcu quis interdum. Quisque et pharetra justo, et "
+ "ullamcorper augue. Phasellus id pellentesque massa. Proin suscipit, mauris et "
+ "congue commodo, mi eros fermentum ante, vel ultricies nisl neque a sem. Mauris "
+ "tristique quis erat vel pulvinar. ");
}
/* Original source of this is here:
* https://stackoverflow.com/questions/15783014/jtextarea-on-jpanel-inside-jscrollpane-does-not-resize-properly */
private static class ScrollablePanel extends JPanel implements Scrollable {
@Override
public Dimension getPreferredScrollableViewportSize() {
return super.getPreferredSize();
}
@Override
public int getScrollableUnitIncrement(Rectangle visibleRect, int orientation,
int direction) {
return 16;
}
@Override
public int getScrollableBlockIncrement(Rectangle visibleRect, int orientation,
int direction) {
return 16;
}
@Override
public boolean getScrollableTracksViewportWidth() {
return true;
}
@Override
public boolean getScrollableTracksViewportHeight() {
return false;
}
}
}
提前谢谢你能给我的任何帮助 好的,因此基本上,您需要根据
JViewport
和组件的高度更改getScrollableTracksViewPerthweight
的工作方式
基本上,当视口的高度大于组件的首选高度时,您希望组件跟踪视口的高度(调整自身以匹配)。当组件的首选高度大于视口的高度时,您希望停止跟踪并允许组件“溢出”滚动窗格,如
@覆盖
公共布尔值getScrollableTracksViewPerThight(){
布尔轨迹=真;
容器父级=getParent();
if(JViewport的父实例){
JViewport viewport=(JViewport)父对象;
if(viewport.getHeight()
这是像
JTable
和JList
这样的“基本”工作方式好的,因此基本上,您需要根据JViewport
和组件的高度更改getScrollableTracksViewPerthweight
的工作方式
基本上,当视口的高度大于组件的首选高度时,您希望组件跟踪视口的高度(调整自身以匹配)。当组件的首选高度大于视口的高度时,您希望停止跟踪并允许组件“溢出”滚动窗格,如
@覆盖
公共布尔值getScrollableTracksViewPerThight(){
布尔轨迹=真;
容器父级=getParent();
if(JViewport的父实例){
JViewport viewport=(JViewport)父对象;
if(viewport.getHeight()
这就是像
JTable
和JList
这样的东西“基本上”是如何工作的。先生,你跑得很快。就像,真的很快。我现在有点吃惊D我怀疑这可能是对getScrollableTracksViewPerThight()方法的一些补充,但我不知道那里到底需要什么。非常感谢你的快速回答!你甚至制作了一个动画图像,哇。多亏了你,这个网站又给我留下了深刻的印象如果你不能为自己的辉煌而炫目,那就用BS来迷惑自己吧;)-很高兴这有帮助;)哇!先生,你跑得很快。就像,真的很快。我现在有点吃惊D我怀疑这可能是对getScrollableTracksViewPerThight()方法的一些补充,但我不知道那里到底需要什么。非常感谢你的快速回答!你甚至制作了一个动画图像,哇。多亏了你,这个网站又给我留下了深刻的印象如果你不能为自己的辉煌而炫目,那就用BS来迷惑自己吧;)-很高兴这有帮助;)
@Override
public boolean getScrollableTracksViewportHeight() {
boolean track = true;
Container parent = getParent();
if (parent instanceof JViewport) {
JViewport viewport = (JViewport) parent;
if (viewport.getHeight() < getPreferredSize().height) {
track = false;
}
}
return track;
}