我的GUI在JTextPane中显示一个日志,该日志位于JScrollPane中,占据JSplitPane的左侧。 JSplitPane的右侧包含GUI的有趣部分。日志窗格的宽度通常很窄(20),但在用户想要查看活动时会增加。如果日志文本被包装,则当日志窗格很窄时,显示的文本将毫无用处,但如果某些日志条目很长,则需要增加日志窗格的大小以占用大部分GUI以查看条目。
我知道JTextPane不会自然地换文本,而Stack Overflow上的其他帖子也描述了如何包装文本。但是,我希望只有当窗格比某些提供的限制更宽时才会包装文本,例如100.因此,当窗格很窄时,只显示每条消息的开头,如果用户想要查看完整的文本,窗格宽度增加,文本被包裹到窗口宽度。
如果无法做到这一点,我会在将文本写入窗格之前关闭包装并将文本格式化为某个固定宽度。 提前谢谢!
答案 0 :(得分:2)
我知道JTextPane不会自然地换行文本
那是错的。默认情况下,JTextPane会包装文本。
查看Text Pane No Wrap以获取有关此主题的更多信息,该信息构成了以下解决方案的基础。
您需要覆盖getScrollableTrackViewportWidth()
方法。我认为应该是:
JTextPane textPane = new JTextPane()
{
public boolean getScrollableTracksViewportWidth()
{
return getParent().getSize().width > 100;
}
};
编辑:
此问题似乎与文本窗格中的文本有关。不知道文本的长度或文本中的非字母字符是否会造成差异:
import java.awt.*;
import javax.swing.*;
import javax.swing.event.*;
public class TextPaneNoWrap
{
private static void createAndShowGUI()
{
JTextPane textPane = new JTextPane()
{
@Override
public boolean getScrollableTracksViewportWidth()
{
Dimension d = getParent().getSize();
System.out.println(d);
return false;
}
};
textPane.setText("one two three four five six\nthis wraps at 39");
// textPane.setText("import javax.swing.event.*;\nthis wraps at 111");
JFrame frame = new JFrame("TextPaneNoWrap");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(new JLabel("Some Component"), BorderLayout.LINE_START);
frame.add(new JScrollPane( textPane ));
frame.setSize(300, 100);
frame.setLocationByPlatform( true );
frame.setVisible( true );
}
public static void main(String[] args)
{
EventQueue.invokeLater( () -> createAndShowGUI() );
/*
EventQueue.invokeLater(new Runnable()
{
public void run()
{
createAndShowGUI();
}
});
*/
}
}
EDIT2:
使用"包装器"似乎工作正常实现Scrollable接口的面板:
import java.awt.*;
import javax.swing.*;
import javax.swing.event.*;
public class TextPaneNoWrap
{
private static void createAndShowGUI()
{
JTextPane textPane = new JTextPane();
// textPane.setText("one two three four five six\nthis wraps at 39");
textPane.setText("import javax.swing.event.*;\nthis wraps at 111");
JPanel wrapper = new NoWrapPanel(textPane);
JFrame frame = new JFrame("TextPaneNoWrap");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(new JLabel("Some Component"), BorderLayout.LINE_START);
frame.add(new JScrollPane( wrapper ));
frame.setSize(300, 100);
frame.setLocationByPlatform( true );
frame.setVisible( true );
}
static class NoWrapPanel extends JPanel implements Scrollable
{
public NoWrapPanel(JComponent component)
{
setLayout( new BorderLayout() );
add( component );
}
@Override
public Dimension getPreferredScrollableViewportSize()
{
return getPreferredSize();
}
@Override
public int getScrollableBlockIncrement(Rectangle visibleRect, int orientation, int direction)
{
return 20;
}
@Override
public boolean getScrollableTracksViewportHeight()
{
return true;
}
@Override
public boolean getScrollableTracksViewportWidth()
{
// return false;
return getParent().getSize().width > 100;
}
@Override
public int getScrollableUnitIncrement(Rectangle visibleRect, int orientation, int direction)
{
return 10;
}
}
public static void main(String[] args)
{
EventQueue.invokeLater( () -> createAndShowGUI() );
/*
EventQueue.invokeLater(new Runnable()
{
public void run()
{
createAndShowGUI();
}
});
*/
}
}