All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jdesktop.swingx.plaf.LookAndFeelAddons Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2004 Sun Microsystems, Inc., 4150 Network Circle,
 * Santa Clara, California 95054, U.S.A. All rights reserved.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 */
package org.jdesktop.swingx.plaf;

import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.security.PrivilegedAction;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.ServiceLoader;
import java.util.logging.Level;
import java.util.logging.Logger;

import javax.swing.JComponent;
import javax.swing.LookAndFeel;
import javax.swing.Painter;
import javax.swing.UIDefaults;
import javax.swing.UIManager;
import javax.swing.plaf.ComponentUI;
import javax.swing.plaf.UIResource;

/**
 * Provides additional pluggable UI for new components added by the library. By default, the library
 * uses the pluggable UI returned by {@link #getBestMatchAddonClassName()}.
 * 

* The default addon can be configured using the swing.addon system property as follow: *

    *
  • on the command line, java -Dswing.addon=ADDONCLASSNAME ...
  • *
  • at runtime and before using the library components * System.getProperties().put("swing.addon", ADDONCLASSNAME);
  • *
*

* The default {@link #getCrossPlatformAddonClassName() cross platform addon} can be configured * using the swing.crossplatformlafaddon system property as follow: *

    *
  • on the command line, java -Dswing.crossplatformlafaddon=ADDONCLASSNAME ...
  • *
  • at runtime and before using the library components * System.getProperties().put("swing.crossplatformlafaddon", ADDONCLASSNAME);
    * Note: changing this property after the UI has been initialized may result in unexpected behavior. *
  • *
*

* The addon can also be installed directly by calling the {@link #setAddon(String)}method. For * example, to install the Windows addons, add the following statement * LookAndFeelAddons.setAddon("org.jdesktop.swingx.plaf.windows.WindowsLookAndFeelAddons");. * * @author Frederic Lavigne * @author Karl Schaefer */ public abstract class LookAndFeelAddons { private static final Logger LOG = Logger.getLogger(LookAndFeelAddons.class.getName()); private static List contributedComponents = new ArrayList(); /** * Key used to ensure the current UIManager has been populated by the LookAndFeelAddons. */ private static final Object APPCONTEXT_INITIALIZED = new Object(); private static boolean trackingChanges = false; private static PropertyChangeListener changeListener; @Override public String toString() { return ""+contributedComponents.size()+" contributedComponents, trackingChanges="+trackingChanges; } static { // load the default addon String addonClassname = getBestMatchAddonClassName(); try { addonClassname = System.getProperty("swing.addon", addonClassname); } catch (SecurityException ignore) { // security exception may arise in Java Web Start LOG.log(Level.FINE, "not allowed to access property swing.addon", ignore); } try { setAddon(addonClassname); } catch (Exception e) { // PENDING(fred) do we want to log an error and continue with a default // addon class or do we just fail? // throw new java.lang.ExceptionInInitializerError(e); // fehlt LOG.warning("ExceptionInInitializerError "+e); } setTrackingLookAndFeelChanges(true); } private static LookAndFeelAddons currentAddon; /** * Determines if the addon is a match for the {@link UIManager#getLookAndFeel() current Look and * Feel}. * * @return {@code true} if this addon matches (is compatible); {@code false} otherwise */ protected boolean matches() { return false; } /** * Determines if the addon is a match for the system Look and Feel. * * @return {@code true} if this addon matches (is compatible with) the system Look and Feel; * {@code false} otherwise */ protected boolean isSystemAddon() { return false; } /** * Initializes the look and feel addon. * * @see #uninitialize * @see UIManager#setLookAndFeel */ public void initialize() { for (Iterator iter = contributedComponents.iterator(); iter.hasNext();) { ComponentAddon addon = iter.next(); addon.initialize(this); } } /** * Uninitializes the look and feel addon. */ public void uninitialize() { for (Iterator iter = contributedComponents.iterator(); iter.hasNext();) { ComponentAddon addon = iter.next(); addon.uninitialize(this); } } /** * Adds the given defaults in UIManager. * * Note: the values are added only if they do not exist in the existing look and feel defaults. * This makes it possible for look and feel implementors to override SwingX defaults. * * Note: the array is traversed in reverse order. If a key is found twice in the array, the * key/value with the highest position in the array gets precedence over the other key in the * array * * @param keysAndValues array of keys and values */ public void loadDefaults(Object[] keysAndValues) { // Go in reverse order so the most recent keys get added first... for (int i = keysAndValues.length - 2; i >= 0; i = i - 2) { if (UIManager.getLookAndFeelDefaults().get(keysAndValues[i]) == null) { UIManager.getLookAndFeelDefaults().put(keysAndValues[i], keysAndValues[i + 1]); } } } /** * unload Objects * @param keysAndValues Objects to unload */ @Deprecated public void unloadDefaults(Object[] keysAndValues) { // commented after Issue 446. /* * for (int i = 0, c = keysAndValues.length; i < c; i = i + 2) { * UIManager.getLookAndFeelDefaults().put(keysAndValues[i], null); } */ } /** * load and install addon * @param addonClassName the addon class name * @throws ClassNotFoundException throws by forName * @throws InstantiationException throws by newInstance() * @throws IllegalAccessException throws by newInstance() * @throws IllegalArgumentException throws by newInstance() * @throws InvocationTargetException throws by newInstance() * @throws NoSuchMethodException throws by getDeclaredConstructor() * @throws SecurityException throws by getDeclaredConstructor() */ public static void setAddon(String addonClassName) throws ClassNotFoundException, InstantiationException, IllegalAccessException , IllegalArgumentException, InvocationTargetException, NoSuchMethodException, SecurityException { setAddon(Class.forName(addonClassName, true, getClassLoader())); } /** * install addon * @param addonClass the addon class * @throws InstantiationException throws by newInstance() * @throws IllegalAccessException throws by newInstance() * @throws IllegalArgumentException throws by newInstance() * @throws InvocationTargetException throws by newInstance() * @throws NoSuchMethodException throws by getDeclaredConstructor() * @throws SecurityException throws by getDeclaredConstructor() */ public static void setAddon(Class addonClass) throws InstantiationException, IllegalAccessException , IllegalArgumentException, InvocationTargetException, NoSuchMethodException, SecurityException { LookAndFeelAddons addon = (LookAndFeelAddons) addonClass.getDeclaredConstructor().newInstance(); setAddon(addon); } /** * install addon * @param addon org.jdesktop.swingx.plaf.LookAndFeelAddons */ public static void setAddon(LookAndFeelAddons addon) { if (currentAddon != null) { currentAddon.uninitialize(); } addon.initialize(); currentAddon = addon; // JW: we want a marker to discover if the LookAndFeelDefaults have been // swept from under our feet. The following line looks suspicious, // as it is setting a user default instead of a LF default. User defaults // are not touched when resetting a LF UIManager.put(APPCONTEXT_INITIALIZED, Boolean.TRUE); // trying to fix #784-swingx: frequent NPE on getUI // JW: we want a marker to discover if the LookAndFeelDefaults have been // swept from under our feet. UIManager.getLookAndFeelDefaults().put(APPCONTEXT_INITIALIZED, Boolean.TRUE); } /** * get the current addons * @return current LookAndFeelAddons */ public static LookAndFeelAddons getAddon() { return currentAddon; } static class ClassLoaderPrivilegedAction implements PrivilegedAction { Class type = null; Thread thread = null; ClassLoaderPrivilegedAction(Class clazz) { this.type = clazz; } ClassLoaderPrivilegedAction(Thread t) { this.thread = t; } ClassLoaderPrivilegedAction() { } @Override public ClassLoader run() { if(type!=null) return type.getClassLoader(); if(thread!=null) return thread.getContextClassLoader(); return ClassLoader.getSystemClassLoader(); } } private static ClassLoader getClassLoader() { ClassLoader cl = null; try { ClassLoaderPrivilegedAction action = new ClassLoaderPrivilegedAction(LookAndFeelAddons.class); cl = action.run(); } catch (SecurityException ignore) { } if (cl == null) { final Thread t = Thread.currentThread(); try { ClassLoaderPrivilegedAction action = new ClassLoaderPrivilegedAction(t); cl = action.run(); } catch (SecurityException ignore) { } } if (cl == null) { try { ClassLoaderPrivilegedAction action = new ClassLoaderPrivilegedAction(); cl = action.run(); } catch (SecurityException ignore) { } } return cl; } /** * Based on the current look and feel (as returned by UIManager.getLookAndFeel()), * this method returns the name of the closest LookAndFeelAddons to use. *

* The lookup fallback is implemented *

    *
  1. check for cross-platform LAF and return the cross-platform addons *
  2. check for system LAF and return the system addons *
  3. loop through the addons provided by services and return the first that matches *
  4. return the cross-platform addons *
* * @return the addon matching the currently installed look and feel * * @see #getCrossPlatformAddonClassName() * @see #getSystemAddonClassName() * @see #getProvidedLookAndFeelAddons() */ public static String getBestMatchAddonClassName() { LookAndFeel laf = UIManager.getLookAndFeel(); String className = null; if (UIManager.getCrossPlatformLookAndFeelClassName().equals(laf.getClass().getName())) { className = getCrossPlatformAddonClassName(); } else if (UIManager.getSystemLookAndFeelClassName().equals(laf.getClass().getName())) { className = getSystemAddonClassName(); } else { Iterable loadedAddons = getProvidedLookAndFeelAddons(); for (LookAndFeelAddons addon : loadedAddons) { if (addon.matches()) { className = addon.getClass().getName(); break; } } } if (className == null) { className = getSystemAddonClassName(); } return className; } /** * Returns the addon class name best suited for cross-platform laf. * The lookup sequence is implemented to *
    *
  1. get and return the system property * swing.crossplatformlafaddon if available *
  2. return the addon (hard-coded!) for Metal *
* * @return the class name of the cross-platform addon */ public static String getCrossPlatformAddonClassName() { PrivilegedAction action = new PrivilegedAction() { @Override public String run() { return System.getProperty("swing.crossplatformlafaddon", "org.jdesktop.swingx.plaf.metal.MetalLookAndFeelAddons"); } }; return action.run(); } /** * Gets the addon best suited for the operating system where the virtual machine is running. * * The lookup is implemented like *
    *
  1. loop through the addons provided by services and return the first * that isSystemAddon *
  2. return the cross-platform addons *
* * @return the addon matching the native operating system platform. * * @see #getProvidedLookAndFeelAddons() * @see #getCrossPlatformAddonClassName() */ public static String getSystemAddonClassName() { Iterable loadedAddons = getProvidedLookAndFeelAddons(); String className = null; for (LookAndFeelAddons addon : loadedAddons) { if (addon.isSystemAddon()) { className = addon.getClass().getName(); break; } } if (className == null) { className = getCrossPlatformAddonClassName(); } return className; } /** * a simple inner class that implements PrivilegedAction */ public static class IterableLAFAddonsPrivilegedAction implements PrivilegedAction> { ServiceLoader loader; // class ServiceLoader implements Iterable IterableLAFAddonsPrivilegedAction(ServiceLoader l) { this.loader = l; } @Override public Iterable run() { loader.iterator().hasNext(); return loader; } } /** * Returns the LookAndFeelAddons from the ServiceLoader. * * The actual lookup of the provider must * be triggered in a privileged action to force * loading the classes. Without, it might not have access to the * provider configuration file in security restricted contexts. * * @return the LookAndFeelAddons from the ServiceLoader */ protected static Iterable getProvidedLookAndFeelAddons() { final ServiceLoader loader = ServiceLoader.load(LookAndFeelAddons.class, getClassLoader()); // need to access the iterator inside a privileged action // probably because it's lazily loaded IterableLAFAddonsPrivilegedAction action = new IterableLAFAddonsPrivilegedAction(loader); return action.run(); } /** * Each new component added by the library will contribute its default UI classes, * colors and fonts to the LookAndFeelAddons. * See {@link ComponentAddon}. * * @param component ComponentAddon that contribute */ public static void contribute(ComponentAddon component) { contributedComponents.add(component); if (currentAddon != null) { // make sure to initialize any addons added after the // LookAndFeelAddons has been installed component.initialize(currentAddon); } } /** * Removes the contribution of the given addon * * @param component ComponentAddon */ public static void uncontribute(ComponentAddon component) { contributedComponents.remove(component); if (currentAddon != null) { component.uninitialize(currentAddon); } } /** * Workaround for IDE mixing up with classloaders and Applets environments. * Consider this method as API private. It must not be called directly! * * @param component javax.swing.JComponent * @param expectedUIClass the UI Class * @return an instance of expectedUIClass */ public static ComponentUI getUI(JComponent component, Class expectedUIClass) { maybeInitialize(); // solve issue with ClassLoader not able to find classes String uiClassname = (String) UIManager.get(component.getUIClassID()); // possible workaround and more debug info on #784 if (uiClassname == null) { Logger logger = Logger.getLogger("LookAndFeelAddons"); logger.warning("Failed to retrieve UI for " + component.getClass().getName() + " with UIClassID " + component.getUIClassID()); if (logger.isLoggable(Level.FINE)) { logger.fine("Existing UI defaults keys: " + new ArrayList(UIManager.getDefaults().keySet())); } // really ugly hack. Should be removed as soon as we figure out what is causing the // issue uiClassname = "org.jdesktop.swingx.plaf.basic.Basic" + expectedUIClass.getSimpleName(); } try { Class uiClass = Class.forName(uiClassname); UIManager.put(uiClassname, uiClass); } catch (ClassNotFoundException e) { // we ignore the ClassNotFoundException } ComponentUI ui = UIManager.getUI(component); if (expectedUIClass.isInstance(ui)) { return ui; } else if (ui == null) { barkOnUIError("no ComponentUI class for: " + component); } else { String realUI = ui.getClass().getName(); Class realUIClass = null; try { realUIClass = expectedUIClass.getClassLoader().loadClass(realUI); } catch (ClassNotFoundException e) { barkOnUIError("failed to load class " + realUI); } if (realUIClass != null) { try { Method createUIMethod = realUIClass.getMethod("createUI", new Class[] { JComponent.class }); return (ComponentUI) createUIMethod.invoke(null, new Object[] { component }); } catch (NoSuchMethodException e) { barkOnUIError("static createUI() method not found in " + realUIClass); } catch (Exception e) { barkOnUIError("createUI() failed for " + component + " " + e); } } } return null; } // this is how core UIDefaults yells about bad components; we do the same private static void barkOnUIError(String message) { System.err.println(message); new Error().printStackTrace(); } /** * With applets, if you reload the current applet, the UIManager will be reinitialized (entries * previously added by LookAndFeelAddons will be removed) but the addon will not reinitialize * because addon initialize itself through the static block in components and the classes do not * get reloaded. This means component.updateUI will fail because it will not find its UI. * * This method ensures LookAndFeelAddons get re-initialized if needed. It must be called in * every component updateUI methods. */ private static synchronized void maybeInitialize() { if (currentAddon != null) { // this is to ensure "UIManager#maybeInitialize" gets called and the // LAFState initialized UIDefaults defaults = UIManager.getLookAndFeelDefaults(); // if (!UIManager.getBoolean(APPCONTEXT_INITIALIZED)) { // JW: trying to fix #784-swingx: frequent NPE in getUI // moved the "marker" property into the LookAndFeelDefaults if (!defaults.getBoolean(APPCONTEXT_INITIALIZED)) { setAddon(currentAddon); } } } // // TRACKING OF THE CURRENT LOOK AND FEEL // private static class UpdateAddon implements PropertyChangeListener { @Override public void propertyChange(PropertyChangeEvent evt) { try { setAddon(getBestMatchAddonClassName()); } catch (Exception e) { // should not happen throw new RuntimeException(e); } } } /** * If true, everytime the Swing look and feel is changed, the addon which best matches the * current look and feel will be automatically selected. * * @param tracking * true to automatically update the addon, false to not automatically track the * addon. Defaults to false. * @see #getBestMatchAddonClassName() */ public static synchronized void setTrackingLookAndFeelChanges(boolean tracking) { if (trackingChanges != tracking) { if (tracking) { if (changeListener == null) { changeListener = new UpdateAddon(); } UIManager.addPropertyChangeListener(changeListener); } else { if (changeListener != null) { UIManager.removePropertyChangeListener(changeListener); } changeListener = null; } trackingChanges = tracking; } } /** * @return true if the addon will be automatically change to match the current look and feel * @see #setTrackingLookAndFeelChanges(boolean) */ public static synchronized boolean isTrackingLookAndFeelChanges() { return trackingChanges; } /** * Convenience method for setting a component's background painter property with a value from * the defaults. The painter is only set if the painter is {@code null} or an instance of * {@code UIResource}. * * @param c * component to set the painter on * @param painter * key specifying the painter * @throws NullPointerException * if the component or painter is {@code null} * @throws IllegalArgumentException * if the component does not contain the "backgroundPainter" property or the * property cannot be set */ public static void installBackgroundPainter(JComponent c, String painter) { Class clazz = c.getClass(); try { Method getter = clazz.getMethod("getBackgroundPainter"); Method setter = clazz.getMethod("setBackgroundPainter", Painter.class); Painter p = (Painter) getter.invoke(c); if (p == null || p instanceof UIResource) { LOG.finer("painter:"+painter + " UIManager.get::"+UIManager.get(painter)); LOG.finer("painter:"+painter + " UIManagerExt::"+UIManagerExt.getPainter(painter)); setter.invoke(c, UIManagerExt.getPainter(painter)); } } catch (RuntimeException e) { throw e; } catch (Exception e) { throw new IllegalArgumentException("cannot set painter on " + c.getClass()); } } /** * Convenience method for uninstalling a background painter. If the painter of the component is * a {@code UIResource}, it is set to {@code null}. * * @param c * component to uninstall the painter on * @throws NullPointerException * if {@code c} is {@code null} * @throws IllegalArgumentException * if the component does not contain the "backgroundPainter" property or the * property cannot be set */ public static void uninstallBackgroundPainter(JComponent c) { Class clazz = c.getClass(); try { Method getter = clazz.getMethod("getBackgroundPainter"); Method setter = clazz.getMethod("setBackgroundPainter", Painter.class); Painter p = (Painter) getter.invoke(c); if (p == null || p instanceof UIResource) { setter.invoke(c, (Painter) null); } } catch (RuntimeException e) { throw e; } catch (Exception e) { throw new IllegalArgumentException("cannot set painter on " + c.getClass()); } } }