我正在编写一个文本编辑器,用户可以自由地将编辑器窗口垂直或水平地划分任意次数(即,分成任意数量的窗格)。单个窗口可以垂直和水平分割(例如,2行,一行包含3列,等等)。每个窗格都包含一个JScrollPane内的JTextArea和一个状态栏。
到目前为止,我的方法是使用嵌套的JSplitPanes。我一直在努力安排分割窗格分隔线,以便窗口中的空间在所有垂直或水平分割的窗格中平均分配。我已经非常接近于正确,但我不得不求助于在许多地方使用setPreferredSize()(Should I avoid the use of set[Preferred|Maximum|Minimum]Size methods in Java Swing?)。
我想知道完全采用不同的方法是否更容易/更好。 MultiSplitPane看起来很诱人......
对于我的情况,最好的布局/方法是什么?
答案 0 :(得分:1)
现在我对你想要的东西做了一些假设。我想你是否有一种简单的方法可以动态添加所有相同宽度/高度的文本窗格(垂直或水平分割,但不能混合两者)。
水平分割的示例
垂直分割的示例
如果是这种情况,我建议使用BoxLayout - 它完全没有任何配置。
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
public class SplitablePanel extends Box{
Box container;
Dimension minSize = new Dimension(400, 300);
public SplitablePanel(int axis){
super(BoxLayout.Y_AXIS);
//Container that holds all the text areas
container = new Box(axis);
container.setAlignmentX(Box.LEFT_ALIGNMENT);
add(container);
JTextArea text = new JTextArea();
container.add(new JScrollPane(text));
//Button to add another pane
JButton split = new JButton("Split");
split.setAlignmentX(Box.LEFT_ALIGNMENT);
split.addActionListener(new ActionListener(){
@Override
public void actionPerformed(ActionEvent e) {
JTextArea text = new JTextArea();
container.add(new JScrollPane(text));
revalidate();
}});
add(split);
//Button To switch Axis - more for demo purposes
JButton axisChanger = new JButton("Change Axis");
axisChanger.setAlignmentX(Box.LEFT_ALIGNMENT);
axisChanger.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
Box newContainer;
if(((BoxLayout)container.getLayout()).getAxis() == BoxLayout.X_AXIS){
newContainer = Box.createVerticalBox();
} else{
newContainer = Box.createHorizontalBox();
}
for(Component c : container.getComponents()){
container.remove(c);
newContainer.add(c);
}
remove(container);
add(newContainer, 0);
container = newContainer;
container.setAlignmentX(Box.LEFT_ALIGNMENT);
revalidate();
}
});
add(axisChanger);
}
@Override
public Dimension getPreferredSize() {
Dimension result = super.getPreferredSize();
result.width = result.width > minSize.width ? result.width : minSize.width;
result.height = result.height > minSize.height ? result.height : minSize.height;
return result;
}
public static void main(String[] args) {
final JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(new SplitablePanel(BoxLayout.X_AXIS));
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
}
答案 1 :(得分:1)
我决定花一些时间学习MultiSplitPane布局。这看起来是一个很好的解决方案。
这是我作为测试编写的代码。它本质上是一种布局的模拟,当用户以各种方式分割窗口时,该布局会动态变化。它有点冗长,但也许对尝试学习MultiSplitPane的人有帮助。
最终结果如下:
import java.awt.BorderLayout;
import java.awt.Component;
import java.awt.Container;
import java.awt.Dimension;
import java.util.LinkedList;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import org.jdesktop.swingx.MultiSplitPane;
import org.jdesktop.swingx.MultiSplitLayout.*;
@SuppressWarnings("serial")
class MultiSplitPaneTest extends JFrame {
private final static String sampleText;
static {
String text = "I'm working on a text editor in which the user is free to divide the editor window vertically or horizontally any number of times (ie, into any number of panes).\n";
StringBuilder sb = new StringBuilder();
for (int i = 0; i < 10; i++) {
sb.append(text);
}
sampleText = sb.toString();
}
private class MyScrollPane extends JScrollPane {
public MyScrollPane(final Component view) {
super(view);
}
@Override
public Dimension getPreferredSize() {
return new Dimension(1440, 900);
}
}
public MultiSplitPaneTest() {
// The application opens with a window containing a single pane (a single text area).
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
Container cp = getContentPane();
cp.setLayout(new BorderLayout());
JTextArea ta1 = new JTextArea();
ta1.setText("TEXT AREA 1\n" + sampleText);
MyScrollPane sp1 = new MyScrollPane(ta1);
sp1.setViewportView(ta1);
cp.add(sp1, BorderLayout.CENTER);
pack();
setLocationRelativeTo(null);
setVisible(true);
// -------------------------------------------------
// Let's say the user splits the window horizontally, creating a second pane.
// We'll simulate that with the following code.
JTextArea ta2 = new JTextArea();
ta2.setText("TEXT AREA 2\n" + sampleText);
MyScrollPane sp2 = new MyScrollPane(ta2);
sp2.setViewportView(ta2);
Leaf leaf1 = new Leaf("1");
Leaf leaf2 = new Leaf("2");
LinkedList<Node> rootChildren = new LinkedList<>();
rootChildren.add(leaf1);
rootChildren.add(new Divider());
rootChildren.add(leaf2);
Split root = new Split();
root.setRowLayout(true);
root.setChildren(rootChildren);
MultiSplitPane multiSplitPane = new MultiSplitPane();
multiSplitPane.getMultiSplitLayout().setModel(root);
multiSplitPane.add(sp1, "1");
multiSplitPane.add(sp2, "2");
cp.remove(sp1);
cp.add(multiSplitPane, BorderLayout.CENTER);
// --------------------------------------------------
// Let's say the user splits the window horizontally again, creating a new pane on the very left.
JTextArea ta3 = new JTextArea();
ta3.setText("TEXT AREA 3\n" + sampleText);
MyScrollPane sp3 = new MyScrollPane(ta3);
sp3.setViewportView(ta3);
Leaf leaf3 = new Leaf("3");
rootChildren.add(0, leaf3);
rootChildren.add(1, new Divider());
root.setChildren(rootChildren);
multiSplitPane.add(sp3, "3");
multiSplitPane.revalidate();
// --------------------------------------------------
// Let's say the user decides to remove the center pane (that is, the first pane that we started with).
rootChildren.remove(2); // Remove leaf1.
rootChildren.remove(2); // Remove the divider following leaf1.
root.setChildren(rootChildren);
multiSplitPane.remove(sp1);
multiSplitPane.revalidate();
// --------------------------------------------------
// Let's say the user creates another pane, this time splitting the pane on the right vertically.
rootChildren.remove(leaf2);
JTextArea ta4 = new JTextArea();
ta4.setText("TEXT AREA 4\n" + sampleText);
MyScrollPane sp4 = new MyScrollPane(ta4);
sp4.setViewportView(ta4);
Leaf leaf4 = new Leaf("4");
LinkedList<Node> branchChildren = new LinkedList<>();
branchChildren.add(leaf2);
branchChildren.add(new Divider());
branchChildren.add(leaf4);
Split branch = new Split();
branch.setRowLayout(false);
branch.setChildren(branchChildren);
rootChildren.add(branch);
root.setChildren(rootChildren);
multiSplitPane.add(sp4, "4");
multiSplitPane.revalidate();
}
}