com.alee.painter.decoration.AbstractDecorationPainter 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.clone.Clone;
import com.alee.api.jdk.Objects;
import com.alee.api.merge.Merge;
import com.alee.extended.behavior.AbstractHoverBehavior;
import com.alee.laf.WebLookAndFeel;
import com.alee.laf.grouping.GroupingLayout;
import com.alee.managers.focus.DefaultFocusTracker;
import com.alee.managers.focus.FocusManager;
import com.alee.managers.focus.FocusTracker;
import com.alee.managers.focus.GlobalFocusListener;
import com.alee.managers.style.Bounds;
import com.alee.managers.style.BoundsType;
import com.alee.managers.style.PainterShapeProvider;
import com.alee.managers.style.StyleManager;
import com.alee.painter.AbstractPainter;
import com.alee.painter.SectionPainter;
import com.alee.utils.*;
import javax.swing.*;
import javax.swing.plaf.ComponentUI;
import java.awt.*;
import java.awt.event.ContainerEvent;
import java.awt.event.ContainerListener;
import java.awt.event.HierarchyEvent;
import java.awt.event.HierarchyListener;
import java.util.*;
import java.util.List;
/**
* Abstract decoration painter that can be used by any custom and specific painter.
*
* @param component type
* @param component UI type
* @param decoration type
* @author Mikle Garin
*/
public abstract class AbstractDecorationPainter>
extends AbstractPainter implements IDecorationPainter, PainterShapeProvider
{
/**
* Decoratable states property.
*/
public static final String DECORATION_STATES_PROPERTY = "decorationStates";
/**
* Available decorations.
* Each decoration provides a visual representation of specific component state.
*/
protected Decorations decorations;
/**
* Listeners.
*/
protected transient FocusTracker focusStateTracker;
protected transient GlobalFocusListener inFocusedParentTracker;
protected transient AbstractHoverBehavior hoverStateTracker;
protected transient HierarchyListener hierarchyTracker;
protected transient ContainerListener neighboursTracker;
/**
* Runtime variables.
*/
protected transient List states;
protected transient Map stateDecorationCache;
protected transient Map decorationCache;
protected transient String current;
protected transient boolean focused;
protected transient boolean inFocusedParent;
protected transient boolean hover;
protected transient Container ancestor;
@Override
protected void afterInstall ()
{
/**
* Determining initial decoration states.
* We should always update states last to ensure initial values are correct.
* Although we still do it before updating border in {@link super#afterInstall()}.
*/
this.states = collectDecorationStates ();
/**
* Performing basic actions after installation ends.
*/
super.afterInstall ();
}
@Override
protected void beforeUninstall ()
{
/**
* Performing basic actions before uninstallation starts.
*/
super.beforeUninstall ();
/**
* Making sure last used decoration is properly deactivated.
* If we don't deactivate last decoration it will stay active and will keep recieving component updates.
*/
deactivateLastDecoration ( component );
}
@Override
protected void afterUninstall ()
{
/**
* Cleaning up decoration caches.
*/
this.stateDecorationCache = null;
this.decorationCache = null;
this.states = null;
/**
* Performing basic actions after uninstallation ends.
*/
super.afterUninstall ();
}
@Override
protected void installPropertiesAndListeners ()
{
super.installPropertiesAndListeners ();
// Installing various extra listeners
installFocusListener ();
installInFocusedParentListener ();
installHoverListener ();
installHierarchyListener ();
}
@Override
protected void uninstallPropertiesAndListeners ()
{
// Uninstalling various extra listeners
uninstallHierarchyListener ();
uninstallHoverListener ();
uninstallInFocusedParentListener ();
uninstallFocusListener ();
super.uninstallPropertiesAndListeners ();
}
@Override
protected void propertyChanged ( final String property, final Object oldValue, final Object newValue )
{
// Perform basic actions on property changes
super.propertyChanged ( property, oldValue, newValue );
// Updating focus listener
if ( Objects.equals ( property, WebLookAndFeel.FOCUSABLE_PROPERTY ) )
{
updateFocusListener ();
}
// Updating custom decoration states
if ( Objects.equals ( property, DECORATION_STATES_PROPERTY ) )
{
updateDecorationState ();
}
// Updating enabled state
if ( Objects.equals ( property, WebLookAndFeel.ENABLED_PROPERTY ) )
{
if ( usesState ( DecorationState.enabled ) || usesState ( DecorationState.disabled ) )
{
updateDecorationState ();
}
}
}
/**
* Overridden to provide decoration states update instead of simple view updates.
*/
@Override
protected void orientationChange ()
{
// Saving new orientation
saveOrientation ();
// Updating decoration states
updateDecorationState ();
}
/**
* Returns whether or not component has distinct focused view.
* Note that this is exactly distinct view and not state, distinct focused state might actually be missing.
*
* @return {@code true} if component has distinct focused view, {@code false} otherwise
*/
protected boolean usesFocusedView ()
{
return component.isFocusable () && usesState ( DecorationState.focused );
}
/**
* Installs listener that will perform decoration updates on focus state change.
*/
protected void installFocusListener ()
{
if ( usesFocusedView () )
{
focusStateTracker = new DefaultFocusTracker ( component, true )
{
@Override
public void focusChanged ( final boolean focused )
{
// Ensure component is still available
// This might happen if painter is replaced from another FocusTracker
if ( component != null )
{
AbstractDecorationPainter.this.focusChanged ( focused );
}
}
};
FocusManager.addFocusTracker ( component, focusStateTracker );
this.focused = focusStateTracker.isFocused ();
}
else
{
this.focused = false;
}
}
/**
* Informs about focus state changes.
* Note that this method will only be fired when component {@link #usesFocusedView()}.
*
* @param focused whether or not component has focus
*/
protected void focusChanged ( final boolean focused )
{
this.focused = focused;
updateDecorationState ();
}
/**
* Returns whether or not component has focus.
*
* @return true if component has focus, false otherwise
*/
protected boolean isFocused ()
{
return focused;
}
/**
* Uninstalls focus listener.
*/
protected void uninstallFocusListener ()
{
if ( focusStateTracker != null )
{
FocusManager.removeFocusTracker ( component, focusStateTracker );
focusStateTracker = null;
focused = false;
}
}
/**
* Updates focus listener usage.
*/
protected void updateFocusListener ()
{
if ( usesFocusedView () )
{
installFocusListener ();
}
else
{
uninstallFocusListener ();
}
}
/**
* Returns whether or not component has distinct in-focused-parent view.
* Note that this is exactly distinct view and not state, distinct in-focused-parent state might actually be missing.
*
* @return {@code true} if component has distinct in-focused-parent view, {@code false} otherwise
*/
protected boolean usesInFocusedParentView ()
{
return usesState ( DecorationState.inFocusedParent );
}
/**
* Installs listener that performs decoration updates on focused parent appearance and disappearance.
*/
protected void installInFocusedParentListener ()
{
if ( usesInFocusedParentView () )
{
inFocusedParent = updateInFocusedParent ();
inFocusedParentTracker = new GlobalFocusListener ()
{
@Override
public void focusChanged ( final Component oldFocus, final Component newFocus )
{
// Ensure component is still available
// This might happen if painter is replaced from another GlobalFocusListener
if ( component != null )
{
// Updating {@link #inFocusedParent} state
updateInFocusedParent ();
}
}
};
FocusManager.registerGlobalFocusListener ( component, inFocusedParentTracker );
}
else
{
inFocusedParent = false;
}
}
/**
* Updates {@link #inFocusedParent} mark.
* For that we check whether or not any related component gained or lost focus.
* Note that this method will only be fired when component {@link #usesInFocusedParentView()}.
*
* @return {@code true} if component is placed within focused parent, {@code false} otherwise
*/
protected boolean updateInFocusedParent ()
{
final boolean old = inFocusedParent;
inFocusedParent = false;
Container current = component;
while ( current != null )
{
if ( current.isFocusOwner () )
{
// Directly in a focused parent
inFocusedParent = true;
break;
}
else if ( current != component && current instanceof JComponent )
{
// Ensure that component supports styling
final JComponent jComponent = ( JComponent ) current;
if ( LafUtils.hasWebLafUI ( jComponent ) )
{
// In a parent that tracks children focus and visually displays it
// This case is not obvious but really important for correct visual representation of the state
final ComponentUI ui = LafUtils.getUI ( jComponent );
if ( ui != null )
{
// todo Replace with proper painter retrieval upon Paintable interface implementation
final Object painter = ReflectUtils.getFieldValueSafely ( ui, "painter" );
if ( painter != null && painter instanceof AbstractDecorationPainter )
{
final AbstractDecorationPainter dp = ( AbstractDecorationPainter ) painter;
if ( dp.usesFocusedView () )
{
inFocusedParent = dp.isFocused ();
break;
}
}
}
}
}
current = current.getParent ();
}
if ( Objects.notEquals ( old, inFocusedParent ) )
{
updateDecorationState ();
}
return inFocusedParent;
}
/**
* Returns whether or not one of this component parents displays focused state.
* Returns {@code true} if one of parents owns focus directly or indirectly due to one of its children being focused.
* Indirect focus ownership is only accepted from parents which use {@link DecorationState#focused} decoration state.
*
* @return {@code true} if one of this component parents displays focused state, {@code false} otherwise
*/
protected boolean isInFocusedParent ()
{
return inFocusedParent;
}
/**
* Uninstalls global focus listener.
*/
protected void uninstallInFocusedParentListener ()
{
if ( inFocusedParentTracker != null )
{
FocusManager.unregisterGlobalFocusListener ( component, inFocusedParentTracker );
inFocusedParentTracker = null;
inFocusedParent = false;
}
}
/**
* Returns whether or not component has distinct hover view.
* Note that this is exactly distinct view and not state, distinct hover state might actually be missing.
*
* @return {@code true} if component has distinct hover view, {@code false} otherwise
*/
protected boolean usesHoverView ()
{
return usesState ( DecorationState.hover );
}
/**
* Installs listener that will perform decoration updates on hover state change.
*/
protected void installHoverListener ()
{
if ( usesHoverView () )
{
hover = CoreSwingUtils.isHovered ( component );
hoverStateTracker = new AbstractHoverBehavior ( component, false )
{
@Override
public void hoverChanged ( final boolean hover )
{
// Ensure component is still available
// This might happen if painter is replaced from another AbstractHoverBehavior
if ( component != null )
{
AbstractDecorationPainter.this.hoverChanged ( hover );
}
}
};
hoverStateTracker.install ();
}
else
{
hover = false;
}
}
/**
* Informs about hover state changes.
* Note that this method will only be fired when component {@link #usesHoverView()}.
*
* @param hover whether or not mouse is on the component
*/
protected void hoverChanged ( final boolean hover )
{
this.hover = hover;
updateDecorationState ();
}
/**
* Returns whether or not component is in hover state.
*
* @return true if component is in hover state, false otherwise
*/
protected boolean isHover ()
{
return hover;
}
/**
* Uninstalls hover listener.
*/
protected void uninstallHoverListener ()
{
if ( hoverStateTracker != null )
{
hoverStateTracker.uninstall ();
hoverStateTracker = null;
hover = false;
}
}
/**
* Updates hover listener usage.
*/
protected void updateHoverListener ()
{
if ( usesHoverView () )
{
installHoverListener ();
}
else
{
uninstallHoverListener ();
}
}
/**
* Returns whether or not component has distinct hierarchy-based view.
*
* @return {@code true} if component has distinct hierarchy-based view, {@code false} otherwise
*/
protected boolean usesHierarchyBasedView ()
{
return true;
}
/**
* Installs listener that will perform border updates on component hierarchy changes.
* This is required to properly update decoration borders in case it was moved from or into container with grouping layout.
* It also tracks neighbour components addition and removal to update this component border accordingly.
*/
protected void installHierarchyListener ()
{
if ( usesHierarchyBasedView () )
{
neighboursTracker = new ContainerListener ()
{
@Override
public void componentAdded ( final ContainerEvent e )
{
// Ensure component is still available
// This might happen if painter is replaced from another ContainerListener
if ( component != null )
{
// Updating border when a child was added nearby
if ( ancestor != null && ancestor.getLayout () instanceof GroupingLayout && e.getChild () != component )
{
updateBorder ();
}
}
}
@Override
public void componentRemoved ( final ContainerEvent e )
{
// Ensure component is still available
// This might happen if painter is replaced from another ContainerListener
if ( component != null )
{
// Updating border when a child was removed nearby
if ( ancestor != null && ancestor.getLayout () instanceof GroupingLayout && e.getChild () != component )
{
updateBorder ();
}
}
}
};
hierarchyTracker = new HierarchyListener ()
{
@Override
public void hierarchyChanged ( final HierarchyEvent e )
{
// Ensure component is still available
// This might happen if painter is replaced from another HierarchyListener
if ( component != null )
{
AbstractDecorationPainter.this.hierarchyChanged ( e );
}
}
};
component.addHierarchyListener ( hierarchyTracker );
}
else
{
ancestor = null;
}
}
/**
* Informs about hierarchy changes.
* Note that this method will only be fired when component {@link #usesHierarchyBasedView()}.
*
* @param e {@link java.awt.event.HierarchyEvent}
*/
protected void hierarchyChanged ( final HierarchyEvent e )
{
// Listening only for parent change event
// It will inform us when parent container for this component changes
// Ancestor listener is not really reliable because it might inform about consequent parent changes
if ( ( e.getChangeFlags () & HierarchyEvent.PARENT_CHANGED ) == HierarchyEvent.PARENT_CHANGED )
{
// If there was a previous container...
if ( ancestor != null )
{
// Stop tracking neighbours
ancestor.removeContainerListener ( neighboursTracker );
}
// Updating ancestor
ancestor = component.getParent ();
// If there is a new container...
if ( ancestor != null )
{
// Start tracking neighbours
ancestor.addContainerListener ( neighboursTracker );
// Updating border
updateBorder ();
}
}
}
/**
* Uninstalls hierarchy listener.
*/
protected void uninstallHierarchyListener ()
{
if ( hierarchyTracker != null )
{
component.removeHierarchyListener ( hierarchyTracker );
hierarchyTracker = null;
if ( ancestor != null )
{
ancestor.removeContainerListener ( neighboursTracker );
ancestor = null;
}
neighboursTracker = null;
}
}
/**
* Returns whether or not component is in enabled state.
*
* @return {@code true} if component is in enabled state, {@code false} otherwise
*/
protected boolean isEnabled ()
{
return component != null && component.isEnabled ();
}
/**
* Returns properly sorted current component decoration states.
*
* @return properly sorted current component decoration states
*/
protected final List collectDecorationStates ()
{
// Retrieving current decoration states
final List states = getDecorationStates ();
// Adding custom Skin decoration states
states.addAll ( DecorationUtils.getExtraStates ( StyleManager.getSkin ( component ) ) );
// Adding custom UI decoration states
states.addAll ( DecorationUtils.getExtraStates ( ui ) );
// Adding custom component decoration states
states.addAll ( DecorationUtils.getExtraStates ( component ) );
// Sorting states to always keep the same order
Collections.sort ( states );
return states;
}
@Override
public List getDecorationStates ()
{
final List states = new ArrayList ( 12 );
states.add ( SystemUtils.getShortOsName () );
states.add ( isEnabled () ? DecorationState.enabled : DecorationState.disabled );
states.add ( ltr ? DecorationState.leftToRight : DecorationState.rightToLeft );
if ( isFocused () )
{
states.add ( DecorationState.focused );
}
if ( isInFocusedParent () )
{
states.add ( DecorationState.inFocusedParent );
}
if ( isHover () )
{
states.add ( DecorationState.hover );
}
return states;
}
@Override
public final boolean usesState ( final String state )
{
// Checking whether or not this painter uses this decoration state
boolean usesState = usesState ( decorations, state );
// Checking whether or not section painters used by this painter use it
if ( !usesState )
{
final List> sectionPainters = getInstalledSectionPainters ();
if ( CollectionUtils.notEmpty ( sectionPainters ) )
{
for ( final SectionPainter section : sectionPainters )
{
if ( section instanceof IDecorationPainter )
{
if ( ( ( IDecorationPainter ) section ).usesState ( state ) )
{
usesState = true;
break;
}
}
}
}
}
return usesState;
}
/**
* Returns whether specified decorations are associated with specified state.
*
* @param decorations decorations
* @param state decoration state
* @return {@code true} if specified decorations are associated with specified state, {@code false} otherwise
*/
protected final boolean usesState ( final Decorations decorations, final String state )
{
if ( decorations != null && decorations.size () > 0 )
{
for ( final D decoration : decorations )
{
if ( decoration.usesState ( state ) )
{
return true;
}
}
}
return false;
}
/**
* Returns decorations for the specified states.
*
* @param forStates decoration states to retrieve decoration for
* @return decorations for the specified states
*/
protected final List getDecorations ( final List forStates )
{
if ( decorations != null && decorations.size () > 0 )
{
final List d = new ArrayList ( 1 );
for ( final D decoration : decorations )
{
if ( decoration.isApplicableTo ( forStates ) )
{
d.add ( decoration );
}
}
return d;
}
else
{
return null;
}
}
@Override
public final D getDecoration ()
{
// Optimization for painter without decorations
if ( decorations != null && decorations.size () > 0 )
{
// Decoration key
// States are properly sorted, so their order is always the same
final String previous = this.current;
current = TextUtils.listToString ( states, "," );
// Creating decoration caches
if ( stateDecorationCache == null )
{
// State decorations cache
// Entry: [ component state -> built decoration reference ]
// It is used for fastest possible access to component state decorations
stateDecorationCache = new HashMap ( decorations.size () );
// Decoration combinations cache
// Entry: [ decorations combination key -> built decoration reference ]
// It is used to avoid excessive memory usage by duplicate decoration combinations for each specific state
decorationCache = new HashMap ( decorations.size () );
}
// Resolving state decoration if it is not yet cached
if ( !stateDecorationCache.containsKey ( current ) )
{
// Retrieving all decorations fitting current states
final List decorations = getDecorations ( states );
// Retrieving unique key for decorations combination
final String decorationsKey = getDecorationsKey ( decorations );
// Retrieving existing decoration or building a new one
final D decoration;
if ( decorationCache.containsKey ( decorationsKey ) )
{
// Retrieving decoration from existing built decorations cache
decoration = decorationCache.get ( decorationsKey );
}
else
{
// Building single decoration from a set
if ( CollectionUtils.isEmpty ( decorations ) )
{
// No decoration for the states available
decoration = null;
}
else if ( decorations.size () == 1 )
{
// Single existing decoration for the states
decoration = Clone.deep ().clone ( decorations.get ( 0 ) );
}
else
{
// Filter out possible decorations of different type
// We always use type of the last one available since it has higher priority
final Class extends IDecoration> type = decorations.get ( decorations.size () - 1 ).getClass ();
final Iterator iterator = decorations.iterator ();
while ( iterator.hasNext () )
{
final D d = iterator.next ();
if ( d.getClass () != type )
{
iterator.remove ();
}
}
// Merging multiple decorations together
decoration = Merge.deep ().merge ( decorations );
}
// Updating built decoration settings
if ( decoration != null )
{
// Updating section mark
// This is done for each cached decoration once as it doesn't change
decoration.setSection ( isSectionPainter () );
}
// Caching built decoration
decorationCache.put ( decorationsKey, decoration );
}
// Caching resulting decoration under the state key
stateDecorationCache.put ( current, decoration );
}
// Performing decoration activation and deactivation if needed
if ( previous == null && current == null )
{
// Activating initial decoration
final D initialDecoration = stateDecorationCache.get ( current );
if ( initialDecoration != null )
{
initialDecoration.activate ( component );
}
}
else if ( Objects.notEquals ( previous, current ) )
{
// Checking that decoration was actually changed
final D previousDecoration = stateDecorationCache.get ( previous );
final D currentDecoration = stateDecorationCache.get ( current );
if ( previousDecoration != currentDecoration )
{
// Deactivating previous decoration
if ( previousDecoration != null )
{
previousDecoration.deactivate ( component );
}
// Activating current decoration
if ( currentDecoration != null )
{
currentDecoration.activate ( component );
}
}
}
// Returning existing decoration
return stateDecorationCache.get ( current );
}
else
{
// No decorations added
return null;
}
}
/**
* Returns unique decorations combination key.
*
* @param decorations decorations to retrieve unique combination key for
* @return unique decorations combination key
*/
protected final String getDecorationsKey ( final List decorations )
{
final StringBuilder key = new StringBuilder ( 15 * decorations.size () );
for ( final D decoration : decorations )
{
if ( key.length () > 0 )
{
key.append ( ";" );
}
key.append ( decoration.getId () );
}
return key.toString ();
}
/**
* Performs deactivation of the recently used decoration.
*
* @param c painted component
*/
protected final void deactivateLastDecoration ( final C c )
{
final D decoration = getDecoration ();
if ( decoration != null )
{
decoration.deactivate ( c );
}
}
@Override
public final void updateDecorationState ()
{
final List states = collectDecorationStates ();
if ( !CollectionUtils.equals ( this.states, states, true ) )
{
// Saving new decoration states
this.states = states;
// Updating section painters decoration states
// This is required to provide state changes into section painters used within this painter
// Section painters that use default states collection mechanism are dependant on origin painter states
final List> sectionPainters = getInstalledSectionPainters ();
if ( CollectionUtils.notEmpty ( sectionPainters ) )
{
for ( final SectionPainter section : sectionPainters )
{
if ( section instanceof IDecorationPainter )
{
( ( IDecorationPainter ) section ).updateDecorationState ();
}
}
}
// Updating component visual state
revalidate ();
repaint ();
}
}
@Override
protected Insets getBorder ()
{
final D decoration = getDecoration ();
if ( isDecorationAvailable ( decoration ) )
{
// Return decoration border insets
return decoration.getBorderInsets ( component );
}
else
{
// Return {@code null} insets
return null;
}
}
@Override
public Shape provideShape ( final C component, final Rectangle bounds )
{
final D decoration = getDecoration ();
if ( isDecorationAvailable ( decoration ) )
{
// Return shape provided by the decoration
return decoration.provideShape ( component, bounds );
}
else
{
// Simply return bounds
return bounds;
}
}
@Override
public Boolean isOpaque ()
{
final D decoration = getDecoration ();
if ( isDecorationAvailable ( decoration ) )
{
// Decorated components opacity check
return isOpaqueDecorated ();
}
else
{
// Undecorated components opacity check
return isOpaqueUndecorated ();
}
}
/**
* Returns opacity state for decorated component.
* This is separated from base opacity state method to allow deep customization.
*
* @return opacity state for decorated component
*/
protected Boolean isOpaqueDecorated ()
{
// Returns {@code false} to make component non-opaque when decoration is available
// This is convenient because almost any custom decoration might have transparent spots in it
// And if it has any it cannot be opaque or it will cause major painting glitches
return false;
}
/**
* Returns opacity state for undecorated component.
* This is separated from base opacity state method to allow deep customization.
*
* @return opacity state for undecorated component
*/
protected Boolean isOpaqueUndecorated ()
{
// Returns {@code null} to disable automatic opacity changes by default
// You may still provide a non-null opacity in your own painter implementations
return null;
}
@Override
public boolean contains ( final C c, final U ui, final Bounds bounds, final int x, final int y )
{
final D decoration = getDecoration ();
if ( isDecorationAvailable ( decoration ) )
{
// Creating additional bounds
final Bounds marginBounds = new Bounds ( bounds, BoundsType.margin, c, decoration );
// Using decoration contains method
return decoration.contains ( c, marginBounds, x, y );
}
else
{
// Using default contains method
return super.contains ( c, ui, bounds, x, y );
}
}
@Override
public int getBaseline ( final C c, final U ui, final Bounds bounds )
{
final D decoration = getDecoration ();
if ( isDecorationAvailable ( decoration ) )
{
// Creating additional bounds
final Bounds marginBounds = new Bounds ( bounds, BoundsType.margin, c, decoration );
// Calculating decoration baseline
return decoration.getBaseline ( c, marginBounds );
}
else
{
// Calculating default baseline
return super.getBaseline ( c, ui, bounds );
}
}
@Override
public Component.BaselineResizeBehavior getBaselineResizeBehavior ( final C c, final U ui )
{
final D decoration = getDecoration ();
if ( isDecorationAvailable ( decoration ) )
{
// Returning decoration baseline behavior
return decoration.getBaselineResizeBehavior ( c );
}
else
{
// Returning default baseline behavior
return super.getBaselineResizeBehavior ( c, ui );
}
}
@Override
public void paint ( final Graphics2D g2d, final C c, final U ui, final Bounds bounds )
{
// Checking whether plain background is required
if ( isPlainBackgroundRequired ( c ) )
{
// Painting simple background
// This block added to avoid various visual glitches
g2d.setPaint ( c.getBackground () );
g2d.fill ( bounds.get () );
}
// Painting current decoration state
final D decoration = getDecoration ();
if ( isDecorationAvailable ( decoration ) )
{
// Creating additional bounds
final Bounds marginBounds = new Bounds ( bounds, BoundsType.margin, c, decoration );
// Painting current decoration state
decoration.paint ( g2d, c, marginBounds );
}
// Painting content
paintContent ( g2d, bounds.get (), c, ui );
}
/**
* Paints additional custom content provided by this painter.
* todo This might eventually be removed if all contents will be painted within IContent implementations
*
* @param g2d graphics context
* @param bounds painting bounds
* @param c painted component
* @param ui painted component UI
*/
protected void paintContent ( final Graphics2D g2d, final Rectangle bounds, final C c, final U ui )
{
// No content by default
}
/**
* Returns whether or not painting plain component background is required.
* When component is opaque we must fill every single pixel in its bounds with something to avoid issues.
* By default this condition is limited to component being opaque.
*
* @param c component to paint background for
* @return {@code true} if painting plain component background is required, {@code false} otherwise
*/
protected boolean isPlainBackgroundRequired ( final C c )
{
return c.isOpaque ();
}
/**
* Returns whether or not painting specified decoration is available.
*
* @param decoration decoration to be painted
* @return {@code true} if painting specified decoration is available, {@code false} otherwise
*/
protected boolean isDecorationAvailable ( final D decoration )
{
return decoration != null;
}
@Override
public Dimension getPreferredSize ()
{
final Dimension ps = super.getPreferredSize ();
final D d = getDecoration ();
return d != null ? SwingUtils.max ( d.getPreferredSize ( component ), ps ) : ps;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy