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!
/*
 * $Id: LookAndFeelAddons.java 3132 2008-12-05 14:34:58Z kschaefe $
 *
 * 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.Method;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;

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

import org.jdesktop.swingx.painter.Painter;
import org.jdesktop.swingx.plaf.linux.LinuxLookAndFeelAddons;
import org.jdesktop.swingx.plaf.macosx.MacOSXLookAndFeelAddons;
import org.jdesktop.swingx.plaf.metal.MetalLookAndFeelAddons;
import org.jdesktop.swingx.plaf.motif.MotifLookAndFeelAddons;
import org.jdesktop.swingx.plaf.nimbus.NimbusLookAndFeelAddons;
import org.jdesktop.swingx.plaf.windows.WindowsClassicLookAndFeelAddons;
import org.jdesktop.swingx.plaf.windows.WindowsLookAndFeelAddons;
import org.jdesktop.swingx.util.OS;

/**
 * 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 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 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; static { // load the default addon String addonClassname = getBestMatchAddonClassName(); try { addonClassname = System.getProperty("swing.addon", addonClassname); } catch (SecurityException e) { // security exception may arise in Java Web Start } 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 ExceptionInInitializerError(e); } setTrackingLookAndFeelChanges(true); } private static LookAndFeelAddons currentAddon; public void initialize() { for (Iterator iter = contributedComponents.iterator(); iter .hasNext();) { ComponentAddon addon = iter.next(); addon.initialize(this); } } 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 */ 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]); } } } 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); } */ } public static void setAddon(String addonClassName) throws InstantiationException, IllegalAccessException, ClassNotFoundException { setAddon(Class.forName(addonClassName)); } public static void setAddon(Class addonClass) throws InstantiationException, IllegalAccessException { LookAndFeelAddons addon = (LookAndFeelAddons)addonClass.newInstance(); setAddon(addon); } 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); } public static LookAndFeelAddons getAddon() { return currentAddon; } /** * Based on the current look and feel (as returned by * UIManager.getLookAndFeel()), this method returns * the name of the closest LookAndFeelAddons to use. * * @return the addon matching the currently installed look and feel */ public static String getBestMatchAddonClassName() { String lnf = UIManager.getLookAndFeel().getClass().getName(); String addon; if (UIManager.getCrossPlatformLookAndFeelClassName().equals(lnf)) { addon = MetalLookAndFeelAddons.class.getName(); } else if (UIManager.getSystemLookAndFeelClassName().equals(lnf)) { addon = getSystemAddonClassName(); } else if ("com.sun.java.swing.plaf.windows.WindowsLookAndFeel".equals(lnf) || "com.jgoodies.looks.windows.WindowsLookAndFeel".equals(lnf)) { if (OS.isUsingWindowsVisualStyles()) { addon = WindowsLookAndFeelAddons.class.getName(); } else { addon = WindowsClassicLookAndFeelAddons.class.getName(); } } else if ("com.sun.java.swing.plaf.windows.WindowsClassicLookAndFeel" .equals(lnf)) { addon = WindowsClassicLookAndFeelAddons.class.getName(); } else if (UIManager.getLookAndFeel().getID().equals("Motif")) { addon = MotifLookAndFeelAddons.class.getName(); } else if (UIManager.getLookAndFeel().getID().equals("Nimbus")) { addon = NimbusLookAndFeelAddons.class.getName(); } else { addon = getSystemAddonClassName(); } return addon; } /** * Gets the addon best suited for the operating system where the * virtual machine is running. * * @return the addon matching the native operating system platform. */ public static String getSystemAddonClassName() { String addon = WindowsClassicLookAndFeelAddons.class.getName(); if (OS.isMacOSX()) { addon = MacOSXLookAndFeelAddons.class.getName(); } else if (OS.isWindows()) { // see whether of not visual styles are used if (OS.isUsingWindowsVisualStyles()) { addon = WindowsLookAndFeelAddons.class.getName(); } else { addon = WindowsClassicLookAndFeelAddons.class.getName(); } } else if (OS.isLinux()) { addon = LinuxLookAndFeelAddons.class.getName(); } return addon; } /** * Each new component added by the library will contribute its * default UI classes, colors and fonts to the LookAndFeelAddons. * See {@link ComponentAddon}. * * @param component */ 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 */ 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 * @param expectedUIClass * @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 { String realUI = ui.getClass().getName(); Class realUIClass; try { realUIClass = expectedUIClass.getClassLoader() .loadClass(realUI); } catch (ClassNotFoundException e) { throw new RuntimeException("Failed to load class " + realUI, e); } Method createUIMethod = null; try { createUIMethod = realUIClass.getMethod("createUI", new Class[]{JComponent.class}); } catch (NoSuchMethodException e1) { throw new RuntimeException("Class " + realUI + " has no method createUI(JComponent)"); } try { return (ComponentUI)createUIMethod.invoke(null, new Object[]{component}); } catch (Exception e2) { throw new RuntimeException("Failed to invoke " + realUI + "#createUI(JComponent)"); } } } /** * 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 { 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) { 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()); } } }