/*
* Copyright (c) 2005-2016 Substance Kirill Grouchnikov. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* o Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* o Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* o Neither the name of Substance Kirill Grouchnikov nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.pushingpixels.substance.internal.ui;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.FontMetrics;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Insets;
import java.awt.Rectangle;
import java.awt.event.MouseEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import javax.swing.AbstractButton;
import javax.swing.Icon;
import javax.swing.JCheckBox;
import javax.swing.JComponent;
import javax.swing.JRadioButton;
import javax.swing.JToggleButton;
import javax.swing.LookAndFeel;
import javax.swing.SwingUtilities;
import javax.swing.border.Border;
import javax.swing.plaf.ComponentUI;
import javax.swing.plaf.basic.BasicButtonListener;
import javax.swing.plaf.basic.BasicHTML;
import javax.swing.plaf.basic.BasicToggleButtonUI;
import javax.swing.text.View;
import org.pushingpixels.lafwidget.LafWidgetUtilities;
import org.pushingpixels.lafwidget.animation.AnimationConfigurationManager;
import org.pushingpixels.lafwidget.animation.AnimationFacet;
import org.pushingpixels.lafwidget.animation.effects.GhostPaintingUtils;
import org.pushingpixels.lafwidget.animation.effects.GhostingListener;
import org.pushingpixels.lafwidget.utils.RenderingUtils;
import org.pushingpixels.substance.api.SubstanceLookAndFeel;
import org.pushingpixels.substance.api.shaper.SubstanceButtonShaper;
import org.pushingpixels.substance.internal.animation.StateTransitionTracker;
import org.pushingpixels.substance.internal.animation.TransitionAwareUI;
import org.pushingpixels.substance.internal.utils.ButtonBackgroundDelegate;
import org.pushingpixels.substance.internal.utils.ButtonVisualStateTracker;
import org.pushingpixels.substance.internal.utils.SubstanceCoreUtilities;
import org.pushingpixels.substance.internal.utils.SubstanceSizeUtils;
import org.pushingpixels.substance.internal.utils.SubstanceTextUtilities;
import org.pushingpixels.substance.internal.utils.border.SubstanceButtonBorder;
import org.pushingpixels.substance.internal.utils.icon.GlowingIcon;
/**
* UI for toggle buttons in <b>Substance</b> look and feel.
*
* @author Kirill Grouchnikov
*/
public class SubstanceToggleButtonUI extends BasicToggleButtonUI implements
TransitionAwareUI {
/**
* Painting delegate.
*/
private ButtonBackgroundDelegate delegate;
/**
* The matching glowing icon. Is used only when
* {@link AnimationConfigurationManager#isAnimationAllowed(AnimationFacet, Component)}
* returns true on {@link AnimationFacet#ICON_GLOW}.
*/
protected GlowingIcon glowingIcon;
/**
* Property change listener. Listens on changes to the
* {@link SubstanceLookAndFeel#BUTTON_SHAPER_PROPERTY} property and
* {@link AbstractButton#MODEL_CHANGED_PROPERTY} property.
*/
protected PropertyChangeListener substancePropertyListener;
/**
* Tracker for visual state transitions.
*/
protected ButtonVisualStateTracker substanceVisualStateTracker;
/**
* Model change listener for ghost image effects.
*/
private GhostingListener ghostModelChangeListener;
protected JToggleButton toggleButton;
private Rectangle viewRect = new Rectangle();
private Rectangle iconRect = new Rectangle();
private Rectangle textRect = new Rectangle();
/*
* (non-Javadoc)
*
* @see javax.swing.plaf.ComponentUI#createUI(javax.swing.JComponent)
*/
public static ComponentUI createUI(JComponent comp) {
SubstanceCoreUtilities.testComponentCreationThreadingViolation(comp);
return new SubstanceToggleButtonUI((JToggleButton) comp);
}
/**
* Simple constructor.
*/
public SubstanceToggleButtonUI(JToggleButton toggleButton) {
this.toggleButton = toggleButton;
this.delegate = new ButtonBackgroundDelegate();
}
/*
* (non-Javadoc)
*
* @seejavax.swing.plaf.basic.BasicButtonUI#installDefaults(javax.swing.
* AbstractButton)
*/
@Override
public void installDefaults(AbstractButton b) {
super.installDefaults(b);
if (b.getClientProperty(SubstanceButtonUI.BORDER_ORIGINAL) == null)
b.putClientProperty(SubstanceButtonUI.BORDER_ORIGINAL, b
.getBorder());
if (b.getClientProperty(SubstanceButtonUI.BORDER_ORIGINAL) == null)
b.putClientProperty(SubstanceButtonUI.BORDER_ORIGINAL, b
.getBorder());
SubstanceButtonShaper shaper = SubstanceCoreUtilities
.getButtonShaper(b);
if (b.getClientProperty(SubstanceButtonUI.BORDER_COMPUTED) == null) {
b.setBorder(shaper.getButtonBorder(b));
} else {
Border currBorder = b.getBorder();
if (!(currBorder instanceof SubstanceButtonBorder)) {
b.setBorder(shaper.getButtonBorder(b));
} else {
SubstanceButtonBorder sbCurrBorder = (SubstanceButtonBorder) currBorder;
if (shaper.getClass() != sbCurrBorder.getButtonShaperClass())
b.setBorder(shaper.getButtonBorder(b));
}
}
b.putClientProperty(SubstanceButtonUI.OPACITY_ORIGINAL, b.isOpaque());
// fix for defect 140
b.setOpaque(false);
b.setRolloverEnabled(true);
LookAndFeel.installProperty(b, "iconTextGap", SubstanceSizeUtils
.getTextIconGap(SubstanceSizeUtils.getComponentFontSize(b)));
}
/*
* (non-Javadoc)
*
* @seejavax.swing.plaf.basic.BasicButtonUI#uninstallDefaults(javax.swing.
* AbstractButton)
*/
@Override
public void uninstallDefaults(AbstractButton b) {
super.uninstallDefaults(b);
b.setBorder((Border) b
.getClientProperty(SubstanceButtonUI.BORDER_ORIGINAL));
b.setOpaque((Boolean) b
.getClientProperty(SubstanceButtonUI.OPACITY_ORIGINAL));
b.putClientProperty(SubstanceButtonUI.OPACITY_ORIGINAL, null);
}
/*
* (non-Javadoc)
*
* @see
* javax.swing.plaf.basic.BasicButtonUI#createButtonListener(javax.swing
* .AbstractButton)
*/
@Override
protected BasicButtonListener createButtonListener(AbstractButton b) {
return null;
}
/*
* (non-Javadoc)
*
* @seejavax.swing.plaf.basic.BasicButtonUI#installListeners(javax.swing.
* AbstractButton)
*/
@Override
protected void installListeners(final AbstractButton b) {
super.installListeners(b);
this.substanceVisualStateTracker = new ButtonVisualStateTracker();
this.substanceVisualStateTracker.installListeners(b, true);
this.trackGlowingIcon();
this.substancePropertyListener = (PropertyChangeEvent evt) -> {
if (AbstractButton.ICON_CHANGED_PROPERTY.equals(evt
.getPropertyName())) {
trackGlowingIcon();
}
if (AbstractButton.MODEL_CHANGED_PROPERTY.equals(evt.getPropertyName())) {
if (ghostModelChangeListener != null)
ghostModelChangeListener.unregisterListeners();
ghostModelChangeListener = new GhostingListener(b, b
.getModel());
ghostModelChangeListener.registerListeners();
}
};
b.addPropertyChangeListener(this.substancePropertyListener);
this.ghostModelChangeListener = new GhostingListener(b, b.getModel());
this.ghostModelChangeListener.registerListeners();
}
/*
* (non-Javadoc)
*
* @seejavax.swing.plaf.basic.BasicButtonUI#uninstallListeners(javax.swing.
* AbstractButton)
*/
@Override
protected void uninstallListeners(AbstractButton b) {
this.substanceVisualStateTracker.uninstallListeners(b);
this.substanceVisualStateTracker = null;
b.removePropertyChangeListener(this.substancePropertyListener);
this.substancePropertyListener = null;
this.ghostModelChangeListener.unregisterListeners();
this.ghostModelChangeListener = null;
super.uninstallListeners(b);
}
/*
* (non-Javadoc)
*
* @see javax.swing.plaf.basic.BasicToggleButtonUI#paint(java.awt.Graphics,
* javax.swing.JComponent)
*/
@Override
public void paint(Graphics g, JComponent c) {
final AbstractButton b = (AbstractButton) c;
FontMetrics fm = g.getFontMetrics();
Insets i = c.getInsets();
viewRect.x = i.left;
viewRect.y = i.top;
viewRect.width = b.getWidth() - (i.right + viewRect.x);
viewRect.height = b.getHeight() - (i.bottom + viewRect.y);
textRect.x = textRect.y = textRect.width = textRect.height = 0;
iconRect.x = iconRect.y = iconRect.width = iconRect.height = 0;
Font f = c.getFont();
// layout the text and icon
String text = SwingUtilities.layoutCompoundLabel(c, fm, b.getText(), b
.getIcon(), b.getVerticalAlignment(), b
.getHorizontalAlignment(), b.getVerticalTextPosition(), b
.getHorizontalTextPosition(), viewRect, iconRect, textRect, b
.getText() == null ? 0 : b.getIconTextGap());
Graphics2D g2d = (Graphics2D) g.create();
View v = (View) c.getClientProperty(BasicHTML.propertyKey);
g2d.setFont(f);
this.delegate.updateBackground(g2d, b);
if (v != null) {
v.paint(g2d, textRect);
} else {
this.paintButtonText(g2d, b, textRect, text);
}
// Paint the Icon
if (b.getIcon() != null) {
paintIcon(g2d, b, iconRect);
}
if (b.isFocusPainted()) {
SubstanceCoreUtilities.paintFocus(g, b, b, this, null, textRect,
1.0f, SubstanceSizeUtils
.getFocusRingPadding(SubstanceSizeUtils
.getComponentFontSize(b)));
}
}
/*
* (non-Javadoc)
*
* @see
* javax.swing.plaf.ComponentUI#getPreferredSize(javax.swing.JComponent)
*/
@Override
public Dimension getPreferredSize(JComponent c) {
AbstractButton button = (AbstractButton) c;
SubstanceButtonShaper shaper = SubstanceCoreUtilities
.getButtonShaper(button);
// fix for defect 263
Dimension superPref = super.getPreferredSize(button);
if (superPref == null)
return null;
if (shaper == null)
return superPref;
return shaper.getPreferredSize(button, superPref);
}
/*
* (non-Javadoc)
*
* @see javax.swing.plaf.ComponentUI#contains(javax.swing.JComponent, int,
* int)
*/
@Override
public boolean contains(JComponent c, int x, int y) {
return ButtonBackgroundDelegate.contains((JToggleButton) c, x, y);
}
/*
* (non-Javadoc)
*
* @see
* javax.swing.plaf.basic.BasicToggleButtonUI#paintIcon(java.awt.Graphics,
* javax.swing.AbstractButton, java.awt.Rectangle)
*/
@Override
protected void paintIcon(Graphics g, AbstractButton b, Rectangle iconRect) {
b.putClientProperty("icon.bounds", new Rectangle(iconRect));
Graphics2D graphics = (Graphics2D) g.create();
GhostPaintingUtils.paintGhostIcon(graphics, b, iconRect);
// We have three types of icons:
// 1. The original button icon
// 2. The themed version of 1.
// 3. The glowing version of 1.
Icon originalIcon = SubstanceCoreUtilities.getOriginalIcon(b, b
.getIcon());
Icon themedIcon = (!(b instanceof JRadioButton)
&& !(b instanceof JCheckBox) && SubstanceCoreUtilities
.useThemedDefaultIcon(b)) ? SubstanceCoreUtilities
.getThemedIcon(b, originalIcon) : originalIcon;
graphics.setComposite(LafWidgetUtilities.getAlphaComposite(b, g));
float activeAmount = this.substanceVisualStateTracker
.getStateTransitionTracker().getActiveStrength();
graphics.translate(iconRect.x, iconRect.y);
if (activeAmount >= 0.0f) {
if (AnimationConfigurationManager.getInstance().isAnimationAllowed(
AnimationFacet.ICON_GLOW, b)
&& this.substanceVisualStateTracker
.getStateTransitionTracker().getIconGlowTracker()
.isPlaying()) {
this.glowingIcon.paintIcon(b, graphics, 0, 0);
} else {
themedIcon.paintIcon(b, graphics, 0, 0);
graphics.setComposite(LafWidgetUtilities.getAlphaComposite(b,
activeAmount, g));
originalIcon.paintIcon(b, graphics, 0, 0);
}
} else {
originalIcon.paintIcon(b, graphics, 0, 0);
}
graphics.dispose();
}
/**
* Paints the text.
*
* @param g
* Graphic context
* @param button
* Button
* @param textRect
* Text rectangle
* @param text
* Text to paint
*/
protected void paintButtonText(Graphics g, AbstractButton button,
Rectangle textRect, String text) {
SubstanceTextUtilities.paintText(g, button, textRect, text, (button)
.getDisplayedMnemonicIndex());
}
/**
* Tracks possible usage of glowing icon.
*
* @param b
* Button.
*/
protected void trackGlowingIcon() {
Icon currIcon = this.toggleButton.getIcon();
if (currIcon instanceof GlowingIcon)
return;
if (currIcon == null)
return;
this.glowingIcon = new GlowingIcon(currIcon,
this.substanceVisualStateTracker.getStateTransitionTracker()
.getIconGlowTracker());
}
/*
* (non-Javadoc)
*
* @see javax.swing.plaf.ComponentUI#update(java.awt.Graphics,
* javax.swing.JComponent)
*/
@Override
public void update(Graphics g, JComponent c) {
Graphics2D g2d = (Graphics2D) g.create();
RenderingUtils.installDesktopHints(g2d, c);
this.paint(g2d, c);
g2d.dispose();
}
@Override
public boolean isInside(MouseEvent me) {
return this.contains(this.toggleButton, me.getX(), me.getY());
}
@Override
public StateTransitionTracker getTransitionTracker() {
return this.substanceVisualStateTracker.getStateTransitionTracker();
}
}