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

com.alee.extended.split.WebMultiSplitPaneModel Maven / Gradle / Ivy

There is a newer version: 1.2.14
Show newest version
/*
 * This file is part of WebLookAndFeel library.
 *
 * WebLookAndFeel library is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * WebLookAndFeel library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with WebLookAndFeel library.  If not, see .
 */

package com.alee.extended.split;

import com.alee.api.annotations.NotNull;
import com.alee.api.annotations.Nullable;
import com.alee.api.clone.Clone;
import com.alee.extended.layout.AbstractLayoutManager;
import com.alee.utils.CollectionUtils;
import com.alee.utils.CoreSwingUtils;
import com.alee.utils.swing.SizeType;

import javax.swing.event.EventListenerList;
import java.awt.*;
import java.awt.event.ComponentAdapter;
import java.awt.event.ComponentEvent;
import java.awt.event.ComponentListener;
import java.awt.event.MouseEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.ArrayList;
import java.util.List;

/**
 * Default {@link MultiSplitPaneModel} implementation.
 *
 * @author Mikle Garin
 * @see How to use WebMultiSplitPane
 * @see WebMultiSplitPane
 * @see MultiSplitPaneModel
 */
public class WebMultiSplitPaneModel extends AbstractLayoutManager implements MultiSplitPaneModel, PropertyChangeListener
{
    /**
     * {@link MultiSplitView}s created based on components added to {@link WebMultiSplitPane}.
     */
    protected final List views;

    /**
     * {@link WebMultiSplitPaneDivider}s created to separate components added to {@link WebMultiSplitPane}.
     */
    protected final List dividers;

    /**
     * {@link WebMultiSplitPane} listeners.
     */
    protected EventListenerList listeners;

    /**
     * {@link WebMultiSplitPane} this model is attached to.
     */
    protected WebMultiSplitPane multiSplitPane;

    /**
     * Whether or not {@link MultiSplitViewState}s have been initialized.
     */
    protected transient boolean initialized;

    /**
     * Type of the last {@link Operation} that have been performed within the model.
     */
    protected transient Operation lastOperation;

    /**
     * Index of currently dragged {@link WebMultiSplitPaneDivider}.
     */
    protected transient int draggedDividerIndex;

    /**
     * {@link WebMultiSplitPaneDivider} drag start mouse location.
     */
    protected transient Point dragStart;

    /**
     * Copy of {@link MultiSplitView}s used upon resize and drag operations.
     */
    protected transient List viewsCopy;

    /**
     * Constructs new {@link WebMultiSplitPaneModel}.
     */
    public WebMultiSplitPaneModel ()
    {
        this.views = new ArrayList ();
        this.dividers = new ArrayList ( 2 );
        this.listeners = new EventListenerList ();
        this.listeners.add ( ComponentListener.class, new ComponentAdapter ()
        {
            @Override
            public void componentResized ( final ComponentEvent e )
            {
                // Informing about operation
                onOperation ( Operation.splitPaneResized );
            }
        } );
        draggedDividerIndex = -1;
        dragStart = null;
    }

    @Override
    public void install ( @NotNull final WebMultiSplitPane multiSplitPane, @Nullable final List views,
                          @Nullable final List dividers )
    {
        if ( this.multiSplitPane == null )
        {
            this.initialized = false;
            this.multiSplitPane = multiSplitPane;
            this.multiSplitPane.addPropertyChangeListener ( this );
            for ( final ComponentListener listener : listeners.getListeners ( ComponentListener.class ) )
            {
                this.multiSplitPane.addComponentListener ( listener );
            }
            if ( CollectionUtils.notEmpty ( views ) )
            {
                this.views.addAll ( views );
            }
            if ( CollectionUtils.notEmpty ( dividers ) )
            {
                this.dividers.addAll ( dividers );
            }
            onOperation ( Operation.splitPaneModelInstalled );
        }
        else if ( this.multiSplitPane == multiSplitPane )
        {
            throw new IllegalStateException ( "This MultiSplitPaneModel is already installed in specified WebMultiSplitPane" );
        }
        else
        {
            throw new IllegalStateException ( "MultiSplitPaneModel can only be installed into single WebMultiSplitPane at a time" );
        }
    }

    @Override
    public void uninstall ( @NotNull final WebMultiSplitPane multiSplitPane )
    {
        if ( this.multiSplitPane != null && this.multiSplitPane == multiSplitPane )
        {
            // Informing about operation
            onOperation ( Operation.splitPaneModelUninstalled );

            // Clearing data
            this.views.clear ();
            this.dividers.clear ();
            for ( final ComponentListener listener : listeners.getListeners ( ComponentListener.class ) )
            {
                this.multiSplitPane.removeComponentListener ( listener );
            }
            this.multiSplitPane.removePropertyChangeListener ( this );
            this.multiSplitPane = null;
            this.initialized = false;
        }
        else if ( this.multiSplitPane == null )
        {
            throw new IllegalStateException ( "This MultiSplitPaneModel is not yet installed in any WebMultiSplitPane" );
        }
        else
        {
            throw new IllegalStateException ( "This MultiSplitPaneModel is installed in different WebMultiSplitPane" );
        }
    }

    @Override
    public void propertyChange ( final PropertyChangeEvent event )
    {
        if ( event.getPropertyName ().equals ( WebMultiSplitPane.ORIENTATION_PROPERTY ) )
        {
            onOperation ( Operation.splitOrientationChanged );
        }
    }

    @Override
    public void addComponent ( @NotNull final Component component, @Nullable final Object constraints )
    {
        if ( !( component instanceof WebMultiSplitPaneDivider ) )
        {
            // Parsing constraints
            final MultiSplitConstraints multiSplitConstraints;
            if ( constraints == null )
            {
                multiSplitConstraints = new MultiSplitConstraints ();
            }
            else if ( constraints instanceof Number )
            {
                multiSplitConstraints = new MultiSplitConstraints ( ( Number ) constraints );
            }
            else if ( constraints instanceof String )
            {
                multiSplitConstraints = new MultiSplitConstraints ( ( String ) constraints );
            }
            else if ( constraints instanceof MultiSplitConstraints )
            {
                multiSplitConstraints = ( MultiSplitConstraints ) constraints;
            }
            else
            {
                final String msg = "Unknown WebMultiSplitPane layout constraints type: %s";
                throw new IllegalArgumentException ( String.format ( msg, constraints ) );
            }

            // Calculating actual view index
            final int zOrder = multiSplitPane.getComponentZOrder ( component );
            final int viewIndex = zOrder <= views.size () ? zOrder : views.size ();

            // Adding view
            final MultiSplitView view = new MultiSplitView ( component, multiSplitConstraints );
            views.add ( viewIndex, view );

            // Initializing additional view size
            if ( initialized )
            {
                if ( multiSplitConstraints.isPixels () )
                {
                    // Saving pixel size as it is
                    view.state ().setSize ( multiSplitConstraints.pixels () );
                }
                else if ( multiSplitConstraints.isPercents () || multiSplitConstraints.isFill () || multiSplitConstraints.isPreferred () )
                {
                    // Saving percents, fill and preferred as preferred size
                    // We do not have any good way to determine how percents or fill should work at this point
                    final Dimension ps = component.getPreferredSize ();
                    final int preferred = multiSplitPane.getOrientation ().isHorizontal () ? ps.width : ps.height;
                    view.state ().setSize ( preferred );
                }
                else if ( multiSplitConstraints.isMinimum () )
                {
                    // Saving minimum size
                    final Dimension ms = component.getMinimumSize ();
                    final int minimum = multiSplitPane.getOrientation ().isHorizontal () ? ms.width : ms.height;
                    view.state ().setSize ( minimum );
                }
                else
                {
                    throw new IllegalArgumentException ( "Unknown view size value: " + multiSplitConstraints.size () );
                }
            }

            // Adding extra divider if needed
            if ( views.size () - 1 > dividers.size () )
            {
                // Adjusting indices for existing dividers first
                if ( dividers.size () > 0 )
                {
                    int dividerIndex = views.size ();
                    for ( final WebMultiSplitPaneDivider divider : dividers )
                    {
                        multiSplitPane.setComponentZOrderImpl ( divider, dividerIndex );
                        dividerIndex++;
                    }
                }

                // Adding new divider
                final int dividerIndex = viewIndex == 0 ? 0 : viewIndex - 1;
                final WebMultiSplitPaneDivider divider = multiSplitPane.createDivider ();
                multiSplitPane.addImpl ( divider, null, views.size () + dividerIndex );
                dividers.add ( dividerIndex, divider );
            }

            // Informing about operation
            onOperation ( Operation.splitViewsChanged );
        }
    }

    @Override
    public void moveComponent ( @NotNull final Component component, final int index )
    {
        if ( !( component instanceof WebMultiSplitPaneDivider ) )
        {
            // Checking old view index
            final int oldIndex = getViewIndex ( component );
            if ( oldIndex == -1 )
            {
                throw new IllegalArgumentException ( "Cannot find view for component: " + component );
            }

            // Checking new index
            if ( oldIndex < 0 || views.size () <= oldIndex )
            {
                throw new IndexOutOfBoundsException ( "Incorrect new index: " + oldIndex );
            }

            // Checking that index actually changed
            if ( oldIndex != index )
            {
                // Moving view
                final MultiSplitView view = views.remove ( oldIndex );
                views.add ( index, view );

                // Informing about operation
                onOperation ( Operation.splitViewsChanged );

                // Informing about occurred adjustments
                multiSplitPane.fireViewSizeAdjusted ();
            }
        }
    }

    @Override
    public void removeComponent ( @NotNull final Component component )
    {
        if ( !( component instanceof WebMultiSplitPaneDivider ) )
        {
            // Checking view index
            final int index = getViewIndex ( component );
            if ( index == -1 )
            {
                throw new IllegalArgumentException ( "Cannot find view for component: " + component );
            }

            // Removing related divider
            if ( views.size () > 1 )
            {
                final WebMultiSplitPaneDivider divider = dividers.get ( index > 0 ? index - 1 : index );
                final int zIndex = multiSplitPane.getComponentZOrder ( divider );
                multiSplitPane.removeImpl ( zIndex );
                dividers.remove ( divider );
            }

            // Removing view
            views.remove ( index );

            // Informing about operation
            onOperation ( Operation.splitViewsChanged );
        }
    }

    @Override
    public int getViewCount ()
    {
        return views.size ();
    }

    @Nullable
    @Override
    public List getViews ()
    {
        return Clone.deep ().clone ( views );
    }

    @Override
    public int getViewIndex ( @NotNull final Component component )
    {
        int index = -1;
        for ( int i = 0; i < views.size (); i++ )
        {
            final MultiSplitView view = views.get ( i );
            if ( view.component () == component )
            {
                index = i;
                break;
            }
        }
        return index;
    }

    @NotNull
    @Override
    public Component getViewComponent ( final int index )
    {
        return views.get ( index ).component ();
    }

    @NotNull
    @Override
    public List getViewComponents ()
    {
        final List viewComponents = new ArrayList ( views.size () );
        for ( final MultiSplitView view : views )
        {
            viewComponents.add ( view.component () );
        }
        return viewComponents;
    }

    @NotNull
    @Override
    public List getDividers ()
    {
        return new ArrayList ( dividers );
    }

    @Override
    public int getDividerIndex ( @NotNull final WebMultiSplitPaneDivider divider )
    {
        return dividers.indexOf ( divider );
    }

    @Nullable
    @Override
    public MultiSplitState getMultiSplitState ()
    {
        final MultiSplitState state;
        if ( initialized )
        {
            state = new MultiSplitState ();
            final List states = new ArrayList ( views.size () );
            for ( final MultiSplitView view : views )
            {
                states.add ( view.state () );
            }
            state.setStates ( states );
        }
        else
        {
            state = null;
        }
        return state;
    }

    @Override
    public void setMultiSplitState ( @NotNull final MultiSplitState state )
    {
        final List states = state.states ();
        if ( states.size () == views.size () )
        {
            final int previouslyExpandedViewIndex = getExpandedViewIndex ();

            // Updating states
            for ( int i = 0; i < views.size (); i++ )
            {
                views.get ( i ).setState ( states.get ( i ) );
            }

            // Informing about expansion
            // We have to do this manually to ensure appropriate events are thrown
            final int expandedViewIndex = getExpandedViewIndex ();
            if ( expandedViewIndex != previouslyExpandedViewIndex )
            {
                // Firing collapse event
                if ( previouslyExpandedViewIndex != -1 )
                {
                    multiSplitPane.fireViewCollapsed ( views.get ( previouslyExpandedViewIndex ).component () );
                }

                // Firing expansion event
                if ( expandedViewIndex != -1 )
                {
                    multiSplitPane.fireViewExpanded ( views.get ( expandedViewIndex ).component () );
                }
            }

            // Marking initialized
            initialized = true;

            // Informing about operation
            onOperation ( Operation.splitViewsStateApplied );
        }
    }

    @Override
    public void layoutContainer ( @NotNull final Container container )
    {
        if ( multiSplitPane.getWidth () > 0 && multiSplitPane.getHeight () > 0 && views.size () > 0 )
        {
            initializeViewsStates ();
            adjustViewsStates ();
            layoutMultiSplitPane ();
        }
    }

    /**
     * Initializes actual {@link MultiSplitView}s sizes if they are empty.
     */
    protected void initializeViewsStates ()
    {
        if ( !initialized )
        {
            // Updating view sizes from constraints
            setupViewStatesFromConstraints ();

            // Marking initialized
            initialized = true;

            // Informing about operation
            onOperation ( Operation.splitViewsSizesInitialized );
        }
    }

    @Override
    public void resetViewStates ()
    {
        // Updating view sizes from constraints
        setupViewStatesFromConstraints ();

        // Updating layout
        multiSplitPane.revalidate ();
        multiSplitPane.repaint ();
    }

    /**
     * Replaces all {@link MultiSplitView}s sizes with ones specified by {@link MultiSplitConstraints}.
     */
    protected void setupViewStatesFromConstraints ()
    {
        // Initializing variables
        final MultiSplitLayoutHelper.Static helper =
                new MultiSplitLayoutHelper.Static ( multiSplitPane, views, SizeType.current );

        // Updating view sizes
        for ( final MultiSplitView view : views )
        {
            final Component component = view.component ();
            final MultiSplitConstraints constraints = view.constraints ();
            final MultiSplitViewState viewState = view.state ();

            // Calculating and updating view size
            if ( constraints.isPixels () )
            {
                // Saving pixel size as it is
                viewState.setSize ( constraints.pixels () );
            }
            else if ( constraints.isPercents () )
            {
                // Normalizing percentage view size
                final double normalizedPercents = constraints.percents () / helper.totalPercentViews;
                final int size = ( int ) Math.round ( helper.spaceForPercentViews * normalizedPercents );
                viewState.setSize ( size );
            }
            else if ( constraints.isFill () )
            {
                // Normalizing fill view size
                final double normalizedPercents = helper.fillViewSize / helper.totalPercentViews;
                final int size = ( int ) Math.round ( helper.spaceForPercentViews * normalizedPercents );
                viewState.setSize ( size );
            }
            else if ( constraints.isPreferred () )
            {
                // Saving preferred view size
                final Dimension ps = helper.size ( component, SizeType.preferred );
                final int preferred = helper.horizontal ? ps.width : ps.height;
                viewState.setSize ( preferred );
            }
            else if ( constraints.isMinimum () )
            {
                // Saving minimum view size
                final Dimension ms = helper.size ( component, SizeType.minimum );
                final int minimum = helper.horizontal ? ms.width : ms.height;
                viewState.setSize ( minimum );
            }
            else
            {
                throw new IllegalArgumentException ( "Unknown view size value: " + constraints.size () );
            }
        }
    }

    /**
     * Adjusts actual {@link MultiSplitView}s sizes if they are inconsistent with current container size.
     * This is the most important method in the model as it keeps all {@link MultiSplitView}s sizes consistent with container size.
     */
    protected void adjustViewsStates ()
    {
        // Adjustments flag
        boolean adjusted = false;

        // Choosing initial views
        List baseViews = lastOperation == Operation.splitPaneResized ? viewsCopy : views;

        // Initializing helper instance
        MultiSplitLayoutHelper.Runtime helper = new MultiSplitLayoutHelper.Runtime ( multiSplitPane, baseViews );

        // Adjusting sizes according to area space
        while ( helper.space != helper.totalSizes )
        {
            // Delta between previously available space and currently available space
            int remainingSpaceDelta = helper.space - helper.totalSizes;

            // Act differently for major and minor deltas
            if ( Math.abs ( remainingSpaceDelta ) > 1 )
            {
                // Act differently when any weights are available and not
                if ( remainingSpaceDelta > 0 ? helper.totalWeights > 0.0 : helper.nonZeroViewsWeights > 0.0 )
                {
                    // Using weights to distribute space delta
                    // For increasing - we are using all weights as we want to distribute extra space to all eligible views
                    // For decreasing - we are only using weights of views with non-zero size
                    double remainingDeltaWeight = remainingSpaceDelta > 0 ? helper.totalWeights : helper.nonZeroViewsWeights;

                    // Adjusting views sizes
                    for ( int i = 0; i < baseViews.size (); i++ )
                    {
                        final MultiSplitView view = baseViews.get ( i );
                        final MultiSplitViewState viewState = view.state ();
                        final int size = viewState.size ();
                        final double viewWeight = view.constraints ().weight ();

                        // Only adjusting view size if it has weight
                        // Only adjusting view size if delta is positive or view size is not zero
                        if ( viewWeight > 0.0 && ( remainingSpaceDelta > 0 || size > 0 ) )
                        {
                            // We have specific weight so we can distribute space to views
                            final int rawViewDelta = ( int ) ( remainingSpaceDelta * viewWeight / remainingDeltaWeight );
                            final int actualDelta;
                            if ( rawViewDelta >= 0 )
                            {
                                // Increasing view size
                                // We can always safely proceed with this
                                actualDelta = rawViewDelta;
                            }
                            else if ( size >= Math.abs ( rawViewDelta ) )
                            {
                                // Decreasing view size
                                // We can safely proceed if remaining size is larger than delta
                                actualDelta = rawViewDelta;
                            }
                            else
                            {
                                // Decreasing view size to zero
                                // We have to be aware of delta leftover in this case
                                actualDelta = -size;
                            }

                            // Adjusting actual view size
                            views.get ( i ).state ().setSize ( size + actualDelta );

                            // Updating deltas
                            remainingSpaceDelta -= actualDelta;
                            remainingDeltaWeight -= viewWeight;
                        }
                        else
                        {
                            views.get ( i ).state ().setSize ( size );
                        }
                    }
                }
                else
                {
                    // Using weights synthetic weight to distribute space delta
                    // This will basically attempt to split space delta equally between views
                    final double averageWeight = 1.0;
                    double remainingDeltaWeight = 1.0 * ( remainingSpaceDelta > 0 ? baseViews.size () : helper.nonZeroViews );

                    // Adjusting view sizes
                    for ( int i = 0; i < baseViews.size (); i++ )
                    {
                        final MultiSplitView view = baseViews.get ( i );
                        final MultiSplitViewState viewState = view.state ();
                        final int size = viewState.size ();
                        final double viewWeight = view.constraints ().weight ();

                        // Only adjusting view size if it doesn't have weight
                        // Only adjusting view size if delta is positive or view size is not zero
                        if ( viewWeight == 0.0 && ( remainingSpaceDelta > 0 || size > 0 ) )
                        {
                            // Trying to equally distribute delta space
                            final int rawViewDelta = ( int ) ( remainingSpaceDelta * averageWeight / remainingDeltaWeight );
                            final int actualDelta;
                            if ( rawViewDelta >= 0 )
                            {
                                // Increasing view size
                                // We can always safely proceed with this
                                actualDelta = rawViewDelta;
                            }
                            else if ( size >= Math.abs ( rawViewDelta ) )
                            {
                                // Decreasing view size
                                // We can safely proceed if remaining size is larger than delta
                                actualDelta = rawViewDelta;
                            }
                            else
                            {
                                // Decreasing view size to zero
                                // We have to be aware of delta leftover in this case
                                actualDelta = -size;
                            }

                            // Adjusting actual view size
                            views.get ( i ).state ().setSize ( size + actualDelta );

                            // Updating deltas
                            remainingSpaceDelta -= actualDelta;
                            remainingDeltaWeight -= averageWeight;
                        }
                        else
                        {
                            views.get ( i ).state ().setSize ( size );
                        }
                    }
                }
            }
            else
            {
                // Adjusting first non-zero view size
                // For increasing - we don't want to affect zero size views either as they would randomly get to 1 pixel size
                // For decreasing - we don't want to affect views which already have zero size
                // The only case when we affect zero size views is when all views have zero size
                for ( int i = 0; i < baseViews.size (); i++ )
                {
                    final MultiSplitViewState viewState = baseViews.get ( i ).state ();
                    final int size = viewState.size ();
                    if ( size > 0 || helper.nonZeroViews == 0 )
                    {
                        views.get ( i ).state ().setSize ( size + remainingSpaceDelta );
                        break;
                    }
                }
            }

            // Further iterations should be done on the actual views
            // Otherwise we will lose results of the current iteration
            baseViews = views;

            // Updating helper instance to retry calculations to ensure we have distributed all available space
            helper = new MultiSplitLayoutHelper.Runtime ( multiSplitPane, baseViews );

            // Marking adjusted
            adjusted = true;
        }

        // Informing about occurred adjustments
        if ( adjusted )
        {
            multiSplitPane.fireViewSizeAdjusted ();
        }
    }

    /**
     * Performs actual layouting of {@link MultiSplitView}s.
     */
    protected void layoutMultiSplitPane ()
    {
        // Basic settings
        final Insets insets = multiSplitPane.getInsets ();
        final boolean horizontal = multiSplitPane.getOrientation ().isHorizontal ();
        final int expandedIndex = getExpandedViewIndex ();

        // Layout heavily depends on whether or not any of views is expanded
        if ( expandedIndex == -1 )
        {
            // Divider sizes
            final int dividerSize = multiSplitPane.getDividerSize ();
            final int dividerWidth = horizontal ? dividerSize : multiSplitPane.getWidth () - insets.left - insets.right;
            final int dividerHeight = horizontal ? multiSplitPane.getHeight () - insets.top - insets.bottom : dividerSize;

            // Positioning all components
            int x = insets.left;
            int y = insets.top;
            final int width = horizontal ? 0 : multiSplitPane.getWidth () - insets.left - insets.right;
            final int height = horizontal ? multiSplitPane.getHeight () - insets.top - insets.bottom : 0;
            for ( final MultiSplitView view : views )
            {
                final int index = views.indexOf ( view );
                final boolean lastComponent = index == views.size () - 1;
                final Component component = view.component ();
                final MultiSplitViewState viewState = view.state ();
                final int size = viewState.size ();

                // Positioning view component
                if ( horizontal )
                {
                    component.setBounds ( x, y, size, height );
                }
                else
                {
                    component.setBounds ( x, y, width, size );
                }

                // Positioning divider and adjusting coordinates for next cycle run
                if ( !lastComponent )
                {
                    // Positioning divider
                    final WebMultiSplitPaneDivider divider = dividers.get ( index );
                    if ( horizontal )
                    {
                        divider.setBounds ( x + size, y, dividerWidth, dividerHeight );
                    }
                    else
                    {
                        divider.setBounds ( x, y + size, dividerWidth, dividerHeight );
                    }

                    // Adjusting coordinates for next cycle run
                    if ( horizontal )
                    {
                        x += size + dividerSize;
                    }
                    else
                    {
                        y += size + dividerSize;
                    }
                }
            }
        }
        else
        {
            // Divider components
            final WebMultiSplitPaneDivider dividerBefore = expandedIndex > 0 ? dividers.get ( expandedIndex - 1 ) : null;
            final MultiSplitView expandedView = views.get ( expandedIndex );
            final WebMultiSplitPaneDivider dividerAfter = expandedIndex < views.size () - 1 ? dividers.get ( expandedIndex ) : null;

            // Divider sizes
            final int dividerBeforeSize;
            final int dividerAfterSize;

            // Placing divider in front of the view
            if ( dividerBefore != null )
            {
                final Dimension ps = dividerBefore.getPreferredSize ();
                dividerBeforeSize = horizontal ? ps.width : ps.height;
                final int width = horizontal ? dividerBeforeSize : multiSplitPane.getWidth () - insets.left - insets.right;
                final int height = horizontal ? multiSplitPane.getHeight () - insets.top - insets.bottom : dividerBeforeSize;
                dividerBefore.setBounds ( insets.left, insets.top, width, height );
            }
            else
            {
                dividerBeforeSize = 0;
            }

            // Placing divider after the view
            if ( dividerAfter != null )
            {
                final Dimension ps = dividerAfter.getPreferredSize ();
                dividerAfterSize = horizontal ? ps.width : ps.height;
                final int width = horizontal ? dividerAfterSize : multiSplitPane.getWidth () - insets.left - insets.right;
                final int height = horizontal ? multiSplitPane.getHeight () - insets.top - insets.bottom : dividerAfterSize;
                final int x = horizontal ? multiSplitPane.getWidth () - insets.right - width : insets.left;
                final int y = horizontal ? insets.top : multiSplitPane.getHeight () - insets.bottom - height;
                dividerAfter.setBounds ( x, y, width, height );
            }
            else
            {
                dividerAfterSize = 0;
            }

            // Placing view component
            final int horBefore = horizontal && dividerBefore != null ? dividerBeforeSize : 0;
            final int horAfter = horizontal && dividerAfter != null ? dividerAfterSize : 0;
            final int verBefore = !horizontal && dividerBefore != null ? dividerBeforeSize : 0;
            final int verAfter = !horizontal && dividerAfter != null ? dividerAfterSize : 0;
            final int viewX = insets.left + horBefore;
            final int viewY = insets.top + verBefore;
            final int viewWidth = multiSplitPane.getWidth () - insets.left - insets.right - horBefore - horAfter;
            final int viewHeight = multiSplitPane.getHeight () - insets.top - insets.bottom - verBefore - verAfter;
            expandedView.component ().setBounds ( viewX, viewY, viewWidth, viewHeight );
        }
    }

    @NotNull
    @Override
    public Dimension preferredLayoutSize ( @NotNull final Container container )
    {
        return layoutSize ( SizeType.preferred );
    }

    @NotNull
    @Override
    public Dimension minimumLayoutSize ( @NotNull final Container container )
    {
        return layoutSize ( SizeType.minimum );
    }

    @NotNull
    @Override
    public Dimension maximumLayoutSize ( @NotNull final Container container )
    {
        return layoutSize ( SizeType.maximum );
    }

    /**
     * Returns layout size for the specified {@link SizeType}.
     *
     * @param sizeType {@link SizeType}
     * @return layout size for the specified {@link SizeType}
     */
    protected Dimension layoutSize ( @NotNull final SizeType sizeType )
    {
        // Calculating content size
        final Dimension contentSize;
        if ( views.size () > 0 )
        {
            // Sizes helper
            final MultiSplitLayoutHelper.Static helper = new MultiSplitLayoutHelper.Static ( multiSplitPane, views, sizeType );

            // Calculating width and height
            int width = 0;
            int height = 0;
            int percentageSize = 0;
            for ( final MultiSplitView view : views )
            {
                final int index = views.indexOf ( view );
                final int extra = index != views.size () - 1 ? multiSplitPane.getDividerSize () : 0;
                final Component component = view.component ();

                // Choosing component size
                final Dimension cs = helper.size ( component, sizeType );

                // Calculating component pixel length
                // Percentage and fill views will be saved for later
                final int length;
                final MultiSplitConstraints constraints = view.constraints ();
                if ( constraints.isPixels () )
                {
                    // Saving pixel size as it is
                    length = constraints.pixels ();
                }
                else if ( constraints.isPercents () )
                {
                    // Normalizing percentage views
                    final int componentLength = helper.horizontal ? cs.width : cs.height;
                    final double normalized = constraints.percents () / helper.totalPercentViews;
                    percentageSize = Math.max ( percentageSize, ( int ) ( componentLength / normalized ) );
                    length = 0;
                }
                else if ( constraints.isFill () )
                {
                    // Normalizing fill views
                    final int componentLength = helper.horizontal ? cs.width : cs.height;
                    final double normalized = helper.fillViewSize / helper.totalPercentViews;
                    percentageSize = Math.max ( percentageSize, ( int ) ( componentLength / normalized ) );
                    length = 0;
                }
                else if ( constraints.isPreferred () )
                {
                    // Saving preferred size
                    final Dimension ps = helper.size ( component, SizeType.preferred );
                    length = helper.horizontal ? ps.width : ps.height;
                }
                else if ( constraints.isMinimum () )
                {
                    // Saving minimum size
                    final Dimension ms = helper.size ( component, SizeType.minimum );
                    length = helper.horizontal ? ms.width : ms.height;
                }
                else
                {
                    throw new IllegalArgumentException ( "Unknown view size value: " + constraints.size () );
                }

                // Adjusting resulting width and height
                if ( helper.horizontal )
                {
                    width += length + extra;
                    height = Math.max ( height, cs.height );
                }
                else
                {
                    width = Math.max ( width, cs.width );
                    height += length + extra;
                }
            }

            // Adding largest found percentage and fill views sizes
            if ( helper.horizontal )
            {
                width += percentageSize;
            }
            else
            {
                height += percentageSize;
            }

            // Combining resulting size
            contentSize = new Dimension ( width, height );
        }
        else
        {
            contentSize = new Dimension ( 0, 0 );
        }

        // Adjusting resulting size according to insets
        final Insets insets = multiSplitPane.getInsets ();
        return new Dimension ( insets.left + contentSize.width + insets.right, insets.top + contentSize.height + insets.bottom );
    }

    @Override
    public boolean isAnyViewExpanded ()
    {
        return getExpandedViewIndex () != -1;
    }

    @Override
    public int getExpandedViewIndex ()
    {
        int index = -1;
        for ( int i = 0; i < views.size (); i++ )
        {
            if ( views.get ( i ).state ().isExpanded () )
            {
                index = i;
                break;
            }
        }
        return index;
    }

    @Override
    public void expandView ( final int index )
    {
        // Checking view availability
        if ( 0 <= index && index < views.size () )
        {
            // Checking expanded view
            final int expandedIndex = getExpandedViewIndex ();
            if ( expandedIndex != index )
            {
                // Restoring other expanded view
                if ( expandedIndex != -1 )
                {
                    views.get ( expandedIndex ).state ().setExpanded ( false );
                }

                // Expanding specified view
                final MultiSplitView newlyExpandedView = views.get ( index );
                newlyExpandedView.state ().setExpanded ( true );

                // Updating visibility
                for ( int i = 0; i < views.size (); i++ )
                {
                    views.get ( i ).component ().setVisible ( i == index );
                    if ( i < dividers.size () )
                    {
                        dividers.get ( i ).setVisible ( i == index - 1 || i == index );
                    }
                }

                // Updating layout
                multiSplitPane.revalidate ();
                multiSplitPane.repaint ();

                // Firing expansion event
                multiSplitPane.fireViewExpanded ( newlyExpandedView.component () );
            }
        }
        else
        {
            throw new IndexOutOfBoundsException ( "WebMultiSplitPane doesn't have a view under index: " + index );
        }
    }

    @Override
    public void collapseExpandedView ()
    {
        // Checking expanded view
        final int expandedIndex = getExpandedViewIndex ();
        if ( expandedIndex != -1 )
        {
            // Restoring expanded view
            final MultiSplitView previouslyExpandedView = views.get ( expandedIndex );
            previouslyExpandedView.state ().setExpanded ( false );

            // Updating visibility
            for ( int i = 0; i < views.size (); i++ )
            {
                views.get ( i ).component ().setVisible ( true );
                if ( i < dividers.size () )
                {
                    dividers.get ( i ).setVisible ( true );
                }
            }

            // Updating layout
            multiSplitPane.revalidate ();
            multiSplitPane.repaint ();

            // Firing collapse event
            multiSplitPane.fireViewCollapsed ( previouslyExpandedView.component () );
        }
    }

    @Override
    public void toggleViewExpansion ( final int index )
    {
        final int expandedIndex = getExpandedViewIndex ();
        if ( expandedIndex == -1 )
        {
            expandView ( index );
        }
        else
        {
            collapseExpandedView ();
        }
    }

    @Override
    public void toggleViewToLeft ( @NotNull final WebMultiSplitPaneDivider divider )
    {
        toggleViewExpansion ( dividers.indexOf ( divider ) );
    }

    @Override
    public void toggleViewToRight ( @NotNull final WebMultiSplitPaneDivider divider )
    {
        toggleViewExpansion ( dividers.indexOf ( divider ) + 1 );
    }

    @Override
    public boolean isDragAvailable ( @NotNull final WebMultiSplitPaneDivider divider )
    {
        return multiSplitPane.isEnabled () && getExpandedViewIndex () == -1 && dividers.contains ( divider );
    }

    @Nullable
    @Override
    public WebMultiSplitPaneDivider getDraggedDivider ()
    {
        return draggedDividerIndex != -1 ? dividers.get ( draggedDividerIndex ) : null;
    }

    @Override
    public void dividerDragStarted ( @NotNull final WebMultiSplitPaneDivider divider, @NotNull final MouseEvent e )
    {
        // Updating last operation
        onOperation ( Operation.splitDividerDragged );

        // Retrieving dragged divider index
        draggedDividerIndex = dividers.indexOf ( divider );
        if ( draggedDividerIndex == -1 )
        {
            throw new IllegalArgumentException ( "WebMultiSplitPane doesn't contain specified divider: " + divider );
        }

        // Initial mouse location
        dragStart = CoreSwingUtils.getMouseLocation ();

        // Informing about divider drag start
        multiSplitPane.fireViewResizeStarted ( divider );
    }

    @Override
    public void dividerDragged ( @NotNull final WebMultiSplitPaneDivider divider, @NotNull final MouseEvent e )
    {
        // Updating last operation
        onOperation ( Operation.splitDividerDragged );

        // Current mouse location
        final Point mouse = CoreSwingUtils.getMouseLocation ();

        // Drag delta
        final int delta = multiSplitPane.getOrientation ().isHorizontal () ? mouse.x - dragStart.x : mouse.y - dragStart.y;

        final boolean straightOrder = delta > 0;
        final int startFrom = delta > 0 ? draggedDividerIndex + 1 : draggedDividerIndex;
        final int stretchedIndex = delta > 0 ? draggedDividerIndex : draggedDividerIndex + 1;

        // Adjusting shrinked views sizes
        int remainingDelta = Math.abs ( delta );
        final int startIndex = straightOrder ? 0 : viewsCopy.size () - 1;
        final int increment = straightOrder ? 1 : -1;
        for ( int i = startIndex; straightOrder ? i < viewsCopy.size () : i >= 0; i += increment )
        {
            final int size = viewsCopy.get ( i ).state ().size ();
            if ( ( straightOrder ? i >= startFrom : i <= startFrom ) && remainingDelta > 0 )
            {
                if ( size < remainingDelta )
                {
                    views.get ( i ).state ().setSize ( 0 );
                    remainingDelta -= size;
                }
                else
                {
                    views.get ( i ).state ().setSize ( size - remainingDelta );
                    remainingDelta = 0;
                }
            }
            else
            {
                views.get ( i ).state ().setSize ( size );
            }
        }

        // Delta that we were able to apply
        final int allowedDelta = Math.abs ( delta ) - remainingDelta;
        if ( allowedDelta > 0 )
        {
            // Adjusting stretched view size
            final int stretchedSize = viewsCopy.get ( stretchedIndex ).state ().size ();
            views.get ( stretchedIndex ).state ().setSize ( stretchedSize + allowedDelta );
        }

        // Informing about divider drag
        multiSplitPane.fireViewResized ( divider );

        // Updating layout
        multiSplitPane.revalidate ();
        multiSplitPane.repaint ();
    }

    @Override
    public void dividerDragEnded ( @NotNull final WebMultiSplitPaneDivider divider, @NotNull final MouseEvent e )
    {
        // Updating last operation
        onOperation ( Operation.splitDividerDragEnded );

        // Cleaning up resources
        draggedDividerIndex = -1;
        dragStart = null;

        // Informing about divider drag end
        multiSplitPane.fireViewResizeEnded ( divider );
    }

    /**
     * Called upon different operation executions.
     *
     * @param operation {@link Operation} type
     */
    protected void onOperation ( @NotNull final Operation operation )
    {
        // Reacting to various operations
        if ( lastOperation != Operation.splitPaneResized && operation == Operation.splitPaneResized ||
                lastOperation != Operation.splitDividerDragged && operation == Operation.splitDividerDragged )
        {
            // Copying views information for resize or drag operation
            // This information will be used to provide more precise adjustments to views sizes
            // Unlike common JSpliPane previous initial state is taken into account and used to adjust sizes accordingly
            viewsCopy = getViews ();
        }
        else if ( operation == Operation.splitPaneModelInstalled || operation == Operation.splitPaneModelUninstalled ||
                operation == Operation.splitViewsSizesInitialized || operation == Operation.splitViewsStateApplied ||
                operation == Operation.splitViewsChanged || operation == Operation.splitDividerDragEnded ||
                operation == Operation.splitOrientationChanged )
        {
            // Cleaning up views information upon some operations
            // The only operation we cannot cleanup information after is component resize as it doesn't have specific end time
            // In such case information is cleaned up upon any next operation that forces views copy to be made or cleaned up
            viewsCopy = null;
        }

        // Saving last operation
        lastOperation = operation;
    }

    /**
     * Operations that can be performed with {@link MultiSplitView}s.
     */
    protected enum Operation
    {
        splitPaneModelInstalled,
        splitPaneModelUninstalled,
        splitViewsSizesInitialized,
        splitViewsStateApplied,
        splitOrientationChanged,
        splitViewsChanged,
        splitPaneResized,
        splitDividerDragged,
        splitDividerDragEnded,
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy