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

com.alee.painter.decoration.WebDecoration 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;

import com.alee.api.annotations.NotNull;
import com.alee.managers.style.Bounds;
import com.alee.painter.decoration.background.IBackground;
import com.alee.painter.decoration.border.BorderWidth;
import com.alee.painter.decoration.border.IBorder;
import com.alee.painter.decoration.border.LineBorder;
import com.alee.painter.decoration.shadow.IShadow;
import com.alee.painter.decoration.shadow.ShadowType;
import com.alee.painter.decoration.shape.IPartialShape;
import com.alee.painter.decoration.shape.IShape;
import com.alee.painter.decoration.shape.ShapeType;
import com.alee.utils.CollectionUtils;
import com.alee.utils.GraphicsUtils;
import com.alee.utils.SwingUtils;
import com.thoughtworks.xstream.annotations.XStreamAlias;
import com.thoughtworks.xstream.annotations.XStreamImplicit;

import javax.swing.*;
import java.awt.*;
import java.awt.geom.Rectangle2D;
import java.util.ArrayList;
import java.util.List;

/**
 * Configurable decoration state used for most WebLaF components.
 * It provides basic elements required to paint component parts.
 *
 * @param  component type
 * @param  decoration type
 * @author Mikle Garin
 */
@XStreamAlias ( "decoration" )
public class WebDecoration> extends ContentDecoration
{
    /**
     * Decoration shape.
     * It defines the shape of the decoration shadow, border, background and might be used for other decoration elements.
     * Implicit list is only used to provide convenient XML descriptor for this field, only one shape can be provided at a time.
     *
     * @see com.alee.painter.decoration.shape.IShape
     * @see com.alee.painter.decoration.shape.AbstractShape
     */
    @XStreamImplicit
    protected List shapes = new ArrayList ( 1 );

    /**
     * Optional decoration shadows.
     * Maximum two different shadows could be provided at the same time - outer and inner.
     *
     * @see com.alee.painter.decoration.shadow.IShadow
     * @see com.alee.painter.decoration.shadow.AbstractShadow
     */
    @XStreamImplicit
    protected List shadows = new ArrayList ( 1 );

    /**
     * Optional decoration border.
     * Implicit list is used to provide convenient XML descriptor for this field.
     * Right now only single border can be used per decoration instance, but that might change in future.
     *
     * @see com.alee.painter.decoration.border.IBorder
     * @see com.alee.painter.decoration.border.AbstractBorder
     */
    @XStreamImplicit
    protected List borders = new ArrayList ( 1 );

    /**
     * Optional decoration backgrounds.
     * Multiple backgrounds can be used per decoration instance.
     * Though it is not reasonable to use backgrounds which will fully overlap eachother.
     *
     * @see com.alee.painter.decoration.background.IBackground
     * @see com.alee.painter.decoration.background.AbstractBackground
     */
    @XStreamImplicit
    protected List backgrounds = new ArrayList ( 1 );

    @Override
    public void activate ( final C c )
    {
        // Activating content
        super.activate ( c );

        // Activating decoration elements
        if ( hasShape () )
        {
            getShape ().activate ( c, this );
        }
        if ( hasShadow ( ShadowType.outer ) )
        {
            getShadow ( ShadowType.outer ).activate ( c, this );
        }
        if ( hasShadow ( ShadowType.inner ) )
        {
            getShadow ( ShadowType.inner ).activate ( c, this );
        }
        if ( hasBorder () )
        {
            getBorder ().activate ( c, this );
        }
        if ( hasBackground () )
        {
            for ( final IBackground background : getBackgrounds () )
            {
                background.activate ( c, this );
            }
        }
    }

    @Override
    public void deactivate ( final C c )
    {
        // Deactivating content
        super.deactivate ( c );

        // Deactivating decoration elements
        if ( hasShape () )
        {
            getShape ().deactivate ( c, this );
        }
        if ( hasShadow ( ShadowType.outer ) )
        {
            getShadow ( ShadowType.outer ).deactivate ( c, this );
        }
        if ( hasShadow ( ShadowType.inner ) )
        {
            getShadow ( ShadowType.inner ).deactivate ( c, this );
        }
        if ( hasBorder () )
        {
            getBorder ().deactivate ( c, this );
        }
        if ( hasBackground () )
        {
            for ( final IBackground background : getBackgrounds () )
            {
                background.deactivate ( c, this );
            }
        }
    }

    @Override
    public boolean isVisible ()
    {
        return super.isVisible () && hasShape ();
    }

    /**
     * Returns whether or not decoration has shape.
     *
     * @return true if decoration has shape, false otherwise
     */
    public boolean hasShape ()
    {
        return CollectionUtils.notEmpty ( shapes );
    }

    /**
     * Returns decoration shape.
     *
     * @return decoration shape
     */
    public IShape getShape ()
    {
        if ( CollectionUtils.notEmpty ( shapes ) )
        {
            if ( shapes.size () == 1 )
            {
                return shapes.get ( 0 );
            }
            else
            {
                throw new DecorationException ( "Only one shape can be provided per decoration" );
            }
        }
        else
        {
            return null;
        }
    }

    /**
     * Returns whether or not decoration has shadow.
     *
     * @param type shadow type
     * @return true if decoration has shadow, false otherwise
     */
    public boolean hasShadow ( final ShadowType type )
    {
        return CollectionUtils.notEmpty ( shadows ) && getShadow ( type ) != null;
    }

    /**
     * Returns shadow of the specified type.
     *
     * @param type shadow type
     * @return shadow of the specified type
     */
    public IShadow getShadow ( final ShadowType type )
    {
        if ( CollectionUtils.notEmpty ( shadows ) )
        {
            if ( shadows.size () == 1 )
            {
                return shadows.get ( 0 ).getType () == type ? shadows.get ( 0 ) : null;
            }
            else if ( shadows.size () == 2 )
            {
                if ( shadows.get ( 0 ).getType () == type )
                {
                    return shadows.get ( 0 );
                }
                else if ( shadows.get ( 1 ).getType () == type )
                {
                    return shadows.get ( 1 );
                }
                else
                {
                    throw new DecorationException ( "Provided shadows must have distinct types" );
                }
            }
            else
            {
                throw new DecorationException ( "Only two shadows with different types can be provided per decoration" );
            }
        }
        else
        {
            return null;
        }
    }

    /**
     * Returns width of the shadow with the specified type.
     *
     * @param type shadow type
     * @return width of the shadow with the specified type
     */
    public int getShadowWidth ( final ShadowType type )
    {
        final IShadow shadow = getShadow ( type );
        return shadow != null ? shadow.getWidth () : 0;
    }

    /**
     * Returns whether or not decoration has border.
     *
     * @return true if decoration has border, false otherwise
     */
    public boolean hasBorder ()
    {
        return CollectionUtils.notEmpty ( borders );
    }

    /**
     * Returns decoration border.
     *
     * @return decoration border
     */
    public IBorder getBorder ()
    {
        if ( CollectionUtils.notEmpty ( borders ) )
        {
            if ( borders.size () == 1 )
            {
                return borders.get ( 0 );
            }
            else
            {
                throw new DecorationException ( "Only one border can be provided per decoration" );
            }
        }
        else
        {
            return null;
        }
    }

    /**
     * Returns border width.
     *
     * @return border width
     */
    @NotNull
    public BorderWidth getBorderWidth ()
    {
        final IBorder border = getBorder ();
        return border != null ? border.getWidth () : BorderWidth.EMPTY;
    }

    /**
     * Returns whether or not decoration has at least one {@link IBackground}.
     *
     * @return {@code true} if decoration has at least one {@link IBackground}, {@code false} otherwise
     */
    public boolean hasBackground ()
    {
        return CollectionUtils.notEmpty ( backgrounds );
    }

    /**
     * Returns decoration {@link IBackground}s.
     *
     * @return decoration {@link IBackground}s
     */
    public List getBackgrounds ()
    {
        return CollectionUtils.notEmpty ( backgrounds ) ? backgrounds : null;
    }

    @Override
    public Insets getBorderInsets ( final C c )
    {
        final Insets insets = super.getBorderInsets ( c );
        if ( isVisible () )
        {
            final IShape shape = getShape ();
            final BorderWidth bw = getBorderWidth ();
            final int sw = getShadowWidth ( ShadowType.outer );
            if ( shape instanceof IPartialShape )
            {
                final IPartialShape ps = ( IPartialShape ) shape;
                final int top = ps.isPaintTop ( c, this ) ? bw.top + sw : ps.isPaintTopLine ( c, this ) ? bw.top : 0;
                final int left = ps.isPaintLeft ( c, this ) ? bw.left + sw : ps.isPaintLeftLine ( c, this ) ? bw.left : 0;
                final int bottom = ps.isPaintBottom ( c, this ) ? bw.bottom + sw : ps.isPaintBottomLine ( c, this ) ? bw.bottom : 0;
                final int right = ps.isPaintRight ( c, this ) ? bw.right + sw : ps.isPaintRightLine ( c, this ) ? bw.right : 0;
                SwingUtils.increase ( insets, new Insets ( top, left, bottom, right ) );
            }
            else
            {
                SwingUtils.increase ( insets, new Insets ( bw.top + sw, bw.left + sw, bw.bottom + sw, bw.right + sw ) );
            }
        }
        return insets;
    }

    @Override
    public boolean contains ( final C c, final Bounds b, final int x, final int y )
    {
        final boolean contains;
        if ( isVisible () )
        {
            final IShape shape = getShape ();
            if ( shape != null )
            {
                final Rectangle bounds = b.get ();
                final Shape backgroundShape = shape.getShape ( ShapeType.background, bounds, c, this );
                contains = backgroundShape.contains ( x, y );
            }
            else
            {
                contains = super.contains ( c, b, x, y );
            }
        }
        else
        {
            contains = super.contains ( c, b, x, y );
        }
        return contains;
    }

    @NotNull
    @Override
    public Shape provideShape ( @NotNull final C component, @NotNull final Rectangle bounds )
    {
        // todo Add ShapeType into PainterShapeProvider interface
        return isVisible () ? getShape ().getShape ( ShapeType.background, bounds, component, this ) : bounds;
    }

    @Override
    public void paint ( final Graphics2D g2d, final C c, final Bounds b )
    {
        // Painting only if bounds are enough
        final Rectangle bounds = b.get ();
        if ( bounds.width > 0 && bounds.height > 0 )
        {
            // Painting only if margin bounds ar enough and intersect visible area
            final Rectangle2D cl = g2d.getClip () instanceof Rectangle2D ? ( Rectangle2D ) g2d.getClip () : c.getVisibleRect ();
            if ( bounds.intersects ( cl ) )
            {
                // Painting decoration elements
                if ( isVisible () )
                {
                    // Setup settings
                    final Object oaa = GraphicsUtils.setupAntialias ( g2d );
                    final Composite oc = GraphicsUtils.setupAlphaComposite ( g2d, getOpacity (), getOpacity () < 1f );
                    final Shape ocl = GraphicsUtils.setupClip ( g2d, cl.createIntersection ( bounds ) );

                    // Painting decoration
                    paintDecoration ( g2d, c, b );

                    // Restoring settings
                    GraphicsUtils.restoreClip ( g2d, ocl );
                    GraphicsUtils.restoreAntialias ( g2d, oaa );
                    GraphicsUtils.restoreComposite ( g2d, oc );
                }

                // Painting contents
                paintContent ( g2d, b, c );
            }
        }
    }

    /**
     * Paints various component decoration parts.
     * All basic checks and painting preparations are done outside of this method.
     *
     * @param g2d graphics context
     * @param c   painted component
     * @param b   painting bounds
     */
    protected void paintDecoration ( final Graphics2D g2d, final C c, final Bounds b )
    {
        final Rectangle bounds = b.get ();

        // Base decoration shape
        final IShape shape = getShape ();

        // Outer shadow
        if ( hasShadow ( ShadowType.outer ) && shape.isVisible ( ShapeType.outerShadow, bounds, c, WebDecoration.this ) )
        {
            final Shape s = shape.getShape ( ShapeType.outerShadow, bounds, c, WebDecoration.this );
            getShadow ( ShadowType.outer ).paint ( g2d, bounds, c, WebDecoration.this, s );
        }

        // Painting all available backgrounds
        if ( hasBackground () && shape.isVisible ( ShapeType.background, bounds, c, WebDecoration.this ) )
        {
            final Shape s = shape.getShape ( ShapeType.background, bounds, c, WebDecoration.this );
            for ( final IBackground background : getBackgrounds () )
            {
                background.paint ( g2d, bounds, c, WebDecoration.this, s );
            }
        }

        // Painting inner shadow
        if ( hasShadow ( ShadowType.inner ) && shape.isVisible ( ShapeType.innerShadow, bounds, c, WebDecoration.this ) )
        {
            final Shape s = shape.getShape ( ShapeType.innerShadow, bounds, c, WebDecoration.this );
            getShadow ( ShadowType.inner ).paint ( g2d, bounds, c, WebDecoration.this, s );
        }

        // Painting border
        if ( hasBorder () && shape.isVisible ( ShapeType.border, bounds, c, WebDecoration.this ) )
        {
            final Shape s = shape.getShape ( ShapeType.border, bounds, c, WebDecoration.this );
            final IBorder border = getBorder ();
            border.paint ( g2d, bounds, c, WebDecoration.this, s );

            // Painting side lines
            // todo Use an interface instead of LineBorder?
            if ( shape instanceof IPartialShape && border instanceof LineBorder )
            {
                final IPartialShape ps = ( IPartialShape ) shape;
                final boolean ltr = c.getComponentOrientation ().isLeftToRight ();
                final boolean paintTop = ps.isPaintTop ( c, this );
                final boolean paintBottom = ps.isPaintBottom ( c, this );
                final boolean actualPaintLeft = ltr ? ps.isPaintLeft ( c, this ) : ps.isPaintRight ( c, this );
                final boolean actualPaintRight = ltr ? ps.isPaintRight ( c, this ) : ps.isPaintLeft ( c, this );
                final boolean paintTopLine = ps.isPaintTopLine ( c, this );
                final boolean paintBottomLine = ps.isPaintBottomLine ( c, this );
                final boolean actualPaintLeftLine = ltr ? ps.isPaintLeftLine ( c, this ) : ps.isPaintRightLine ( c, this );
                final boolean actualPaintRightLine = ltr ? ps.isPaintRightLine ( c, this ) : ps.isPaintLeftLine ( c, this );
                final int shadowWidth = getShadowWidth ( ShadowType.outer );

                final LineBorder lineBorder = ( LineBorder ) border;
                final Stroke stroke = lineBorder.getStroke ();
                final Stroke os = GraphicsUtils.setupStroke ( g2d, stroke, stroke != null );
                final Color bc = lineBorder.getColor ();
                final Paint op = GraphicsUtils.setupPaint ( g2d, bc, bc != null );

                // todo Better solution is required for side lines
                // todo Those should probably be a part of border instead of this
                if ( !paintTop && paintTopLine )
                {
                    final int x1 = bounds.x + ( actualPaintLeft ? shadowWidth : 0 );
                    final int x2 = bounds.x + bounds.width - ( actualPaintRight ? shadowWidth : 0 ) - 1;
                    g2d.drawLine ( x1, bounds.y, x2, bounds.y );
                }
                if ( !paintBottom && paintBottomLine )
                {
                    final int y = bounds.y + bounds.height - 1;
                    final int x1 = bounds.x + ( actualPaintLeft ? shadowWidth : 0 );
                    final int x2 = bounds.x + bounds.width - ( actualPaintRight ? shadowWidth : 0 ) - 1;
                    g2d.drawLine ( x1, y, x2, y );
                }
                if ( !actualPaintLeft && actualPaintLeftLine )
                {
                    final int y1 = bounds.y + ( paintTop ? shadowWidth : 0 );
                    final int y2 = bounds.y + bounds.height - ( paintBottom ? shadowWidth : 0 ) - 1;
                    g2d.drawLine ( bounds.x, y1, bounds.x, y2 );
                }
                if ( !actualPaintRight && actualPaintRightLine )
                {
                    final int x = bounds.x + bounds.width - 1;
                    final int y1 = bounds.y + ( paintTop ? shadowWidth : 0 );
                    final int y2 = bounds.y + bounds.height - ( paintBottom ? shadowWidth : 0 ) - 1;
                    g2d.drawLine ( x, y1, x, y2 );
                }

                GraphicsUtils.restorePaint ( g2d, op, bc != null );
                GraphicsUtils.restoreStroke ( g2d, os, stroke != null );
            }
        }

        // Painting neighbours border
        // todo PainterSupport.
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy