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