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

org.jdesktop.swingx.SwingXUtilities Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2008 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;

import java.awt.Color;
import java.awt.Component;
import java.awt.Container;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.Graphics2D;
import java.awt.Insets;
import java.awt.Point;
import java.awt.Rectangle;
import java.awt.Window;
import java.awt.event.InputEvent;
import java.awt.event.MouseEvent;
import java.io.IOException;
import java.io.StringReader;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Locale;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
import java.util.logging.Logger;

import javax.swing.InputMap;
import javax.swing.JComponent;
import javax.swing.JList;
import javax.swing.JPopupMenu;
import javax.swing.KeyStroke;
import javax.swing.ListCellRenderer;
import javax.swing.ListModel;
import javax.swing.ListSelectionModel;
import javax.swing.MenuElement;
import javax.swing.Painter;
import javax.swing.RepaintManager;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.plaf.ComponentInputMapUIResource;
import javax.swing.plaf.UIResource;
import javax.swing.text.html.HTMLDocument;

/**
 * A collection of utility methods for Swing(X) classes.
 * 
 * 
    *
  • move this class to the swingx utils package which already has a bunch of xxUtils *
  • move methods between xxUtils classes as appropriate (one window/comp related util) *
  • keep here in swingx (consistent with swingutilities in core) *
* @author Karl George Schaefer */ /* * PENDING JW: think about location of this class and/or its methods, Options: */ public final class SwingXUtilities { private static final Logger LOG = Logger.getLogger(SwingXUtilities.class.getName()); private SwingXUtilities() { // all Methods are static //does nothing } /** * A helper for creating and updating key bindings for components with * mnemonics. The {@code pressed} action will be invoked when the mnemonic * is activated. *

* TODO establish an interface for the mnemonic properties, such as {@code * MnemonicEnabled} and change signature to {@code public static void updateMnemonicBinding(T c, String * pressed)} * * @param c * the component bindings to update * @param pressed * the name of the action in the action map to invoke when the * mnemonic is pressed * @throws NullPointerException * if the component is {@code null} */ public static void updateMnemonicBinding(JComponent c, String pressed) { updateMnemonicBinding(c, pressed, null); } /** * A helper for creating and updating key bindings for components with * mnemonics. The {@code pressed} action will be invoked when the mnemonic * is activated and the {@code released} action will be invoked when the * mnemonic is deactivated. *

* TODO establish an interface for the mnemonic properties, such as {@code * MnemonicEnabled} and change signature to {@code public static void updateMnemonicBinding(T c, String * pressed, String released)} * * @param c * the component bindings to update * @param pressed * the name of the action in the action map to invoke when the * mnemonic is pressed * @param released * the name of the action in the action map to invoke when the * mnemonic is released (if the action is a toggle style, then * this parameter should be {@code null}) * @throws NullPointerException * if the component is {@code null} */ public static void updateMnemonicBinding(JComponent c, String pressed, String released) { Class clazz = c.getClass(); int m = -1; try { Method mtd = clazz.getMethod("getMnemonic"); m = (Integer) mtd.invoke(c); } catch (RuntimeException e) { throw e; } catch (Exception e) { throw new IllegalArgumentException("unable to access mnemonic", e); } InputMap map = SwingUtilities.getUIInputMap(c, JComponent.WHEN_IN_FOCUSED_WINDOW); if (m != 0) { if (map == null) { map = new ComponentInputMapUIResource(c); SwingUtilities.replaceUIInputMap(c, JComponent.WHEN_IN_FOCUSED_WINDOW, map); } map.clear(); //TODO is ALT_MASK right for all platforms? Deprecated(since = "9") map.put(KeyStroke.getKeyStroke(m, InputEvent.ALT_DOWN_MASK, false), pressed); map.put(KeyStroke.getKeyStroke(m, InputEvent.ALT_DOWN_MASK, true), released); map.put(KeyStroke.getKeyStroke(m, 0, true), released); } else { if (map != null) { map.clear(); } } } @SuppressWarnings("unchecked") static void paintBackground(C comp, Graphics2D g) { // we should be painting the background behind the painter if we have one // this prevents issues with buffer reuse where visual artifacts sneak in if (comp.isOpaque() || (comp instanceof AlphaPaintable && ((AlphaPaintable) comp).getAlpha() < 1f) || UIManager.getLookAndFeel().getID().equals("Nimbus")) { g.setColor(comp.getBackground()); g.fillRect(0, 0, comp.getWidth(), comp.getHeight()); } Painter painter = comp.getBackgroundPainter(); if (painter != null) { if (comp.isPaintBorderInsets()) { LOG.fine("isPaintBorderInsets"); painter.paint(g, comp, comp.getWidth(), comp.getHeight()); } else { Insets insets = comp.getInsets(); LOG.fine("Insets:"+insets); g.translate(insets.left, insets.top); painter.paint(g, comp, comp.getWidth() - insets.left - insets.right, comp.getHeight() - insets.top - insets.bottom ); g.translate(-insets.left, -insets.top); } } } private static Component[] getChildren(Component c) { Component[] children = null; if (c instanceof MenuElement) { MenuElement[] elements = ((MenuElement) c).getSubElements(); children = new Component[elements.length]; for (int i = 0; i < elements.length; i++) { children[i] = elements[i].getComponent(); } } else if (c instanceof Container) { children = ((Container) c).getComponents(); } return children; } /** * Enables or disables of the components in the tree starting with {@code c}. * * @param c * the starting component * @param enabled * {@code true} if the component is to enabled; {@code false} otherwise */ public static void setComponentTreeEnabled(Component c, boolean enabled) { c.setEnabled(enabled); Component[] children = getChildren(c); if (children != null) { for(int i = 0; i < children.length; i++) { setComponentTreeEnabled(children[i], enabled); } } } /** * Sets the locale for an entire component hierarchy to the specified * locale. * * @param c * the starting component * @param locale * the locale to set */ public static void setComponentTreeLocale(Component c, Locale locale) { c.setLocale(locale); Component[] children = getChildren(c); if (children != null) { for(int i = 0; i < children.length; i++) { setComponentTreeLocale(children[i], locale); } } } /** * Sets the background for an entire component hierarchy to the specified * color. * * @param c * the starting component * @param color * the color to set */ public static void setComponentTreeBackground(Component c, Color color) { c.setBackground(color); Component[] children = getChildren(c); if (children != null) { for(int i = 0; i < children.length; i++) { setComponentTreeBackground(children[i], color); } } } /** * Sets the foreground for an entire component hierarchy to the specified * color. * * @param c * the starting component * @param color * the color to set */ public static void setComponentTreeForeground(Component c, Color color) { c.setForeground(color); Component[] children = getChildren(c); if (children != null) { for(int i = 0; i < children.length; i++) { setComponentTreeForeground(children[i], color); } } } /** * Sets the font for an entire component hierarchy to the specified font. * * @param c * the starting component * @param font * the font to set */ public static void setComponentTreeFont(Component c, Font font) { c.setFont(font); Component[] children = getChildren(c); if (children != null) { for(int i = 0; i < children.length; i++) { setComponentTreeFont(children[i], font); } } } private static String STYLESHEET = "body { margin-top: 0; margin-bottom: 0; margin-left: 0; margin-right: 0;" + " font-family: %s; font-size: %dpt; }" + "a, p, li { margin-top: 0; margin-bottom: 0; margin-left: 0;" + " margin-right: 0; font-family: %s; font-size: %dpt; }"; /** * Sets the font used for HTML displays to the specified font. Components * that display HTML do not necessarily honor font properties, since the * HTML document can override these values. Calling {@code setHtmlFont} * after the data is set will force the HTML display to use the font * specified to this method. * * @param doc * the HTML document to update * @param font * the font to use * @throws NullPointerException * if any parameter is {@code null} */ public static void setHtmlFont(HTMLDocument doc, Font font) { String stylesheet = String.format(STYLESHEET, font.getName(), font.getSize(), font.getName(), font.getSize()); try { doc.getStyleSheet().loadRules(new StringReader(stylesheet), null); } catch (IOException e) { //this should never happen with our sheet throw new IllegalStateException(e); } } /** * Updates the componentTreeUI of all top-level windows of the * current application. * */ public static void updateAllComponentTreeUIs() { for (Window window: Window.getWindows()) { SwingUtilities.updateComponentTreeUI(window); } } /** * Updates the componentTreeUI of the given window and all its * owned windows, recursively. * * * @param window the window to update */ public static void updateAllComponentTreeUIs(Window window) { SwingUtilities.updateComponentTreeUI(window); for (Window owned : window.getOwnedWindows()) { updateAllComponentTreeUIs(owned); } } /** * A version of {@link SwingUtilities#invokeLater(Runnable)} that supports return values. * * @param * the return type of the callable * @param callable * the callable to execute * @return a future task for accessing the return value * @see Callable */ public static FutureTask invokeLater(Callable callable) { FutureTask task = new FutureTask(callable); SwingUtilities.invokeLater(task); return task; } /** * A version of {@link SwingUtilities#invokeAndWait(Runnable)} that supports return values. * * @param * the return type of the callable * @param callable * the callable to execute * @return the value returned by the callable * @throws InterruptedException * if we're interrupted while waiting for the event dispatching thread to finish * executing {@code callable.call()} * @throws InvocationTargetException * if an exception is thrown while running {@code callable} * @see Callable */ public static T invokeAndWait(Callable callable) throws InterruptedException, InvocationTargetException { try { //blocks until future returns return invokeLater(callable).get(); } catch (ExecutionException e) { Throwable t = e.getCause(); if (t instanceof RuntimeException) { throw (RuntimeException) t; } else if (t instanceof InvocationTargetException) { throw (InvocationTargetException) t; } else { throw new InvocationTargetException(t); } } } /** * An improved version of * {@link SwingUtilities#getAncestorOfClass(Class, Component)}. This method * traverses {@code JPopupMenu} invoker and uses generics to return an * appropriately typed object. * * @param * the type of ancestor to find * @param clazz * the class instance of the ancestor to find * @param c * the component to start the search from * @return an ancestor of the correct type or {@code null} if no such * ancestor exists. This method also returns {@code null} if any * parameter is {@code null}. */ @SuppressWarnings("unchecked") public static T getAncestor(Class clazz, Component c) { if (clazz == null || c == null) { return null; } Component parent = c.getParent(); while (parent != null && !(clazz.isInstance(parent))) { parent = parent instanceof JPopupMenu ? ((JPopupMenu) parent).getInvoker() : parent.getParent(); } return (T) parent; } /** * Returns whether the component is part of the parent's * container hierarchy. If a parent in the chain is of type * JPopupMenu, the parent chain of its invoker is walked. * * @param focusOwner * @param parent * @return true if the component is contained under the parent's * hierarchy, coping with JPopupMenus. */ public static boolean isDescendingFrom(Component focusOwner, Component parent) { while (focusOwner != null) { if (focusOwner instanceof JPopupMenu) { focusOwner = ((JPopupMenu) focusOwner).getInvoker(); if (focusOwner == null) { return false; } } if (focusOwner == parent) { return true; } focusOwner = focusOwner.getParent(); } return false; } /** * Obtains a {@code TranslucentRepaintManager} from the specified manager. * If the current manager is a {@code TranslucentRepaintManager} or a * {@code ForwardingRepaintManager} that contains a {@code * TranslucentRepaintManager}, then the passed in manager is returned. * Otherwise a new repaint manager is created and returned. * * @param delegate * the current repaint manager * @return a non-{@code null} {@code TranslucentRepaintManager} * @throws NullPointerException if {@code delegate} is {@code null} */ static RepaintManager getTranslucentRepaintManager(RepaintManager delegate) { RepaintManager manager = delegate; while (manager != null && !manager.getClass().isAnnotationPresent(TranslucentRepaintManager.class)) { if (manager instanceof ForwardingRepaintManager) { manager = ((ForwardingRepaintManager) manager).getDelegateManager(); } else { manager = null; } } return manager == null ? new RepaintManagerX(delegate) : delegate; } /** * Checks and returns whether the given property should be replaced * by the UI's default value. * * @param property the property to check. * @return true if the given property should be replaced by the UI's * default value, false otherwise. */ public static boolean isUIInstallable(Object property) { return (property == null) || (property instanceof UIResource); } //---- methods c&p'ed from SwingUtilities2 to reduce dependencies on sun packages /** * Updates lead and anchor selection index without changing the selection. * * Note: this is pasted from SwingUtilities2 to not have any direct dependency. * * @param selectionModel the selection model to change lead/anchor * @param lead the lead selection index * @param anchor the anchor selection index */ public static void setLeadAnchorWithoutSelection( ListSelectionModel selectionModel, int lead, int anchor) { if (anchor == -1) { anchor = lead; } if (lead == -1) { selectionModel.setAnchorSelectionIndex(-1); selectionModel.setLeadSelectionIndex(-1); } else { if (selectionModel.isSelectedIndex(lead)) selectionModel.addSelectionInterval(lead, lead); else { selectionModel.removeSelectionInterval(lead, lead); } selectionModel.setAnchorSelectionIndex(anchor); } } /** * Ignore mouse events if the component is null, not enabled, the event * is not associated with the left mouse button, or the event has been * consumed. */ // like sun.swing.SwingUtilities2#shouldIgnore public static boolean shouldIgnore(MouseEvent mouseEvent, JComponent component) { return ((component == null) || (!(component.isEnabled())) || (!(SwingUtilities.isLeftMouseButton(mouseEvent))) || (mouseEvent.isConsumed())); } public static int loc2IndexFileList(JList list, Point point) { int i = list.locationToIndex(point); if (i != -1) { Object localObject = list.getClientProperty("List.isFileList"); if ((localObject instanceof Boolean) && (((Boolean) localObject).booleanValue()) // PENDING JW: this isn't aware of sorting/filtering - fix! && (!(pointIsInActualBounds(list, i, point)))) { i = -1; } } return i; } // PENDING JW: this isn't aware of sorting/filtering - fix! private static boolean pointIsInActualBounds(JList list, int index, Point point) { ListCellRenderer renderer = list.getCellRenderer(); ListModel model = list.getModel(); Object element = model.getElementAt(index); Component comp = renderer.getListCellRendererComponent(list, element, index, false, false); Dimension prefSize = comp.getPreferredSize(); Rectangle cellBounds = list.getCellBounds(index, index); if (!(comp.getComponentOrientation().isLeftToRight())) { cellBounds.x += cellBounds.width - prefSize.width; } cellBounds.width = prefSize.width; return cellBounds.contains(point); } public static void adjustFocus(JComponent component) { if ((!(component.hasFocus())) && (component.isRequestFocusEnabled())) component.requestFocus(); } public static int convertModifiersToDropAction(int modifiers, int sourcActions) { // PENDING JW: c'p from a decompiled SunDragSourceContextPeer // PENDING JW: haha ... completely readable, right ;-) int i = 0; switch (modifiers & 0xC0) { case 192: i = 1073741824; break; case 128: i = 1; break; case 64: i = 2; break; default: if ((sourcActions & 0x2) != 0) { i = 2; break; } if ((sourcActions & 0x1) != 0) { i = 1; break; } if ((sourcActions & 0x40000000) == 0) break; i = 1073741824; } // label88: return (i & sourcActions); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy