how can you programmatically set the JSplitPane to hide the right/bottom component when OneTouchExpandable is set to true?

前端 未结 5 1027
遥遥无期
遥遥无期 2021-01-06 06:30

In a JSplitPane, you have the setOneTouchExpandable method which provides you with 2 buttons to quickly fully hide or full show the JSplitPan

相关标签:
5条回答
  • 2021-01-06 06:39

    Working around the problem that setDividerLocation(1.0) doesn't work until the frame has become displayable, one can use an AncestorListener:

    sp.addAncestorListener(new AncestorListener {
      def ancestorAdded  (event: AncestorEvent): Unit = sp.setDividerLocation(1.0)
    
      def ancestorRemoved(event: AncestorEvent): Unit = ()
      def ancestorMoved  (event: AncestorEvent): Unit = ()
    })
    
    0 讨论(0)
  • 2021-01-06 06:44
    import javax.swing.*;
    
    class SplitPaneDefault {
        public static void main(String[] args) {
            SwingUtilities.invokeLater(new Runnable() {
                public void run() {
                    JSplitPane sp = new JSplitPane(
                        JSplitPane.HORIZONTAL_SPLIT,
                        new JTree(),
                        new JTree());
                    sp.setOneTouchExpandable(true);
                    sp.setDividerLocation(0.0);
                    JOptionPane.showMessageDialog(null, sp);
                }
            });
        }
    }
    

    replace 0.0 with 1.0 and you get my problem

    Read the fine manual and solve the problem.

    This method immediately changes the size of the split pane based on its current size. If the split pane is not correctly realized and on screen, this method will have no effect ...

    SplitPaneDefault

    import javax.swing.*;
    
    class SplitPaneDefault {
        public static void main(String[] args) {
            SwingUtilities.invokeLater(new Runnable() {
                public void run() {
                    JSplitPane sp = new JSplitPane(
                        JSplitPane.HORIZONTAL_SPLIT,
                        new JTree(),
                        new JTree());
                    sp.setOneTouchExpandable(true);
                    JFrame f = new JFrame("Split Pane To Right");
                    f.add(sp);
                    f.pack();
                    // sp now has a non-zero size!
                    sp.setDividerLocation(1.0);
                    f.setLocationByPlatform(true);
                    f.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
                    f.setVisible(true);
                }
            });
        }
    }
    
    0 讨论(0)
  • 2021-01-06 06:50

    You can simply use this:

    public void setDividerLocation(double proportionalLocation)
    
    splitPane.setDividerLocation(0.0d);
    

    or.

    splitPane.setDividerLocation(1.0d);
    

    depending on wheter yourwant to hide the left component first or the right component.

    0 讨论(0)
  • 2021-01-06 06:52

    Here is another solution, maybe a little bit dirty, but it works ;) I hope the code speaks for itself.

    public class ExtOneTouchJSplitPane extends JSplitPane {
        private static final long serialVersionUID = -2320161961382260438L;
    
        JButton jBLeftUp;
        JButton jBRightDown;
    
        public ExtOneTouchJSplitPane() {
            super();
            setOneTouchExpandable(true);
            extractDividerButtons();
        }
    
        public ExtOneTouchJSplitPane(int newOrientation) {
            super(newOrientation);
            setOneTouchExpandable(true);
            extractDividerButtons();
        }
    
        public ExtOneTouchJSplitPane(int newOrientation, boolean newContinuousLayout) {
            super(newOrientation, newContinuousLayout);
            setOneTouchExpandable(true);
            extractDividerButtons();
        }
    
        public ExtOneTouchJSplitPane(int newOrientation, boolean newContinuousLayout, Component newLeftComponent, Component newRightComponent) {
            super(newOrientation, newContinuousLayout, newLeftComponent, newRightComponent);
            setOneTouchExpandable(true);
            extractDividerButtons();
        }
    
        public ExtOneTouchJSplitPane(int newOrientation, Component newLeftComponent, Component newRightComponent) {
            super(newOrientation, newLeftComponent, newRightComponent);
            setOneTouchExpandable(true);
            extractDividerButtons();
        }
    
        private void extractDividerButtons() {
            BasicSplitPaneUI ui = (BasicSplitPaneUI) getUI();
            jBLeftUp = (JButton) ui.getDivider().getComponent(0);
            jBRightDown = (JButton) ui.getDivider().getComponent(1);
        }
    
        public void oneTouchClickLeft() {
            jBLeftUp.doClick();
        }
    
        public void oneTouchClickRight() {
            jBRightDown.doClick();
        }
    
        public void oneTouchClickUp() {
            jBLeftUp.doClick();
        }
    
        public void oneTouchClickDown() {
            jBRightDown.doClick();
        }
    }
    

    And an example how to use it:

    public class SplitPaneDemo extends JFrame implements Runnable {
    
        public static void main(String[] args) {
            SwingUtilities.invokeLater(new SplitPaneDemo());
        }
    
        ExtOneTouchJSplitPane hSplitPane;
        ExtOneTouchJSplitPane vSplitPane;
    
        public SplitPaneDemo() {
            createView();
        }
    
        public void createView() {
            setTitle("SplitPane-Demo");
            setLayout(new BorderLayout(0, 0));
    
            hSplitPane = new ExtOneTouchJSplitPane();
            JButton jBLeft = new JButton("<html><body> &nbsp;<br>Left Component<br> &nbsp;</body></html>");
            JButton jBRight = new JButton("<html><body> &nbsp;<br>Right Component<br> &nbsp;</body></html>");
            jBLeft.addActionListener(new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    hSplitPane.oneTouchClickLeft();
                }
            });
            jBRight.addActionListener(new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    hSplitPane.oneTouchClickRight();
                }
            });
            hSplitPane.setLeftComponent(jBLeft);
            hSplitPane.setRightComponent(jBRight);
    
            add(hSplitPane, BorderLayout.CENTER);
    
            vSplitPane = new ExtOneTouchJSplitPane(JSplitPane.VERTICAL_SPLIT);
            JButton jBUp = new JButton("<html><body> &nbsp;<br>Up Component<br> &nbsp;</body></html>");
            JButton jBDown = new JButton("<html><body> &nbsp;<br>Down Component<br> &nbsp;</body></html>");
            jBUp.addActionListener(new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    vSplitPane.oneTouchClickUp();
                }
            });
            jBDown.addActionListener(new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    vSplitPane.oneTouchClickDown();
                }
            });
            vSplitPane.setTopComponent(jBUp);
            vSplitPane.setBottomComponent(jBDown);
    
            add(vSplitPane, BorderLayout.SOUTH);
        }
    
        @Override
        public void run() {
            setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
            setSize(400, 400);
            setVisible(true);
    
            hSplitPane.oneTouchClickLeft();
        }
    }
    
    0 讨论(0)
  • 2021-01-06 06:54

    @0__'s answer is a hint that you should be using the AncestorListener to set the divider location, and have it taken into account (ComponentListener is not enough, I'm not sure why).

    However, it's not sufficient: if the split plane somehow gets resized (e.g. because its layout manager decided it should, when the frame has been resized), a tiny fraction of the component you wanted to hide will still show. That's due to the component minimum size not being zero. It can be addressed by zeroing it with setMinimumSize(new Dimension()) (as explained in that other answer), but if that's not an option, you can hack into the split pane UI:

    If you're using the standard BasicSplitPaneUI, you can hack its keepHidden boolean field and force it to true, so the divider will stick to either side:

    sp.addAncestorListener(new AncestorListener() {
        @Override
        public void ancestorAdded(AncestorEvent event) {
            sp.setDividerLocation(1.0); // Divider is positioned
            Field m = BasicSplitPaneUI.class.getDeclaredField("keepHidden");
            m.setAccessible(true);
            m.set(sp.getUI(), true); // Divider position will stick
            //sp.removeAncestorListener(this); // Uncomment for a one-shot event
        }
    
        @Override public void ancestorRemoved(AncestorEvent event) { }
        @Override public void ancestorMoved(AncestorEvent event) { }
    });
    
    0 讨论(0)
提交回复
热议问题