com.alee.painter.decoration.layout.IconTextLayout 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.painter.decoration.layout;
import com.alee.api.annotations.NotNull;
import com.alee.api.annotations.Nullable;
import com.alee.api.data.BoxOrientation;
import com.alee.painter.decoration.IDecoration;
import com.alee.utils.MathUtils;
import com.alee.utils.SwingUtils;
import com.thoughtworks.xstream.annotations.XStreamAlias;
import com.thoughtworks.xstream.annotations.XStreamAsAttribute;
import javax.swing.*;
import java.awt.*;
/**
* Abstract implementation of simple icon and text layout.
* Main constraints are {@link #ICON} and {@link #TEXT} which are always placed in the middle of layout.
* Side spacing constraints are also supported which place contents similar to {@link BorderLayout}.
*
* @param component type
* @param decoration type
* @param layout type
* @author Mikle Garin
*/
@XStreamAlias ( "IconTextLayout" )
public class IconTextLayout, I extends IconTextLayout>
extends AbstractContentLayout implements SwingConstants
{
/**
* Layout constraints.
*/
protected static final String ICON = "icon";
protected static final String TEXT = "text";
protected static final String TOP_SPACE = "top-space";
protected static final String LEFT_SPACE = "left-space";
protected static final String RIGHT_SPACE = "right-space";
protected static final String BOTTOM_SPACE = "bottom-space";
/**
* Gap between icon and text contents.
*/
@Nullable
@XStreamAsAttribute
protected Integer gap;
/**
* Horizontal content alignment.
*/
@Nullable
@XStreamAsAttribute
protected BoxOrientation halign;
/**
* Vertical content alignment.
*/
@Nullable
@XStreamAsAttribute
protected BoxOrientation valign;
/**
* Horizontal text position.
*/
@Nullable
@XStreamAsAttribute
protected BoxOrientation hpos;
/**
* Vertical text position.
*/
@Nullable
@XStreamAsAttribute
protected BoxOrientation vpos;
/**
* Returns gap between icon and text contents.
*
* @param c painted component
* @param d painted decoration state
* @return gap between icon and text contents
*/
protected int getIconTextGap ( @NotNull final C c, @NotNull final D d )
{
return gap != null ? gap : 0;
}
/**
* Returns horizontal content alignment.
*
* @param c painted component
* @param d painted decoration state
* @return horizontal content alignment
*/
protected int getHorizontalAlignment ( @NotNull final C c, @NotNull final D d )
{
return halign != null ? halign.getValue () : CENTER;
}
/**
* Returns vertical content alignment.
*
* @param c painted component
* @param d painted decoration state
* @return vertical content alignment
*/
protected int getVerticalAlignment ( @NotNull final C c, @NotNull final D d )
{
return valign != null ? valign.getValue () : CENTER;
}
/**
* Returns horizontal text position.
*
* @param c painted component
* @param d painted decoration state
* @return horizontal text position
*/
protected int getHorizontalTextPosition ( @NotNull final C c, @NotNull final D d )
{
return hpos != null ? hpos.getValue () : TRAILING;
}
/**
* Returns vertical text position.
*
* @param c painted component
* @param d painted decoration state
* @return vertical text position
*/
protected int getVerticalTextPosition ( @NotNull final C c, @NotNull final D d )
{
return vpos != null ? vpos.getValue () : CENTER;
}
@NotNull
@Override
public ContentLayoutData layoutContent ( @NotNull final C c, @NotNull final D d, @NotNull final Rectangle bounds )
{
final ContentLayoutData layoutData = new ContentLayoutData ( 2 );
final Dimension available = bounds.getSize ();
final boolean ltr = isLeftToRight ( c, d );
final int halign = getHorizontalAlignment ( c, d );
final int valign = getVerticalAlignment ( c, d );
// Retrieving spacing preferred sizes
final Dimension top = getPreferredSize ( c, d, available, TOP_SPACE );
final Dimension left = getPreferredSize ( c, d, available, LEFT_SPACE );
final Dimension right = getPreferredSize ( c, d, available, RIGHT_SPACE );
final Dimension bottom = getPreferredSize ( c, d, available, BOTTOM_SPACE );
// Calculating actual spacing sizes
// We have to find maximum of two in case axis alignment for main content is CENTER
final int lw = halign == CENTER ? Math.max ( left.width, right.width ) : left.width;
final int rw = halign == CENTER ? Math.max ( left.width, right.width ) : right.width;
final int th = valign == CENTER ? Math.max ( top.height, bottom.height ) : top.height;
final int bh = valign == CENTER ? Math.max ( top.height, bottom.height ) : bottom.height;
// Calculating icon and text preferred size
final Dimension iconTextAvailable = new Dimension ( available.width - lw - rw, available.height - th - bh );
final Dimension iconTextPS = getIconTextPreferredSize ( c, d, iconTextAvailable );
// Adjusting available size
iconTextPS.width = Math.max ( 0, Math.min ( iconTextPS.width, bounds.width - lw - rw ) );
iconTextPS.height = Math.max ( 0, Math.min ( iconTextPS.height, bounds.height - th - bh ) );
// Calculating smallest content bounds required for text and icon
final Rectangle b = new Rectangle ( 0, 0, iconTextPS.width, iconTextPS.height );
if ( halign == LEFT || halign == LEADING && ltr || halign == TRAILING && !ltr )
{
b.x = bounds.x + lw;
}
else if ( halign == CENTER )
{
b.x = bounds.x + bounds.width / 2 - iconTextPS.width / 2;
}
else
{
b.x = bounds.x + bounds.width - iconTextPS.width - rw;
}
if ( valign == TOP )
{
b.y = bounds.y + th;
}
else if ( valign == CENTER )
{
b.y = bounds.y + bounds.height / 2 - iconTextPS.height / 2;
}
else
{
b.y = bounds.y + bounds.height - iconTextPS.height - bh;
}
// Calculating spacings positioning
if ( !isEmpty ( c, d, TOP_SPACE ) )
{
layoutData.put ( TOP_SPACE, new Rectangle ( bounds.x, bounds.y,
bounds.width, b.y - bounds.y ) );
}
if ( !isEmpty ( c, d, LEFT_SPACE ) )
{
layoutData.put ( LEFT_SPACE, new Rectangle ( bounds.x, b.y,
b.x - bounds.x, b.height ) );
}
if ( !isEmpty ( c, d, RIGHT_SPACE ) )
{
layoutData.put ( RIGHT_SPACE, new Rectangle ( b.x + b.width, b.y,
bounds.x + bounds.width - b.x - b.width, b.height ) );
}
if ( !isEmpty ( c, d, BOTTOM_SPACE ) )
{
layoutData.put ( BOTTOM_SPACE, new Rectangle ( bounds.x, b.y + b.height,
bounds.width, bounds.y + bounds.height - b.y - b.height ) );
}
// Calculating text and icon positioning
final boolean hasIcon = !isEmpty ( c, d, ICON );
final boolean hasText = !isEmpty ( c, d, TEXT );
if ( hasIcon && hasText )
{
final int hpos = getHorizontalTextPosition ( c, d );
final int vpos = getVerticalTextPosition ( c, d );
if ( hpos != CENTER || vpos != CENTER )
{
final Dimension ips = getPreferredSize ( c, d, available, ICON );
final int gap = getIconTextGap ( c, d );
if ( hpos == RIGHT || hpos == TRAILING && ltr )
{
layoutData.put ( ICON, new Rectangle ( b.x, b.y, ips.width, b.height ) );
layoutData.put ( TEXT, new Rectangle ( b.x + gap + ips.width, b.y, b.width - ips.width - gap, b.height ) );
}
else if ( hpos == CENTER )
{
if ( vpos == TOP )
{
layoutData.put ( ICON, new Rectangle ( b.x, b.y + b.height - ips.height, b.width, ips.height ) );
layoutData.put ( TEXT, new Rectangle ( b.x, b.y, b.width, b.height - gap - ips.height ) );
}
else
{
layoutData.put ( ICON, new Rectangle ( b.x, b.y, b.width, ips.height ) );
layoutData.put ( TEXT, new Rectangle ( b.x, b.y + ips.height + gap, b.width, b.height - ips.height - gap ) );
}
}
else
{
layoutData.put ( ICON, new Rectangle ( b.x + b.width - ips.width, b.y, ips.width, b.height ) );
layoutData.put ( TEXT, new Rectangle ( b.x, b.y, b.width - ips.width - gap, b.height ) );
}
}
else
{
layoutData.put ( ICON, b );
layoutData.put ( TEXT, b );
}
}
else if ( hasIcon )
{
layoutData.put ( ICON, b );
}
else if ( hasText )
{
layoutData.put ( TEXT, b );
}
return layoutData;
}
@NotNull
@Override
protected Dimension getContentPreferredSize ( @NotNull final C c, @NotNull final D d, @NotNull final Dimension available )
{
final int halign = getHorizontalAlignment ( c, d );
final int valign = getVerticalAlignment ( c, d );
// Retrieving spacing preferred sizes
final Dimension top = getPreferredSize ( c, d, available, TOP_SPACE );
final Dimension left = getPreferredSize ( c, d, available, LEFT_SPACE );
final Dimension right = getPreferredSize ( c, d, available, RIGHT_SPACE );
final Dimension bottom = getPreferredSize ( c, d, available, BOTTOM_SPACE );
// Calculating actual spacing sizes
// We have to find maximum of two in case axis alignment for main content is CENTER
final int lw = halign == CENTER ? Math.max ( left.width, right.width ) : left.width;
final int rw = halign == CENTER ? Math.max ( left.width, right.width ) : right.width;
final int th = valign == CENTER ? Math.max ( top.height, bottom.height ) : top.height;
final int bh = valign == CENTER ? Math.max ( top.height, bottom.height ) : bottom.height;
// Calculating icon and text preferred size
final Dimension iconTextAvailable = new Dimension ( available.width - lw - rw, available.height - th - bh );
final Dimension iconText = getIconTextPreferredSize ( c, d, iconTextAvailable );
// Calculating resulting preferred sizes
final int pw = MathUtils.max ( top.width, lw + iconText.width + rw, bottom.width );
final int ph = th + MathUtils.max ( left.height, iconText.height, right.height ) + bh;
return new Dimension ( pw, ph );
}
/**
* Returns icon and text preferred size.
*
* @param c painted component
* @param d painted decoration state
* @param available available space
* @return icon and text preferred size
*/
@NotNull
protected Dimension getIconTextPreferredSize ( @NotNull final C c, @NotNull final D d, @NotNull final Dimension available )
{
final Dimension size;
final boolean hasIcon = !isEmpty ( c, d, ICON );
final boolean hasText = !isEmpty ( c, d, TEXT );
if ( hasIcon && hasText )
{
final int hpos = getHorizontalTextPosition ( c, d );
final int vpos = getVerticalTextPosition ( c, d );
final Dimension ips = getPreferredSize ( c, d, available, ICON );
if ( hpos != CENTER || vpos != CENTER )
{
final int gap = getIconTextGap ( c, d );
if ( hpos == LEFT || hpos == LEADING || hpos == RIGHT || hpos == TRAILING )
{
final Dimension havailable = new Dimension ( available.width - gap - ips.width, available.height );
final Dimension cps = getPreferredSize ( c, d, havailable, TEXT );
size = new Dimension ( ips.width + gap + cps.width, Math.max ( ips.height, cps.height ) );
}
else
{
final Dimension vavailable = new Dimension ( available.width, available.height - gap - ips.height );
final Dimension cps = getPreferredSize ( c, d, vavailable, TEXT );
size = new Dimension ( Math.max ( ips.width, cps.width ), ips.height + gap + cps.height );
}
}
else
{
size = SwingUtils.max ( ips, getPreferredSize ( c, d, available, TEXT ) );
}
}
else if ( hasIcon )
{
size = getPreferredSize ( c, d, available, ICON );
}
else if ( hasText )
{
size = getPreferredSize ( c, d, available, TEXT );
}
else
{
size = new Dimension ( 0, 0 );
}
return size;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy