All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.jidesoft.swing.JideSplitPaneLayout Maven / Gradle / Ivy

There is a newer version: 3.6.18
Show newest version
/*
 * @(#)JideSplitPaneLayout.java 7/7/2009
 *
 * Copyright 2002 - 2009 JIDE Software Inc. All rights reserved.
 *
 */

package com.jidesoft.swing;

import javax.swing.*;
import java.awt.*;
import java.util.*;

/**
     * Layout manager used by JideSplitPane.
     */
public class JideSplitPaneLayout extends JideBoxLayout {
    private static final long serialVersionUID = -1826651835409198865L;

    public JideSplitPaneLayout(Container target) {
        super(target);
        setResetWhenInvalidate(false);
    }

    public JideSplitPaneLayout(Container target, int axis) {
        super(target, axis);
        setResetWhenInvalidate(false);
    }

    public JideSplitPaneLayout(Container target, int axis, int gap) {
        super(target, axis, gap);
        setResetWhenInvalidate(false);
    }

    int getDividerLocation(int index) {
        if (_componentSizes == null) {
            return -1;
        }
        if (index < 0 || (index + 1) << 1 >= _componentSizes.length)
            return -1;

        boolean ltr = _target.getComponentOrientation().isLeftToRight();
        boolean reversed = !ltr && ((JideSplitPane) _target).getOrientation() == JideSplitPane.HORIZONTAL_SPLIT;

        int location = 0;
        if (reversed) {
            for (int i = _componentSizes.length - 1; i >= (index * 2) + 1; i--)
                location += _componentSizes[i];
        }
        else {
            for (int i = 0; i < (index * 2) + 1; i++)
                location += _componentSizes[i];
        }

        Insets insets = _target.getInsets();
        if (insets != null) {
            if (((JideSplitPane) _target).getOrientation() == JideSplitPane.HORIZONTAL_SPLIT) {
                location += reversed ? insets.right : insets.left;
            }
            else {
                location += insets.top;
            }
        }
        return location;
    }

    @SuppressWarnings({"RedundantCast"})
    int setDividerLocation(int index, int location, boolean isOriginator) {
        int oldLocation = getDividerLocation(index);
        if (oldLocation == -1 || oldLocation == location)
            return -1;
        boolean ltr = _target.getComponentOrientation().isLeftToRight();
        boolean reversed = !ltr && ((JideSplitPane) _target).getOrientation() == JideSplitPane.HORIZONTAL_SPLIT;
        int prevIndex = reversed ? 2 * index + 2 : 2 * index;
        int nextIndex = reversed ? 2 * index : 2 * index + 2;
        int nextDividerIndex = reversed ? index - 1 : index + 1;
        int prevDividerIndex = reversed ? index + 1 : index - 1;
        int flexibleNextIndex;
        int flexiblePrevIndex;
        java.util.List componentIndexChanged = new ArrayList();

        if (reversed) {
            while (nextIndex >= 0 && !isPaneVisible(nextIndex)) {
                nextIndex -= 2;
                nextDividerIndex --;
            }

            while (prevIndex < _componentSizes.length && !isPaneVisible(prevIndex)) {
                prevIndex += 2;
                prevDividerIndex ++;
            }

            flexibleNextIndex = nextIndex;
            while (flexibleNextIndex >= 0 &&
                    (getConstraintMap().get(_target.getComponent(flexibleNextIndex)) == JideBoxLayout.FIX || !isPaneVisible(flexibleNextIndex))) {
                flexibleNextIndex -= 2;
            }
            if (flexibleNextIndex < 0) {
                return -1;
            }

            flexiblePrevIndex = prevIndex;
            while (flexiblePrevIndex < _componentSizes.length &&
                    (getConstraintMap().get(_target.getComponent(flexiblePrevIndex)) == JideBoxLayout.FIX || !isPaneVisible(flexiblePrevIndex))) {
                flexiblePrevIndex += 2;
            }
            if (flexiblePrevIndex >= _componentSizes.length) {
                return -1;
            }
        }
        else {
            while (nextIndex < _componentSizes.length && !isPaneVisible(nextIndex)) {
                nextIndex += 2;
                nextDividerIndex ++;
            }

            while (prevIndex >= 0 && !isPaneVisible(prevIndex)) {
                prevIndex -= 2;
                prevDividerIndex --;
            }

            flexibleNextIndex = nextIndex;
            while (flexibleNextIndex < _componentSizes.length &&
                    (getConstraintMap().get(_target.getComponent(flexibleNextIndex)) == JideBoxLayout.FIX || !isPaneVisible(flexibleNextIndex))) {
                flexibleNextIndex += 2;
            }
            if (flexibleNextIndex >= _componentSizes.length) {
                return -1;
            }

            flexiblePrevIndex = prevIndex;
            while (flexiblePrevIndex >= 0 &&
                    (getConstraintMap().get(_target.getComponent(flexiblePrevIndex)) == JideBoxLayout.FIX || !isPaneVisible(flexiblePrevIndex))) {
                flexiblePrevIndex -= 2;
            }
            if (flexiblePrevIndex < 0) {
                return -1;
            }
        }

        if (isOriginator
                && getConstraintMap().get(_target.getComponent(nextIndex)) == JideBoxLayout.FIX
                && getConstraintMap().get(_target.getComponent(prevIndex)) == JideBoxLayout.FIX) {
            return -1;
        }

        if (location > oldLocation) {
            int size = _componentSizes[2 * index + 1];
            if (getConstraintMap().get(_target.getComponent(nextIndex)) == JideBoxLayout.FIX) {
                size += _componentSizes[nextIndex];
            }
            else {
                if (((JideSplitPane) _target).getOrientation() == JideSplitPane.HORIZONTAL_SPLIT) {
                    size += _target.getComponent(nextIndex).getMinimumSize().getWidth();
                }
                else {
                    size += _target.getComponent(nextIndex).getMinimumSize().getHeight();
                }
            }

            int nextDividerLocation = getDividerLocation(nextDividerIndex);
            if (nextDividerLocation < 0) {
                if (((JideSplitPane) _target).getOrientation() == JideSplitPane.HORIZONTAL_SPLIT) {
                    location = Math.min(location, _target.getWidth() - size);
                }
                else {
                    location = Math.min(location, _target.getHeight() - size);
                }
            }
            else if (location + size > nextDividerLocation) {
                int actualLocation = setDividerLocation(nextDividerIndex, location + size, false);
                if (actualLocation == -1) {
                    return -1;
                }
                location = actualLocation - size;
            }
            if (getConstraintMap().get(_target.getComponent(nextIndex)) != JideBoxLayout.FIX) {
                _componentSizes[nextIndex] -= location - oldLocation;
                componentIndexChanged.add(nextIndex);
            }
            if (isOriginator) {
                _componentSizes[flexiblePrevIndex] += location - oldLocation;
                componentIndexChanged.add(flexiblePrevIndex);
            }
            else if (getConstraintMap().get(_target.getComponent(prevIndex)) != JideBoxLayout.FIX) {
                _componentSizes[prevIndex] += location - oldLocation;
                componentIndexChanged.add(prevIndex);
            }
        }
        else if (location < oldLocation) {
            int size = 0;
            if (prevDividerIndex >= 0) {
                size = _componentSizes[prevIndex - 1];
            }
            if (getConstraintMap().get(_target.getComponent(prevIndex)) == JideBoxLayout.FIX) {
                size += _componentSizes[prevIndex];
            }
            else {
                if (((JideSplitPane) _target).getOrientation() == JideSplitPane.HORIZONTAL_SPLIT) {
                    size += _target.getComponent(prevIndex).getMinimumSize().getWidth();
                }
                else {
                    size += _target.getComponent(prevIndex).getMinimumSize().getHeight();
                }
            }

            int prevDividerLocation = getDividerLocation(prevDividerIndex);
            if (prevDividerLocation < 0) {
                location = Math.max(location, size);
            }
            else if (location - size < prevDividerLocation) {
                int actualLocation = setDividerLocation(prevDividerIndex, location - size, false);
                if (actualLocation == -1) {
                    return -1;
                }
                location = actualLocation + size;
            }
            if (getConstraintMap().get(_target.getComponent(prevIndex)) != JideBoxLayout.FIX) {
                _componentSizes[prevIndex] -= oldLocation - location;
                componentIndexChanged.add(prevIndex);
            }
            if (isOriginator) {
                _componentSizes[flexibleNextIndex] += oldLocation - location;
                componentIndexChanged.add(flexibleNextIndex);
            }
            else if (getConstraintMap().get(_target.getComponent(nextIndex)) != JideBoxLayout.FIX) {
                _componentSizes[nextIndex] += oldLocation - location;
                componentIndexChanged.add(nextIndex);
            }
        }

        _target.firePropertyChange(JideSplitPane.PROPERTY_DIVIDER_LOCATION, oldLocation, location);
        ((JideSplitPane) _target).revalidate();

        if (((JideSplitPane) _target).isProportionalLayout()) {
            replaceProportions();
            return location;
        }

        if (((JideSplitPane) _target).getOrientation() == JideSplitPane.HORIZONTAL_SPLIT) {
            for (int changedIndex : componentIndexChanged) {
                Component component = _target.getComponent(changedIndex);
                if (component instanceof JComponent) {
                    ((JComponent) component).setPreferredSize(new Dimension(_componentSizes[changedIndex], component.getPreferredSize().height));
                }
            }
        }
        else {
            for (int changedIndex : componentIndexChanged) {
                Component component = _target.getComponent(changedIndex);
                if (component instanceof JComponent) {
                    ((JComponent) component).setPreferredSize(new Dimension(component.getPreferredSize().width, _componentSizes[changedIndex]));
                }
            }
        }
        return location;
    }

    private boolean isPaneVisible(int index) {
        // Considering collapse/expand feature, we need consider the visibility of divider as well
        return index < _componentSizes.length && (_componentSizes[index] != 0 || index - 1 < 0 || _componentSizes[index - 1] != 0);
    }

    /**
     * Uses the component sizes to generate a new array of proportions, and replaces the existing one.
     */
    private void replaceProportions() {
        ((JideSplitPane) _target).setProportions(deduceProportions());
    }

    /**
     * Uses the component sizes to generate a new array of proportions.
     *
     * @return a new array of proportions
     */
    private double[] deduceProportions() {
        double total = 0.0; // Total height or width of contained panes (not including dividers)
        for (int i = 0; i < _componentSizes.length; i += 2) // Note we're skipping dividers
            total += _componentSizes[i];
        double[] newProportions;
        if (total == 0.0)
            newProportions = null;
        else {
            newProportions = new double[(_componentSizes.length - 1) / 2];
            for (int i = 0; i < newProportions.length; ++i)
                newProportions[i] = _componentSizes[i * 2] / total;
        }
        return newProportions;
    }

    // For proportional layouts, override the key box layout method:
    @Override
    protected boolean calculateComponentSizes(int availableSize, int startIndex, int endIndex) {
        // Just go to super for non-proportional layout, or if there are no more
        // than one component.
        if (!((JideSplitPane) _target).isProportionalLayout()
                || _target.getComponentCount() <= 1)
            return super.calculateComponentSizes(availableSize, startIndex, endIndex);

        // If we have no set proportions, either call super to get initial
        // proportions or set them up as even.
        if (((JideSplitPane) _target).getProportions() == null) {
            if (!((JideSplitPane) _target).isInitiallyEven()) {
                if (!super.calculateComponentSizes(availableSize, startIndex, endIndex))
                    return false;
// Luke's change
//   David: We now wait to set proportions until the user sets them directly.  Otherwise recreated
//   docked frames end up with tiny sizes instead of using preferred size, since they are
//   initially added to their ContainerContainers without their contents.
//                    _proportions = deduceProportions(); // Note no call to setProportions: no event
//
                return true;
            }
            // What remains is the "initially even" logic:
            int c = ((JideSplitPane) _target).getPaneCount();
            double[] p = new double[c - 1];
            for (int i = 0; i < p.length; ++i)
                p[i] = 1.0 / c;
            ((JideSplitPane) _target).internalSetProportions(p); // Note no call to setProportions: no event may be sent here
        }

        // Spin through the dividers, setting up their sizes and subtracting from available.
        for (int i = 1; i < _target.getComponentCount(); i += 2) {
            if (_target.getComponent(i).isVisible())
                _componentSizes[i] = ((JideSplitPane) _target).getDividerSize();
            availableSize -= _componentSizes[i];
        }

        if (availableSize < 0)
            return false;

        // Then spin through the panes and set their sizes according to the
        // proportions.
        double[] proportions = ((JideSplitPane) _target).getProportions();
//            double last = 1.0;
//            for (double p : proportions) last -= p;
//            double total = 0.0;
//            for (int i = 0; i < getComponentCount(); i += 2) {
//                int j = i / 2;
//                if (getComponent(i).isVisible())
//                    total += (j < proportions.length) ? proportions[j] : last;
//            }
        int size = availableSize;
        for (int i = 0; i < proportions.length; ++i) {
            int j = i * 2;
            if (_target.getComponent(j).isVisible()) {
                double d = proportions[i]/* / total*/;
                if (d <= 1.0)
                    _componentSizes[j] = (int) (0.5 + size * d);
            }
            availableSize -= _componentSizes[j];
        }
        // Now set the last one to whatever is left over.
        if (availableSize < 0)
            return false;

        _componentSizes[_componentSizes.length - 1] = availableSize;
        return true;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy