com.alee.laf.slider.SliderPainter Maven / Gradle / Ivy
package com.alee.laf.slider;
import com.alee.api.annotations.NotNull;
import com.alee.managers.style.Bounds;
import com.alee.painter.AbstractPainter;
import com.alee.utils.ColorUtils;
import com.alee.utils.GraphicsUtils;
import com.alee.utils.MathUtils;
import com.alee.utils.swing.WebTimer;
import javax.swing.*;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import java.awt.*;
import java.awt.event.*;
import java.awt.geom.GeneralPath;
import java.awt.geom.Rectangle2D;
import java.awt.geom.RoundRectangle2D;
import java.util.Dictionary;
import java.util.Enumeration;
/**
* Basic painter for {@link JSlider} component.
* It is used as {@link WebSliderUI} default painter.
*
* @param component type
* @param component UI type
* @author Alexandr Zernov
*/
public class SliderPainter extends AbstractPainter implements ISliderPainter
{
/**
* todo 1. Split into proper AbstractDecorationPainter & AbstractSectionDecorationPainter implementations
*/
/**
* Max rollover darkness.
*/
public static final int MAX_DARKNESS = 5;
/**
* Style settings.
*/
protected int trackHeight;
protected int trackRound;
protected int trackShadeWidth;
protected int progressRound;
protected int progressShadeWidth;
protected int thumbWidth;
protected int thumbHeight;
protected int thumbRound;
protected int thumbAngleLength;
protected Color trackBgTop;
protected Color trackBgBottom;
protected Color progressTrackBgTop;
protected Color progressTrackBgBottom;
protected Color progressBorderColor;
protected Color thumbBgTop;
protected Color thumbBgBottom;
protected boolean drawProgress;
protected boolean drawThumb;
protected boolean angledThumb;
protected boolean sharpThumbAngle;
protected boolean rolloverDarkBorderOnly;
protected boolean animated;
/**
* Listeners.
*/
protected transient MouseWheelListener mouseWheelListener;
protected transient ChangeListener changeListener;
protected transient MouseAdapter mouseAdapter;
/**
* Runtime variables.
*/
protected transient boolean rollover = false;
protected transient int rolloverDarkness = 0;
protected transient WebTimer rolloverTimer;
/**
* Painting variables.
*/
protected transient SliderPaintParameters paintParameters;
protected transient int trackBuffer = 0; // The distance that the track is from the side of the control
protected transient Rectangle focusRect = null;
protected transient Rectangle contentRect = null;
protected transient Rectangle labelRect = null;
protected transient Rectangle tickRect = null;
protected transient Rectangle trackRect = null;
protected transient Rectangle thumbRect = null;
@Override
protected void installPropertiesAndListeners ()
{
super.installPropertiesAndListeners ();
// Initializing caches
focusRect = new Rectangle ();
contentRect = new Rectangle ();
labelRect = new Rectangle ();
tickRect = new Rectangle ();
trackRect = new Rectangle ();
thumbRect = new Rectangle ();
// Rollover mouse wheel scroll
mouseWheelListener = new MouseWheelListener ()
{
@Override
public void mouseWheelMoved ( final MouseWheelEvent e )
{
if ( component.isEnabled () )
{
final int v = component.getValue () - e.getWheelRotation ();
component.setValue ( MathUtils.limit ( component.getMinimum (), v, component.getMaximum () ) );
}
}
};
component.addMouseWheelListener ( mouseWheelListener );
// todo Requires optimizations
// Fixed value change repaint
changeListener = new ChangeListener ()
{
@Override
public void stateChanged ( final ChangeEvent e )
{
calculateThumbLocation ();
component.repaint ();
}
};
component.addChangeListener ( changeListener );
// todo Requires optimizations
// Rollover animator
rolloverTimer = new WebTimer ( "WebProgressBarUI.animator", 40, new ActionListener ()
{
@Override
public void actionPerformed ( final ActionEvent e )
{
if ( rollover && rolloverDarkness < MAX_DARKNESS )
{
rolloverDarkness++;
component.repaint ();
}
else if ( !rollover && rolloverDarkness > 0 )
{
rolloverDarkness--;
component.repaint ();
}
else
{
rolloverTimer.stop ();
}
}
} );
mouseAdapter = new MouseAdapter ()
{
@Override
public void mousePressed ( final MouseEvent e )
{
component.repaint ();
}
@Override
public void mouseReleased ( final MouseEvent e )
{
component.repaint ();
}
@Override
public void mouseDragged ( final MouseEvent e )
{
component.repaint ();
}
@Override
public void mouseEntered ( final MouseEvent e )
{
rollover = true;
if ( animated && component.isEnabled () )
{
rolloverTimer.start ();
}
else
{
rolloverDarkness = MAX_DARKNESS;
component.repaint ();
}
}
@Override
public void mouseExited ( final MouseEvent e )
{
rollover = false;
if ( animated && component.isEnabled () )
{
rolloverTimer.start ();
}
else
{
rolloverDarkness = 0;
component.repaint ();
}
}
};
component.addMouseListener ( mouseAdapter );
component.addMouseMotionListener ( mouseAdapter );
}
@Override
protected void uninstallPropertiesAndListeners ()
{
// Removing listeners
component.removeMouseWheelListener ( mouseWheelListener );
mouseWheelListener = null;
component.removeChangeListener ( changeListener );
changeListener = null;
component.removeMouseListener ( mouseAdapter );
component.removeMouseMotionListener ( mouseAdapter );
mouseAdapter = null;
// Cleaning up caches
focusRect = null;
contentRect = null;
labelRect = null;
tickRect = null;
trackRect = null;
thumbRect = null;
super.uninstallPropertiesAndListeners ();
}
@Override
public void prepareToPaint ( @NotNull final SliderPaintParameters parameters )
{
this.paintParameters = parameters;
}
@Override
public void cleanupAfterPaint ()
{
this.paintParameters = null;
}
@Override
public void paint ( @NotNull final Graphics2D g2d, @NotNull final C c, @NotNull final U ui, @NotNull final Bounds bounds )
{
calculateGeometry ();
final Rectangle clip = g2d.getClipBounds ();
if ( component.getPaintTrack () && clip.intersects ( trackRect ) )
{
paintTrack ( g2d );
}
if ( component.getPaintTicks () && clip.intersects ( tickRect ) )
{
paintTicks ( g2d );
}
if ( component.getPaintLabels () && clip.intersects ( labelRect ) )
{
paintLabels ( g2d );
}
if ( clip.intersects ( thumbRect ) )
{
paintThumb ( g2d );
}
}
protected void calculateGeometry ()
{
calculateFocusRect ();
calculateContentRect ();
calculateThumbSize ();
calculateTrackBuffer ();
calculateTrackRect ();
calculateTickRect ();
calculateLabelRect ();
calculateThumbLocation ();
}
protected void calculateFocusRect ()
{
final Insets insets = component.getInsets ();
focusRect.x = insets.left;
focusRect.y = insets.top;
focusRect.width = component.getWidth () - ( insets.left + insets.right );
focusRect.height = component.getHeight () - ( insets.top + insets.bottom );
}
protected void calculateThumbSize ()
{
final Dimension size = getThumbSize ();
thumbRect.setSize ( size.width, size.height );
}
protected void calculateContentRect ()
{
contentRect.x = focusRect.x;
contentRect.y = focusRect.y;
contentRect.width = focusRect.width;
contentRect.height = focusRect.height;
}
protected void calculateThumbLocation ()
{
if ( component.getSnapToTicks () )
{
final int sliderValue = component.getValue ();
int snappedValue = sliderValue;
final int majorTickSpacing = component.getMajorTickSpacing ();
final int minorTickSpacing = component.getMinorTickSpacing ();
int tickSpacing = 0;
if ( minorTickSpacing > 0 )
{
tickSpacing = minorTickSpacing;
}
else if ( majorTickSpacing > 0 )
{
tickSpacing = majorTickSpacing;
}
if ( tickSpacing != 0 )
{
// If it's not on a tick, change the value
if ( ( sliderValue - component.getMinimum () ) % tickSpacing != 0 )
{
final float temp = ( float ) ( sliderValue - component.getMinimum () ) / ( float ) tickSpacing;
final int whichTick = Math.round ( temp );
snappedValue = component.getMinimum () + whichTick * tickSpacing;
}
if ( snappedValue != sliderValue )
{
component.setValue ( snappedValue );
}
}
}
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
final int valuePosition = xPositionForValue ( component.getValue () );
thumbRect.x = valuePosition - thumbRect.width / 2;
thumbRect.y = trackRect.y;
}
else
{
final int valuePosition = yPositionForValue ( component.getValue () );
thumbRect.x = trackRect.x;
thumbRect.y = valuePosition - thumbRect.height / 2;
}
}
protected void calculateTrackBuffer ()
{
if ( component.getPaintLabels () && component.getLabelTable () != null )
{
final Component highLabel = getHighestValueLabel ();
final Component lowLabel = getLowestValueLabel ();
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
trackBuffer = Math.max ( highLabel.getBounds ().width, lowLabel.getBounds ().width ) / 2;
trackBuffer = Math.max ( trackBuffer, thumbRect.width / 2 );
}
else
{
trackBuffer = Math.max ( highLabel.getBounds ().height, lowLabel.getBounds ().height ) / 2;
trackBuffer = Math.max ( trackBuffer, thumbRect.height / 2 );
}
}
else
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
trackBuffer = thumbRect.width / 2;
}
else
{
trackBuffer = thumbRect.height / 2;
}
}
}
protected void calculateTrackRect ()
{
int centerSpacing; // used to center sliders added using BorderLayout.CENTER (bug 4275631)
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
centerSpacing = thumbRect.height;
if ( component.getPaintTicks () )
{
centerSpacing += getTickLength ();
}
if ( component.getPaintLabels () )
{
centerSpacing += getHeightOfTallestLabel ();
}
trackRect.x = contentRect.x + trackBuffer;
trackRect.y = contentRect.y + ( contentRect.height - centerSpacing - 1 ) / 2;
trackRect.width = contentRect.width - trackBuffer * 2;
trackRect.height = thumbRect.height;
}
else
{
centerSpacing = thumbRect.width;
if ( ltr )
{
if ( component.getPaintTicks () )
{
centerSpacing += getTickLength ();
}
if ( component.getPaintLabels () )
{
centerSpacing += getWidthOfWidestLabel ();
}
}
else
{
if ( component.getPaintTicks () )
{
centerSpacing -= getTickLength ();
}
if ( component.getPaintLabels () )
{
centerSpacing -= getWidthOfWidestLabel ();
}
}
trackRect.x = contentRect.x + ( contentRect.width - centerSpacing - 1 ) / 2;
trackRect.y = contentRect.y + trackBuffer;
trackRect.width = thumbRect.width;
trackRect.height = contentRect.height - trackBuffer * 2;
}
}
/**
* Gets the height of the tick area for horizontal sliders and the width of the
* tick area for vertical sliders. BasicSliderUI uses the returned value to
* determine the tick area rectangle. If you want to give your ticks some room,
* make this larger than you need and paint your ticks away from the sides in paintTicks().
*/
protected int getTickLength ()
{
return 8;
}
protected void calculateTickRect ()
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
tickRect.x = trackRect.x;
tickRect.y = trackRect.y + trackRect.height;
tickRect.width = trackRect.width;
tickRect.height = component.getPaintTicks () ? getTickLength () : 0;
}
else
{
tickRect.width = component.getPaintTicks () ? getTickLength () : 0;
if ( ltr )
{
tickRect.x = trackRect.x + trackRect.width;
}
else
{
tickRect.x = trackRect.x - tickRect.width;
}
tickRect.y = trackRect.y;
tickRect.height = trackRect.height;
}
}
protected void calculateLabelRect ()
{
if ( component.getPaintLabels () )
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
labelRect.x = tickRect.x - trackBuffer;
labelRect.y = tickRect.y + tickRect.height;
labelRect.width = tickRect.width + trackBuffer * 2;
labelRect.height = getHeightOfTallestLabel ();
}
else
{
if ( ltr )
{
labelRect.x = tickRect.x + tickRect.width;
labelRect.width = getWidthOfWidestLabel ();
}
else
{
labelRect.width = getWidthOfWidestLabel ();
labelRect.x = tickRect.x - labelRect.width;
}
labelRect.y = tickRect.y - trackBuffer;
labelRect.height = tickRect.height + trackBuffer * 2;
}
}
else
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
labelRect.x = tickRect.x;
labelRect.y = tickRect.y + tickRect.height;
labelRect.width = tickRect.width;
labelRect.height = 0;
}
else
{
if ( ltr )
{
labelRect.x = tickRect.x + tickRect.width;
}
else
{
labelRect.x = tickRect.x;
}
labelRect.y = tickRect.y;
labelRect.width = 0;
labelRect.height = tickRect.height;
}
}
}
protected int xPositionForValue ( final int value )
{
final int min = component.getMinimum ();
final int max = component.getMaximum ();
final int trackLength = trackRect.width;
final double valueRange = ( double ) max - ( double ) min;
final double pixelsPerValue = ( double ) trackLength / valueRange;
final int trackLeft = trackRect.x;
final int trackRight = trackRect.x + trackRect.width - 1;
int xPosition;
if ( !drawInverted () )
{
xPosition = trackLeft;
xPosition += Math.round ( pixelsPerValue * ( ( double ) value - min ) );
}
else
{
xPosition = trackRight;
xPosition -= Math.round ( pixelsPerValue * ( ( double ) value - min ) );
}
xPosition = Math.max ( trackLeft, xPosition );
xPosition = Math.min ( trackRight, xPosition );
return xPosition;
}
protected int yPositionForValue ( final int value )
{
return yPositionForValue ( value, trackRect.y, trackRect.height );
}
/**
* Returns the y location for the specified value.
* No checking is done on the arguments.
* In particular if {@code trackHeight} is negative undefined results may occur.
*
* @param value the slider value to get the location for
* @param trackY y-origin of the track
* @param trackHeight the height of the track
*/
protected int yPositionForValue ( final int value, final int trackY, final int trackHeight )
{
final int min = component.getMinimum ();
final int max = component.getMaximum ();
final double valueRange = ( double ) max - ( double ) min;
final double pixelsPerValue = ( double ) trackHeight / valueRange;
final int trackBottom = trackY + trackHeight - 1;
int yPosition;
if ( !drawInverted () )
{
yPosition = trackY;
yPosition += Math.round ( pixelsPerValue * ( ( double ) max - value ) );
}
else
{
yPosition = trackY;
yPosition += Math.round ( pixelsPerValue * ( ( double ) value - min ) );
}
yPosition = Math.max ( trackY, yPosition );
yPosition = Math.min ( trackBottom, yPosition );
return yPosition;
}
protected boolean drawInverted ()
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
if ( ltr )
{
return component.getInverted ();
}
else
{
return !component.getInverted ();
}
}
else
{
return component.getInverted ();
}
}
/**
* Returns the biggest value that has an entry in the label table.
*
* @return biggest value that has an entry in the label table, or
* null.
*/
protected Integer getHighestValue ()
{
final Dictionary dictionary = component.getLabelTable ();
if ( dictionary == null )
{
return null;
}
final Enumeration keys = dictionary.keys ();
Integer max = null;
while ( keys.hasMoreElements () )
{
final Integer i = ( Integer ) keys.nextElement ();
if ( max == null || i > max )
{
max = i;
}
}
return max;
}
/**
* Returns the smallest value that has an entry in the label table.
*
* @return smallest value that has an entry in the label table, or
* null.
*/
protected Integer getLowestValue ()
{
final Dictionary dictionary = component.getLabelTable ();
if ( dictionary == null )
{
return null;
}
final Enumeration keys = dictionary.keys ();
Integer min = null;
while ( keys.hasMoreElements () )
{
final Integer i = ( Integer ) keys.nextElement ();
if ( min == null || i < min )
{
min = i;
}
}
return min;
}
/**
* Returns the label that corresponds to the highest slider value in the label table.
*
* @see javax.swing.JSlider#setLabelTable
*/
protected Component getLowestValueLabel ()
{
final Integer min = getLowestValue ();
if ( min != null )
{
return ( Component ) component.getLabelTable ().get ( min );
}
return null;
}
/**
* Returns the label that corresponds to the lowest slider value in the label table.
*
* @see javax.swing.JSlider#setLabelTable
*/
protected Component getHighestValueLabel ()
{
final Integer max = getHighestValue ();
if ( max != null )
{
return ( Component ) component.getLabelTable ().get ( max );
}
return null;
}
protected int getWidthOfWidestLabel ()
{
final Dictionary dictionary = component.getLabelTable ();
int widest = 0;
if ( dictionary != null )
{
final Enumeration keys = dictionary.keys ();
while ( keys.hasMoreElements () )
{
final Component label = ( Component ) dictionary.get ( keys.nextElement () );
widest = Math.max ( label.getPreferredSize ().width, widest );
}
}
return widest;
}
protected int getHeightOfTallestLabel ()
{
final Dictionary dictionary = component.getLabelTable ();
int tallest = 0;
if ( dictionary != null )
{
final Enumeration keys = dictionary.keys ();
while ( keys.hasMoreElements () )
{
final Component label = ( Component ) dictionary.get ( keys.nextElement () );
tallest = Math.max ( label.getPreferredSize ().height, tallest );
}
}
return tallest;
}
protected Dimension getThumbSize ()
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
return new Dimension ( thumbWidth, thumbHeight );
}
else
{
return new Dimension ( thumbHeight, thumbWidth );
}
}
public void paintThumb ( final Graphics2D g2d )
{
if ( drawThumb )
{
final Object aa = GraphicsUtils.setupAntialias ( g2d );
// Thumb shape
final Shape ts = getThumbShape ();
// // Thumb shade
// if ( slider.isEnabled () )
// {
// LafUtils.drawShade ( g2d, ts, StyleConstants.shadeColor, thumbShadeWidth );
// }
// Thumb background
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
g2d.setPaint ( new GradientPaint ( 0, thumbRect.y, thumbBgTop, 0, thumbRect.y + thumbRect.height, thumbBgBottom ) );
}
else
{
g2d.setPaint ( new GradientPaint ( thumbRect.x, 0, thumbBgTop, thumbRect.x + thumbRect.width, 0, thumbBgBottom ) );
}
g2d.fill ( ts );
// Thumb border
g2d.setPaint ( component.isEnabled () ? Color.GRAY : Color.LIGHT_GRAY );
g2d.draw ( ts );
// Thumb focus
// LafUtils.drawCustomWebFocus ( g2d, slider, FocusType.fieldFocus, ts );
GraphicsUtils.restoreAntialias ( g2d, aa );
}
}
protected Shape getThumbShape ()
{
if ( angledThumb && ( component.getPaintLabels () || component.getPaintTicks () ) )
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
final GeneralPath gp = new GeneralPath ( GeneralPath.WIND_EVEN_ODD );
gp.moveTo ( thumbRect.x, thumbRect.y + thumbRound );
gp.quadTo ( thumbRect.x, thumbRect.y, thumbRect.x + thumbRound, thumbRect.y );
gp.lineTo ( thumbRect.x + thumbRect.width - thumbRound, thumbRect.y );
gp.quadTo ( thumbRect.x + thumbRect.width, thumbRect.y, thumbRect.x + thumbRect.width, thumbRect.y + thumbRound );
gp.lineTo ( thumbRect.x + thumbRect.width, thumbRect.y + thumbRect.height - thumbAngleLength );
if ( sharpThumbAngle )
{
gp.lineTo ( thumbRect.x + thumbRect.width / 2, thumbRect.y + thumbRect.height );
gp.lineTo ( thumbRect.x, thumbRect.y + thumbRect.height - thumbAngleLength );
}
else
{
gp.quadTo ( thumbRect.x + thumbRect.width, thumbRect.y + thumbRect.height - thumbAngleLength / 2,
thumbRect.x + thumbRect.width / 2, thumbRect.y + thumbRect.height );
gp.quadTo ( thumbRect.x, thumbRect.y + thumbRect.height - thumbAngleLength / 2, thumbRect.x,
thumbRect.y + thumbRect.height - thumbAngleLength );
}
gp.closePath ();
return gp;
}
else
{
final GeneralPath gp = new GeneralPath ( GeneralPath.WIND_EVEN_ODD );
if ( ltr )
{
gp.moveTo ( thumbRect.x, thumbRect.y + thumbRound );
gp.quadTo ( thumbRect.x, thumbRect.y, thumbRect.x + thumbRound, thumbRect.y );
gp.lineTo ( thumbRect.x + thumbRect.width - thumbAngleLength, thumbRect.y );
if ( sharpThumbAngle )
{
gp.lineTo ( thumbRect.x + thumbRect.width, thumbRect.y + thumbRect.height / 2 );
gp.lineTo ( thumbRect.x + thumbRect.width - thumbAngleLength, thumbRect.y + thumbRect.height );
}
else
{
gp.quadTo ( thumbRect.x + thumbRect.width - thumbAngleLength / 2, thumbRect.y, thumbRect.x + thumbRect.width,
thumbRect.y + thumbRect.height / 2 );
gp.quadTo ( thumbRect.x + thumbRect.width - thumbAngleLength / 2, thumbRect.y + thumbRect.height,
thumbRect.x + thumbRect.width - thumbAngleLength, thumbRect.y + thumbRect.height );
}
gp.lineTo ( thumbRect.x + thumbRound, thumbRect.y + thumbRect.height );
gp.quadTo ( thumbRect.x, thumbRect.y + thumbRect.height, thumbRect.x, thumbRect.y + thumbRect.height - thumbRound );
}
else
{
gp.moveTo ( thumbRect.x + thumbRect.width - thumbRound, thumbRect.y );
gp.quadTo ( thumbRect.x + thumbRect.width, thumbRect.y, thumbRect.x + thumbRect.width, thumbRect.y + thumbRound );
gp.lineTo ( thumbRect.x + thumbRect.width, thumbRect.y + thumbRect.height - thumbRound );
gp.quadTo ( thumbRect.x + thumbRect.width, thumbRect.y + thumbRect.height, thumbRect.x + thumbRect.width - thumbRound,
thumbRect.y + thumbRect.height );
gp.lineTo ( thumbRect.x + thumbAngleLength, thumbRect.y + thumbRect.height );
if ( sharpThumbAngle )
{
gp.lineTo ( thumbRect.x, thumbRect.y + thumbRect.height / 2 );
gp.lineTo ( thumbRect.x + thumbAngleLength, thumbRect.y );
}
else
{
gp.quadTo ( thumbRect.x + thumbAngleLength / 2, thumbRect.y + thumbRect.height, thumbRect.x,
thumbRect.y + thumbRect.height / 2 );
gp.quadTo ( thumbRect.x + thumbAngleLength / 2, thumbRect.y, thumbRect.x + thumbAngleLength, thumbRect.y );
}
}
gp.closePath ();
return gp;
}
}
else
{
return new RoundRectangle2D.Double ( thumbRect.x, thumbRect.y, thumbRect.width, thumbRect.height, thumbRound * 2,
thumbRound * 2 );
}
}
public void paintTrack ( final Graphics2D g2d )
{
final Object aa = GraphicsUtils.setupAntialias ( g2d );
// Track shape
final Shape ss = getTrackShape ();
// Track background & shade
{
// Track shade
if ( component.isEnabled () )
{
GraphicsUtils.drawShade ( g2d, ss, component.isFocusOwner () ? new Color ( 85, 142, 239 ) : new Color ( 210, 210, 210 ),
trackShadeWidth );
}
// Track background
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
g2d.setPaint ( new GradientPaint ( 0, trackRect.y, trackBgTop, 0, trackRect.y + trackRect.height, trackBgBottom ) );
}
else
{
g2d.setPaint ( new GradientPaint ( trackRect.x, 0, trackBgTop, trackRect.x + trackRect.width, 0, trackBgBottom ) );
}
g2d.fill ( ss );
}
// Inner progress line
if ( drawProgress )
{
// Progress shape
final Shape ps = getProgressShape ();
// Progress shade
if ( component.isEnabled () )
{
GraphicsUtils.drawShade ( g2d, ps, new Color ( 210, 210, 210 ), progressShadeWidth );
}
// Progress background
final Rectangle bounds = ss.getBounds ();
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
g2d.setPaint ( new GradientPaint ( 0, bounds.y + progressShadeWidth, progressTrackBgTop, 0,
bounds.y + bounds.height - progressShadeWidth, progressTrackBgBottom ) );
}
else
{
g2d.setPaint ( new GradientPaint ( bounds.x + progressShadeWidth, 0, progressTrackBgTop,
bounds.x + bounds.width - progressShadeWidth, 0, progressTrackBgBottom ) );
}
g2d.fill ( ps );
// Progress border
g2d.setPaint ( component.isEnabled () ? progressBorderColor : Color.LIGHT_GRAY );
g2d.draw ( ps );
}
// Track border & focus
{
// Track border
g2d.setPaint ( component.isEnabled () ? rolloverDarkBorderOnly && !paintParameters.dragging ?
getBorderColor () : Color.GRAY : Color.LIGHT_GRAY );
g2d.draw ( ss );
}
GraphicsUtils.restoreAntialias ( g2d, aa );
}
protected Shape getTrackShape ()
{
if ( trackRound > 0 )
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
return new RoundRectangle2D.Double ( trackRect.x - trackRound, trackRect.y + trackRect.height / 2 - trackHeight / 2,
trackRect.width + trackRound * 2 - 1, trackHeight, trackRound * 2, trackRound * 2 );
}
else
{
return new RoundRectangle2D.Double ( trackRect.x + trackRect.width / 2 - trackHeight / 2, trackRect.y - trackRound,
trackHeight, trackRect.height + trackRound * 2 - 1, trackRound * 2, trackRound * 2 );
}
}
else
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
return new Rectangle2D.Double ( trackRect.x, trackRect.y + trackRect.height / 2 - trackHeight / 2, trackRect.width - 1,
trackHeight );
}
else
{
return new Rectangle2D.Double ( trackRect.x + trackRect.width / 2 - trackHeight / 2, trackRect.y, trackHeight,
trackRect.height - 1 );
}
}
}
protected Shape getProgressShape ()
{
if ( trackRound > 0 )
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
final int x;
final int w;
if ( ltr )
{
x = trackRect.x - trackRound + progressShadeWidth;
w = thumbRect.x + thumbRect.width / 2 + progressRound - x;
}
else
{
x = thumbRect.x + thumbRect.width / 2 - progressRound;
w = trackRect.x + trackRect.width + trackRound -
progressShadeWidth - 1 - x;
}
return new RoundRectangle2D.Double ( x, trackRect.y + trackRect.height / 2 - trackHeight / 2 + progressShadeWidth, w,
trackHeight - progressShadeWidth * 2, progressRound * 2, progressRound * 2 );
}
else
{
final int y = thumbRect.y + thumbRect.height / 2;
final int h = trackRect.y + trackRect.height + trackRound - progressShadeWidth - y - 1;
return new RoundRectangle2D.Double ( trackRect.x + progressShadeWidth + trackRect.width / 2 - trackHeight / 2, y,
trackHeight - progressShadeWidth * 2, h, progressRound * 2, progressRound * 2 );
}
}
else
{
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
final int x;
final int w;
if ( ltr )
{
x = trackRect.x + progressShadeWidth;
w = thumbRect.x + thumbRect.width / 2 - x;
}
else
{
x = thumbRect.x + thumbRect.width / 2;
w = trackRect.x + trackRect.width - progressShadeWidth - 1 - x;
}
return new Rectangle2D.Double ( x, trackRect.y + trackRect.height / 2 - trackHeight / 2 + progressShadeWidth, w,
trackHeight - progressShadeWidth * 2 );
}
else
{
final int y = trackRect.y + progressShadeWidth;
final int h = thumbRect.y + thumbRect.height / 2 - y;
return new Rectangle2D.Double ( trackRect.x + progressShadeWidth + trackRect.width / 2 - trackHeight / 2, y,
trackHeight - progressShadeWidth * 2, h );
}
}
}
public void paintTicks ( final Graphics g )
{
final Rectangle tickBounds = tickRect;
g.setColor ( component.isEnabled () ? Color.BLACK : Color.LIGHT_GRAY );
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
g.translate ( 0, tickBounds.y );
int value = component.getMinimum ();
int xPos;
if ( component.getMinorTickSpacing () > 0 )
{
while ( value <= component.getMaximum () )
{
xPos = xPositionForValue ( value );
paintMinorTickForHorizontalSlider ( g, tickBounds, xPos );
value += component.getMinorTickSpacing ();
}
}
if ( component.getMajorTickSpacing () > 0 )
{
value = component.getMinimum ();
while ( value <= component.getMaximum () )
{
xPos = xPositionForValue ( value );
paintMajorTickForHorizontalSlider ( g, tickBounds, xPos );
value += component.getMajorTickSpacing ();
}
}
g.translate ( 0, -tickBounds.y );
}
else
{
g.translate ( tickBounds.x, 0 );
int value = component.getMinimum ();
int yPos;
if ( component.getMinorTickSpacing () > 0 )
{
int offset = 0;
if ( !ltr )
{
offset = tickBounds.width - tickBounds.width / 2;
g.translate ( offset, 0 );
}
while ( value <= component.getMaximum () )
{
yPos = yPositionForValue ( value );
paintMinorTickForVerticalSlider ( g, tickBounds, yPos );
value += component.getMinorTickSpacing ();
}
if ( !ltr )
{
g.translate ( -offset, 0 );
}
}
if ( component.getMajorTickSpacing () > 0 )
{
value = component.getMinimum ();
if ( !ltr )
{
g.translate ( 2, 0 );
}
while ( value <= component.getMaximum () )
{
yPos = yPositionForValue ( value );
paintMajorTickForVerticalSlider ( g, tickBounds, yPos );
value += component.getMajorTickSpacing ();
}
if ( !ltr )
{
g.translate ( -2, 0 );
}
}
g.translate ( -tickBounds.x, 0 );
}
}
protected void paintMinorTickForHorizontalSlider ( final Graphics g, final Rectangle tickBounds, final int x )
{
g.drawLine ( x, 0, x, tickBounds.height / 2 - 1 );
}
protected void paintMajorTickForHorizontalSlider ( final Graphics g, final Rectangle tickBounds, final int x )
{
g.drawLine ( x, 0, x, tickBounds.height - 2 );
}
protected void paintMinorTickForVerticalSlider ( final Graphics g, final Rectangle tickBounds, final int y )
{
g.drawLine ( 0, y, tickBounds.width / 2 - 1, y );
}
protected void paintMajorTickForVerticalSlider ( final Graphics g, final Rectangle tickBounds, final int y )
{
g.drawLine ( 0, y, tickBounds.width - 2, y );
}
public void paintLabels ( final Graphics g )
{
final Rectangle labelBounds = labelRect;
final Dictionary dictionary = component.getLabelTable ();
if ( dictionary != null )
{
final Enumeration keys = dictionary.keys ();
final int minValue = component.getMinimum ();
final int maxValue = component.getMaximum ();
final boolean enabled = component.isEnabled ();
while ( keys.hasMoreElements () )
{
final Integer key = ( Integer ) keys.nextElement ();
final int value = key;
if ( value >= minValue && value <= maxValue )
{
final Component label = ( Component ) dictionary.get ( key );
if ( label instanceof JComponent )
{
label.setEnabled ( enabled );
}
if ( component.getOrientation () == JSlider.HORIZONTAL )
{
g.translate ( 0, labelBounds.y );
paintHorizontalLabel ( g, value, label );
g.translate ( 0, -labelBounds.y );
}
else
{
int offset = 0;
if ( !ltr )
{
offset = labelBounds.width - label.getPreferredSize ().width;
}
g.translate ( labelBounds.x + offset, 0 );
paintVerticalLabel ( g, value, label );
g.translate ( -labelBounds.x - offset, 0 );
}
}
}
}
}
/**
* Called for every label in the label table. Used to draw the labels for horizontal sliders.
* The graphics have been translated to labelRect.y already.
*
* @see JSlider#setLabelTable
*/
protected void paintHorizontalLabel ( final Graphics g, final int value, final Component label )
{
final int labelCenter = xPositionForValue ( value );
final int labelLeft = labelCenter - label.getPreferredSize ().width / 2;
g.translate ( labelLeft, 0 );
label.paint ( g );
g.translate ( -labelLeft, 0 );
}
/**
* Called for every label in the label table. Used to draw the labels for vertical sliders.
* The graphics have been translated to labelRect.x already.
*
* @see JSlider#setLabelTable
*/
protected void paintVerticalLabel ( final Graphics g, final int value, final Component label )
{
final int labelCenter = yPositionForValue ( value );
final int labelTop = labelCenter - label.getPreferredSize ().height / 2;
g.translate ( 0, labelTop );
label.paint ( g );
g.translate ( 0, -labelTop );
}
protected float getProgress ()
{
return ( float ) rolloverDarkness / MAX_DARKNESS;
}
protected Color getBorderColor ()
{
return ColorUtils.intermediate ( new Color ( 170, 170, 170 ), Color.GRAY, getProgress () );
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy