/*************************************************** * * cismet GmbH, Saarbruecken, Germany * * ... and it just works. * ****************************************************/ package de.cismet.tools.gui; import java.awt.Color; import java.awt.Graphics; import java.awt.Rectangle; import java.awt.event.MouseEvent; import javax.swing.JComponent; import javax.swing.JSlider; import javax.swing.SwingUtilities; import javax.swing.event.ChangeEvent; import javax.swing.event.ChangeListener; import javax.swing.plaf.metal.MetalSliderUI; /** * UI delegate for the RangeSlider component with Metal LaF. It paints two thumbs, one for the lower value and one for * the upper value. * * @version $Revision$, $Date$ */ public class MetalRangeSliderUI extends MetalSliderUI { //~ Instance fields -------------------------------------------------------- /** Track color of selected range. */ private Color rangeColor = Color.GREEN; private Rectangle upperThumbRect; private boolean upperThumbSelected; private transient boolean lowerDragging; private transient boolean upperDragging; //~ Methods ---------------------------------------------------------------- /** * Installs this UI delegate on the specified component. * * @param c The component which this UI is installed on. */ @Override public void installUI(final JComponent c) { upperThumbRect = new Rectangle(); super.installUI(c); rangeColor = darkShadowColor; } /** * Creates a listener to handle track events in the specified slider. * * @param slider The slider to handle track events for. * * @return The track listener. */ @Override protected TrackListener createTrackListener(final JSlider slider) { return new MetalRangeSliderUI.RangeTrackListener(); } /** * Creates a listener to handle change events in the specified slider. * * @param slider The slider to handle change events for. * * @return The change listener. */ @Override protected ChangeListener createChangeListener(final JSlider slider) { return new MetalRangeSliderUI.ChangeHandler(); } /** * Updates the dimensions for both thumbs. */ @Override protected void calculateThumbSize() { // Call superclass method for lower thumb size. super.calculateThumbSize(); // Set upper thumb size. upperThumbRect.setSize(thumbRect.width, thumbRect.height); } /** * Updates the locations for both thumbs. */ @Override protected void calculateThumbLocation() { // Call superclass method for lower thumb location. super.calculateThumbLocation(); // Adjust upper value to snap to ticks if necessary. if (slider.getSnapToTicks()) { final int upperValue = slider.getValue() + slider.getExtent(); int snappedValue = upperValue; final int majorTickSpacing = slider.getMajorTickSpacing(); final int minorTickSpacing = slider.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 (((upperValue - slider.getMinimum()) % tickSpacing) != 0) { final float temp = (float)(upperValue - slider.getMinimum()) / (float)tickSpacing; final int whichTick = Math.round(temp); snappedValue = slider.getMinimum() + (whichTick * tickSpacing); } if (snappedValue != upperValue) { slider.setExtent(snappedValue - slider.getValue()); } } } // Calculate upper thumb location. The thumb is centered over its // value on the track. if (slider.getOrientation() == JSlider.HORIZONTAL) { final int upperPosition = xPositionForValue(slider.getValue() + slider.getExtent()); upperThumbRect.x = upperPosition - (upperThumbRect.width / 2); upperThumbRect.y = trackRect.y; } else { final int upperPosition = yPositionForValue(slider.getValue() + slider.getExtent()); upperThumbRect.x = trackRect.x; upperThumbRect.y = upperPosition - (upperThumbRect.height / 2); } } /** * Paints the slider. The selected thumb is always painted on top of the other thumb. * * @param g The graphics to paint on. * @param c Will not be used. */ @Override public void paint(final Graphics g, final JComponent c) { super.paint(g, c); final Rectangle clipRect = g.getClipBounds(); if (upperThumbSelected) { // Paint lower thumb first, then upper thumb. if (clipRect.intersects(thumbRect)) { paintLowerThumb(g); } if (clipRect.intersects(upperThumbRect)) { paintUpperThumb(g); } } else { // Paint upper thumb first, then lower thumb. if (clipRect.intersects(upperThumbRect)) { paintUpperThumb(g); } if (clipRect.intersects(thumbRect)) { paintLowerThumb(g); } } } /** * Paints the track. * * @param g The graphics to paint on. */ @Override public void paintTrack(final Graphics g) { // Draw track. super.paintTrack(g); final Rectangle trackBounds = trackRect; if (slider.getOrientation() == JSlider.HORIZONTAL) { // Determine position of selected range by moving from the middle // of one thumb to the other. final int lowerX = thumbRect.x + (thumbRect.width / 2); final int upperX = upperThumbRect.x + (upperThumbRect.width / 2); // Determine track position. final int cy = (trackBounds.height / 2) - 2; // Save color and shift position. final Color oldColor = g.getColor(); g.translate(trackBounds.x, trackBounds.y + cy); // Draw selected range. g.setColor(rangeColor); for (int y = 0; y <= 3; y++) { g.drawLine(lowerX - trackBounds.x, y, upperX - trackBounds.x, y); } // Restore position and color. g.translate(-trackBounds.x, -(trackBounds.y + cy)); g.setColor(oldColor); } else { // Determine position of selected range by moving from the middle // of one thumb to the other. final int lowerY = thumbRect.x + (thumbRect.width / 2); final int upperY = upperThumbRect.x + (upperThumbRect.width / 2); // Determine track position. final int cx = (trackBounds.width / 2) - 2; // Save color and shift position. final Color oldColor = g.getColor(); g.translate(trackBounds.x + cx, trackBounds.y); // Draw selected range. g.setColor(rangeColor); for (int x = 0; x <= 3; x++) { g.drawLine(x, lowerY - trackBounds.y, x, upperY - trackBounds.y); } // Restore position and color. g.translate(-(trackBounds.x + cx), -trackBounds.y); g.setColor(oldColor); } } /** * Overrides superclass method to do nothing. Thumb painting is handled within the <code>paint()</code> method. * * @param g The graphics to (not) paint on. */ @Override public void paintThumb(final Graphics g) { // Do nothing. } /** * Paints the thumb for the lower value using the specified graphics object. * * @param g The graphics to paint on. */ private void paintLowerThumb(final Graphics g) { super.paintThumb(g); } /** * Paints the thumb for the upper value using the specified graphics object. * * @param g The graphics to paint on. */ private void paintUpperThumb(final Graphics g) { final Rectangle tempRect = super.thumbRect; super.thumbRect = upperThumbRect; super.paintThumb(g); super.thumbRect = tempRect; } /** * Sets the location of the upper thumb, and repaints the slider. This is called when the upper thumb is dragged to * repaint the slider. The <code>setThumbLocation()</code> method performs the same task for the lower thumb. * * @param x The x location of the upper thumb's rectangle. * @param y The y location of the upper thumb's rectangle. */ private void setUpperThumbLocation(final int x, final int y) { final Rectangle upperUnionRect = new Rectangle(); upperUnionRect.setBounds(upperThumbRect); upperThumbRect.setLocation(x, y); SwingUtilities.computeUnion( upperThumbRect.x, upperThumbRect.y, upperThumbRect.width, upperThumbRect.height, upperUnionRect); slider.repaint(upperUnionRect.x, upperUnionRect.y, upperUnionRect.width, upperUnionRect.height); } /** * Moves the selected thumb in the specified direction by a block increment. This method is called when the user * presses the Page Up or Down keys. * * @param direction The direction (> 0 means scroll in positive direction, <= 0 scrolls in negative direction). */ @Override public void scrollByBlock(final int direction) { synchronized (slider) { int blockIncrement = (slider.getMaximum() - slider.getMinimum()) / 10; if ((blockIncrement <= 0) && (slider.getMaximum() > slider.getMinimum())) { blockIncrement = 1; } final int delta = blockIncrement * ((direction > 0) ? POSITIVE_SCROLL : NEGATIVE_SCROLL); if (upperThumbSelected) { final int oldValue = ((RangeSlider)slider).getUpperValue(); ((RangeSlider)slider).setUpperValue(oldValue + delta); } else { final int oldValue = slider.getValue(); slider.setValue(oldValue + delta); } } } /** * Moves the selected thumb in the specified direction by a unit increment. This method is called when the user * presses one of the arrow keys. * * @param direction The direction (> 0 means scroll in positive direction, <= 0 scrolls in negative direction). */ @Override public void scrollByUnit(final int direction) { synchronized (slider) { final int delta = 1 * ((direction > 0) ? POSITIVE_SCROLL : NEGATIVE_SCROLL); if (upperThumbSelected) { final int oldValue = ((RangeSlider)slider).getUpperValue(); ((RangeSlider)slider).setUpperValue(oldValue + delta); } else { final int oldValue = slider.getValue(); slider.setValue(oldValue + delta); } } } //~ Inner Classes ---------------------------------------------------------- /** * Listener to handle model change events. This calculates the thumb locations and repaints the slider if the value * change is not caused by dragging a thumb. * * @version $Revision$, $Date$ */ public class ChangeHandler implements ChangeListener { //~ Methods ------------------------------------------------------------ @Override public void stateChanged(final ChangeEvent arg0) { if (!lowerDragging && !upperDragging) { calculateThumbLocation(); slider.repaint(); } } } /** * Listener to handle mouse movements in the slider track. * * @version $Revision$, $Date$ */ public class RangeTrackListener extends TrackListener { //~ Methods ------------------------------------------------------------ @Override public void mousePressed(final MouseEvent e) { if (!slider.isEnabled()) { return; } currentMouseX = e.getX(); currentMouseY = e.getY(); if (slider.isRequestFocusEnabled()) { slider.requestFocus(); } // Determine which thumb is pressed. If the upper thumb is // selected (last one dragged), then check its position first; // otherwise check the position of the lower thumb first. boolean lowerPressed = false; boolean upperPressed = false; if (upperThumbSelected) { if (upperThumbRect.contains(currentMouseX, currentMouseY)) { upperPressed = true; } else if (thumbRect.contains(currentMouseX, currentMouseY)) { lowerPressed = true; } } else { if (thumbRect.contains(currentMouseX, currentMouseY)) { lowerPressed = true; } else if (upperThumbRect.contains(currentMouseX, currentMouseY)) { upperPressed = true; } } // Handle lower thumb pressed. if (lowerPressed) { switch (slider.getOrientation()) { case JSlider.VERTICAL: { offset = currentMouseY - thumbRect.y; break; } case JSlider.HORIZONTAL: { offset = currentMouseX - thumbRect.x; break; } } upperThumbSelected = false; lowerDragging = true; return; } lowerDragging = false; // Handle upper thumb pressed. if (upperPressed) { switch (slider.getOrientation()) { case JSlider.VERTICAL: { offset = currentMouseY - upperThumbRect.y; break; } case JSlider.HORIZONTAL: { offset = currentMouseX - upperThumbRect.x; break; } } upperThumbSelected = true; upperDragging = true; return; } upperDragging = false; } @Override public void mouseReleased(final MouseEvent e) { lowerDragging = false; upperDragging = false; slider.setValueIsAdjusting(false); super.mouseReleased(e); } @Override public void mouseDragged(final MouseEvent e) { if (!slider.isEnabled()) { return; } currentMouseX = e.getX(); currentMouseY = e.getY(); if (lowerDragging) { slider.setValueIsAdjusting(true); moveLowerThumb(); } else if (upperDragging) { slider.setValueIsAdjusting(true); moveUpperThumb(); } } @Override public boolean shouldScroll(final int direction) { return false; } /** * Moves the location of the lower thumb, and sets its corresponding value in the slider. */ private void moveLowerThumb() { int thumbMiddle; switch (slider.getOrientation()) { case JSlider.VERTICAL: { final int halfThumbHeight = thumbRect.height / 2; int thumbTop = currentMouseY - offset; int trackTop = trackRect.y; int trackBottom = trackRect.y + (trackRect.height - 1); final int vMax = yPositionForValue(slider.getValue() + slider.getExtent()); // Apply bounds to thumb position. if (drawInverted()) { trackBottom = vMax; } else { trackTop = vMax; } thumbTop = Math.max(thumbTop, trackTop - halfThumbHeight); thumbTop = Math.min(thumbTop, trackBottom - halfThumbHeight); setThumbLocation(thumbRect.x, thumbTop); // Update slider value. thumbMiddle = thumbTop + halfThumbHeight; slider.setValue(valueForYPosition(thumbMiddle)); break; } case JSlider.HORIZONTAL: { final int halfThumbWidth = thumbRect.width / 2; int thumbLeft = currentMouseX - offset; int trackLeft = trackRect.x; int trackRight = trackRect.x + (trackRect.width - 1); final int hMax = xPositionForValue(slider.getValue() + slider.getExtent()); // Apply bounds to thumb position. if (drawInverted()) { trackLeft = hMax; } else { trackRight = hMax; } thumbLeft = Math.max(thumbLeft, trackLeft - halfThumbWidth); thumbLeft = Math.min(thumbLeft, trackRight - halfThumbWidth); setThumbLocation(thumbLeft, thumbRect.y); // Update slider value. thumbMiddle = thumbLeft + halfThumbWidth; slider.setValue(valueForXPosition(thumbMiddle)); break; } } } /** * Moves the location of the upper thumb, and sets its corresponding value in the slider. */ private void moveUpperThumb() { int thumbMiddle; switch (slider.getOrientation()) { case JSlider.VERTICAL: { final int halfThumbHeight = thumbRect.height / 2; int thumbTop = currentMouseY - offset; int trackTop = trackRect.y; int trackBottom = trackRect.y + (trackRect.height - 1); final int vMin = yPositionForValue(slider.getValue()); // Apply bounds to thumb position. if (drawInverted()) { trackTop = vMin; } else { trackBottom = vMin; } thumbTop = Math.max(thumbTop, trackTop - halfThumbHeight); thumbTop = Math.min(thumbTop, trackBottom - halfThumbHeight); setUpperThumbLocation(thumbRect.x, thumbTop); // Update slider extent. thumbMiddle = thumbTop + halfThumbHeight; slider.setExtent(valueForYPosition(thumbMiddle) - slider.getValue()); break; } case JSlider.HORIZONTAL: { final int halfThumbWidth = thumbRect.width / 2; int thumbLeft = currentMouseX - offset; int trackLeft = trackRect.x; int trackRight = trackRect.x + (trackRect.width - 1); final int hMin = xPositionForValue(slider.getValue()); // Apply bounds to thumb position. if (drawInverted()) { trackRight = hMin; } else { trackLeft = hMin; } thumbLeft = Math.max(thumbLeft, trackLeft - halfThumbWidth); thumbLeft = Math.min(thumbLeft, trackRight - halfThumbWidth); setUpperThumbLocation(thumbLeft, thumbRect.y); // Update slider extent. thumbMiddle = thumbLeft + halfThumbWidth; slider.setExtent(valueForXPosition(thumbMiddle) - slider.getValue()); break; } } } } }