com.alee.extended.layout.AccordionLayout Maven / Gradle / Ivy
/*
* 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.layout;
import com.alee.extended.panel.AccordionStyle;
import com.alee.extended.panel.WebAccordion;
import com.alee.extended.panel.WebCollapsiblePane;
import javax.swing.*;
import java.awt.*;
import java.util.ArrayList;
import java.util.List;
/**
* Custom layout for WebAccordion component.
*
* @author Mikle Garin
*/
public class AccordionLayout extends AbstractLayoutManager
{
/**
* Accordion to which this layout is attached.
*/
private final WebAccordion accordion;
/**
* Constructs layout attached to the specified accordion.
*
* @param accordion accordion
*/
public AccordionLayout ( WebAccordion accordion )
{
super ();
this.accordion = accordion;
}
/**
* {@inheritDoc}
*/
@Override
public Dimension preferredLayoutSize ( Container parent )
{
return getLayoutSize ( parent, true );
}
/**
* {@inheritDoc}
*/
@Override
public Dimension minimumLayoutSize ( Container parent )
{
return getLayoutSize ( parent, false );
}
/**
* Returns either minimum or preferred container size.
*
* @param parent container
* @param preferred whether preferred size should be returned or not
* @return either minimum or preferred container size
*/
private Dimension getLayoutSize ( Container parent, boolean preferred )
{
final List panes = accordion.getActualPanesList ();
final int gap = accordion.getAccordionStyle () == AccordionStyle.separated ? accordion.getGap () : 0;
final Dimension ps = new Dimension ();
final boolean hor = accordion.getOrientation () == SwingConstants.HORIZONTAL;
for ( final WebCollapsiblePane pane : panes )
{
final Dimension cps = preferred || !accordion.isFillSpace () ? pane.getPreferredSize () : pane.getBasePreferredSize ();
if ( hor )
{
ps.width += cps.width;
ps.height = Math.max ( ps.height, cps.height );
}
else
{
ps.width = Math.max ( ps.width, cps.width );
ps.height += cps.height;
}
}
if ( panes.size () > 0 )
{
if ( hor )
{
ps.width += gap * ( panes.size () - 1 );
}
else
{
ps.height += gap * ( panes.size () - 1 );
}
}
final Insets insets = parent.getInsets ();
ps.width += insets.left + insets.right;
ps.height += insets.top + insets.bottom;
return ps;
}
/**
* {@inheritDoc}
*/
@Override
public void layoutContainer ( Container parent )
{
final List panes = accordion.getActualPanesList ();
final int gap = accordion.getAccordionStyle () == AccordionStyle.separated ? accordion.getGap () : 0;
final Insets insets = parent.getInsets ();
final Dimension size = parent.getSize ();
final int w = size.width - insets.left - insets.right;
final int h = size.height - insets.top - insets.bottom;
final boolean hor = accordion.getOrientation () == SwingConstants.HORIZONTAL;
int x = insets.left;
int y = insets.top;
if ( accordion.isFillSpace () )
{
// Computing the part available to fill in with panes content
float totalStates = 0;
int totalFillLength = ( hor ? size.width - insets.left - insets.right : size.height - insets.top - insets.bottom ) + gap;
int visuallyExpanded = 0;
int lastFillIndex = -1;
final List base = new ArrayList ();
for ( final WebCollapsiblePane pane : panes )
{
Dimension bps = pane.getBasePreferredSize ();
base.add ( hor ? bps.width : bps.height );
float expandState = pane.getTransitionProgress ();
totalStates += expandState;
totalFillLength -= ( hor ? bps.width : bps.height ) + gap;
if ( expandState > 0f )
{
lastFillIndex = panes.indexOf ( pane );
visuallyExpanded++;
}
}
totalStates = visuallyExpanded == 1 ? 1f : totalStates;
totalFillLength = Math.max ( totalFillLength, 0 );
// Layouting panes
float end = 0f;
for ( int i = 0; i < panes.size (); i++ )
{
float expandState = panes.get ( i ).getTransitionProgress ();
int length = base.get ( i );
if ( expandState > 0f )
{
end += ( totalFillLength * expandState / totalStates ) % 1;
length += Math.round ( ( float ) Math.floor ( totalFillLength * expandState / totalStates ) ) +
( i == lastFillIndex ? Math.round ( end ) : 0 );
}
panes.get ( i ).setBounds ( x, y, hor ? length : w, hor ? h : length );
if ( hor )
{
x += length + gap;
}
else
{
y += length + gap;
}
}
}
else
{
// Simply layouting panes by preferred size
for ( final WebCollapsiblePane pane : panes )
{
final Dimension cps = pane.getPreferredSize ();
pane.setBounds ( x, y, hor ? cps.width : w, hor ? h : cps.height );
if ( hor )
{
x += cps.width + gap;
}
else
{
y += cps.height + gap;
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy