org.pushingpixels.radiance.theming.internal.utils.WidgetUtilities Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of swingset3-demos Show documentation
Show all versions of swingset3-demos Show documentation
Demonstrating the abilities of the Swing UI Toolkit swingset2 and swingx aka swingset3
The newest version!
/*
* Copyright (c) 2005-2021 Radiance 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 the copyright holder 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.radiance.theming.internal.utils;
import org.pushingpixels.radiance.theming.api.RadianceThemingSlices;
import org.pushingpixels.radiance.theming.api.painter.preview.PreviewPainter;
import org.pushingpixels.radiance.theming.api.password.PasswordStrengthChecker;
import org.pushingpixels.radiance.theming.internal.AnimationConfigurationManager;
import org.pushingpixels.radiance.theming.internal.RadianceSynapse;
import javax.swing.*;
import javax.swing.text.JTextComponent;
import java.awt.*;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.Map;
/**
* Various utility functions.
*
* @author Kirill Grouchnikov
* @author Romain Guy
*/
public class WidgetUtilities {
/**
* Name for the internal client property that marks a component as previewable.
*/
public static final String PREVIEW_MODE = "radiance.theming.internal.widget.previewMode";
/**
* Private constructor. Is here to enforce using static methods only.
*/
private WidgetUtilities() {
}
/**
* Makes the specified component and all its descendants previewable.
*
* @param comp
* Component.
* @param dbSnapshot
* The "snapshot" map that will contain the original double-buffer status of the
* specified component and all its descendants. Key is {@link JComponent}, value is
* {@link Boolean}.
*/
public static void makePreviewable(Component comp, Map dbSnapshot) {
if (comp instanceof JComponent) {
JComponent jcomp = (JComponent) comp;
// if (jcomp.getParent() instanceof CellRendererPane) {
// System.out.println(jcomp.getClass().getSimpleName() + ":"
// + jcomp.hashCode());
// }
dbSnapshot.put(jcomp, Boolean.valueOf(jcomp.isDoubleBuffered()));
jcomp.setDoubleBuffered(false);
jcomp.putClientProperty(WidgetUtilities.PREVIEW_MODE, Boolean.TRUE);
}
if (comp instanceof Container) {
Container cont = (Container) comp;
for (int i = 0; i < cont.getComponentCount(); i++) {
WidgetUtilities.makePreviewable(cont.getComponent(i), dbSnapshot);
}
}
}
/**
* Restores the regular (non-previewable) status of the specified component and all its
* descendants.
*
* @param comp
* Component.
* @param dbSnapshot
* The "snapshot" map that contains the original double-buffer status of the
* specified component and all its descendants. Key is {@link JComponent}, value is
* {@link Boolean}.
*/
public static void restorePreviewable(Component comp, Map dbSnapshot) {
if (comp instanceof JComponent) {
JComponent jcomp = (JComponent) comp;
if (dbSnapshot.containsKey(comp)) {
jcomp.setDoubleBuffered(dbSnapshot.get(comp));
jcomp.putClientProperty(WidgetUtilities.PREVIEW_MODE, null);
} else {
// this can happen in case the application has
// renderers (combos, ...). Take the property from the parent
Component parent = comp.getParent();
if (parent instanceof JComponent) {
jcomp.setDoubleBuffered(dbSnapshot.get(parent));
jcomp.putClientProperty(WidgetUtilities.PREVIEW_MODE, null);
}
// System.out.println("Not found");
// Component c = jcomp;
// while (c != null) {
// System.out.println("\t" + c.getClass().getSimpleName()
// + ":" + c.hashCode());
// c = c.getParent();
// }
}
}
if (comp instanceof Container) {
Container cont = (Container) comp;
for (int i = 0; i < cont.getComponentCount(); i++) {
WidgetUtilities.restorePreviewable(cont.getComponent(i), dbSnapshot);
}
}
}
/**
* Checks whether the specified text component has "select all on focus gain" property.
*
* @param textComp
* Text component.
* @return true
if the specified text component has "select all on focus gain"
* property, false
otherwise.
*/
public static boolean hasTextFocusSelectAllProperty(JTextComponent textComp) {
Component comp = textComp;
while (comp != null) {
if (comp instanceof JComponent) {
Object textFocusSelectAllProperty = ((JComponent) comp)
.getClientProperty(RadianceSynapse.TEXT_SELECT_ON_FOCUS);
if (Boolean.TRUE.equals(textFocusSelectAllProperty)) {
return true;
}
if (Boolean.FALSE.equals(textFocusSelectAllProperty)) {
return false;
}
}
comp = comp.getParent();
}
return (Boolean.TRUE.equals(UIManager.get(RadianceSynapse.TEXT_SELECT_ON_FOCUS)));
}
/**
* Checks whether the specified text component has "flip select on escape" property.
*
* @param textComp
* Text component.
* @return true
if the specified text component has "flip select on escape"
* property, false
otherwise.
*/
public static boolean hasTextFlipSelectOnEscapeProperty(JTextComponent textComp) {
Object textFocusSelectAllProperty = textComp
.getClientProperty(RadianceSynapse.TEXT_FLIP_SELECT_ON_ESCAPE);
return (Boolean.TRUE.equals(textFocusSelectAllProperty));
}
/**
* Checks whether the specified text component has edit context menu property.
*
* @param textComp
* Text component.
* @return true
if the specified text component has edit context menu property,
* false
otherwise.
*/
public static boolean hasTextEditContextMenu(JTextComponent textComp) {
Object textEditContextMenuProperty = textComp
.getClientProperty(RadianceSynapse.TEXT_EDIT_CONTEXT_MENU);
if (Boolean.TRUE.equals(textEditContextMenuProperty)) {
return true;
}
if (Boolean.FALSE.equals(textEditContextMenuProperty)) {
return false;
}
return (Boolean.TRUE.equals(UIManager.get(RadianceSynapse.TEXT_EDIT_CONTEXT_MENU)));
}
/**
* Checks whether the specified scroll pane supports auto scroll.
*
* @param scrollPane
* Scroll pane component.
* @return true
if the specified scroll pane supports auto scroll,
* false
otherwise.
*/
public static boolean hasAutoScroll(JScrollPane scrollPane) {
Object compProperty = scrollPane.getClientProperty(RadianceSynapse.AUTO_SCROLL);
if (Boolean.TRUE.equals(compProperty)) {
return true;
}
if (Boolean.FALSE.equals(compProperty)) {
return false;
}
return (Boolean.TRUE.equals(UIManager.get(RadianceSynapse.AUTO_SCROLL)));
}
/**
* Checks whether the specified tree component has automatic drag and drop support.
*
* @param tree
* Tree component.
* @return true
if the specified text component has automatic drag and drop
* support, false
otherwise.
*/
public static boolean hasAutomaticDnDSupport(JTree tree) {
Object dndProperty = tree.getClientProperty(RadianceSynapse.TREE_AUTO_DND_SUPPORT);
if (Boolean.TRUE.equals(dndProperty)) {
return true;
}
if (Boolean.FALSE.equals(dndProperty)) {
return false;
}
return (Boolean.TRUE.equals(UIManager.get(RadianceSynapse.TREE_AUTO_DND_SUPPORT)));
}
/**
* Checks whether the specified component has been configured (specifically or globally) to have
* no animations of the specific facet. Can be used to cull unnecessary code in animation
* listeners on large tables and lists.
*
* @param comp
* Component.
* @param animationFacet
* Animation facet.
* @return true
if the specified component has been configured (specifically or
* globally) to have no animations of the specific facet, false
otherwise.
*/
public static boolean hasNoAnimations(Component comp, RadianceThemingSlices.AnimationFacet animationFacet) {
return !AnimationConfigurationManager.getInstance().isAnimationAllowed(animationFacet,
comp);
}
/**
* Returns the current icon for the specified button. This method is for internal use
* only.
*
* @param b
* Button.
* @return Icon for the specified button.
*/
public static Icon getIcon(AbstractButton b) {
Icon icon = b.getIcon();
if (icon == null)
return null;
ButtonModel model = b.getModel();
Icon tmpIcon = null;
if (icon != null) {
if (!model.isEnabled()) {
if (model.isSelected()) {
tmpIcon = b.getDisabledSelectedIcon();
} else {
tmpIcon = b.getDisabledIcon();
}
} else if (model.isPressed() && model.isArmed()) {
tmpIcon = b.getPressedIcon();
} else if (b.isRolloverEnabled() && model.isRollover()) {
if (model.isSelected()) {
tmpIcon = b.getRolloverSelectedIcon();
} else {
tmpIcon = b.getRolloverIcon();
}
} else if (model.isSelected()) {
tmpIcon = b.getSelectedIcon();
}
if (tmpIcon != null) {
icon = tmpIcon;
}
}
return icon;
}
public static boolean toIgnoreAnimations(Component comp) {
if (comp instanceof JMenuItem) {
return false;
}
return (SwingUtilities.getAncestorOfClass(CellRendererPane.class, comp) != null);
}
/**
* Fires the matching property change event on the specific component.
*
* @param component
* Component.
* @param propertyName
* Property name.
* @param oldValue
* Old property value.
* @param newValue
* New property value.
*/
public static void firePropertyChangeEvent(JComponent component, String propertyName,
Object oldValue, Object newValue) {
PropertyChangeEvent pce = new PropertyChangeEvent(component, propertyName, oldValue,
newValue);
for (PropertyChangeListener general : component.getPropertyChangeListeners()) {
general.propertyChange(pce);
}
for (PropertyChangeListener specific : component.getPropertyChangeListeners(propertyName)) {
specific.propertyChange(pce);
}
}
/**
* Returns the composite to use for painting the specified component. The result should be set
* on the {@link Graphics2D} before any custom rendering is done. This method can be used by
* application painting code and by look-and-feel delegates.
*
* @param c
* Component.
* @param translucency
* The translucency of the original painting.
* @param g
* The original graphics context.
* @return The composite to use for painting the specified component.
*/
public static Composite getAlphaComposite(Component c, float translucency, Graphics g) {
float xFactor = 1.0f;
if (g instanceof Graphics2D) {
Graphics2D g2d = (Graphics2D) g;
Composite existingComposite = g2d.getComposite();
if (existingComposite instanceof AlphaComposite) {
AlphaComposite ac = (AlphaComposite) existingComposite;
if (ac.getRule() == AlphaComposite.SRC_OVER)
xFactor = ac.getAlpha();
}
}
float finalAlpha = translucency * xFactor;
if (finalAlpha < 0.0f) {
finalAlpha = 0.0f;
}
if (finalAlpha > 1.0f) {
finalAlpha = 1.0f;
}
if (finalAlpha == 1.0f) {
return AlphaComposite.SrcOver;
}
return AlphaComposite.SrcOver.derive(finalAlpha);
}
public static Composite getAlphaComposite(Component c, float translucency) {
return getAlphaComposite(c, translucency, null);
}
/**
* Returns the composite to use for painting the specified component. The result should be set
* on the {@link Graphics2D} before any custom rendering is done. This method can be used by
* application painting code and by look-and-feel delegates.
*
* @param c
* Component.
* @return The composite to use for painting the specified component.
*/
public static Composite getAlphaComposite(Component c, Graphics g) {
return getAlphaComposite(c, 1.0f, g);
}
/**
* Returns the composite to use for painting the specified component. The result should be set
* on the {@link Graphics2D} before any custom rendering is done. This method can be used by
* application painting code and by look-and-feel delegates.
*
* @param c
* Component.
* @return The composite to use for painting the specified component.
*/
public static Composite getAlphaComposite(Component c) {
return getAlphaComposite(c, 1.0f, null);
}
/**
* Returns the preview painter for the specified component.
*
* @param comp
* Component.
* @return Preview painter for the specified component.
*/
public static PreviewPainter getComponentPreviewPainter(Component comp) {
if (comp == null) {
return null;
}
// check property on component
if (comp instanceof JComponent) {
Object compProp = ((JComponent) comp)
.getClientProperty(RadianceSynapse.COMPONENT_PREVIEW_PAINTER);
if (compProp instanceof PreviewPainter) {
return (PreviewPainter) compProp;
}
}
// check property on parent
Container parent = comp.getParent();
if (parent instanceof JComponent) {
Object parentProp = ((JComponent) parent)
.getClientProperty(RadianceSynapse.COMPONENT_PREVIEW_PAINTER);
if (parentProp instanceof PreviewPainter) {
return (PreviewPainter) parentProp;
}
}
Object globProp = UIManager.get(RadianceSynapse.COMPONENT_PREVIEW_PAINTER);
if (globProp instanceof PreviewPainter) {
return (PreviewPainter) globProp;
}
return null;
}
/**
* Returns the password strength checker for the specified password field.
*
* @param jpf
* Password field.
* @return Password strength checker for the specified password field. The result can be
* null
.
*/
public static PasswordStrengthChecker getPasswordStrengthChecker(JPasswordField jpf) {
Object obj = jpf.getClientProperty(RadianceSynapse.PASSWORD_STRENGTH_CHECKER);
if (obj instanceof PasswordStrengthChecker) {
return (PasswordStrengthChecker) obj;
}
return null;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy