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

org.jdesktop.swingx.action.ActionContainerFactory Maven / Gradle / Ivy

/*
 * 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.action;

import java.awt.Insets;
import java.beans.PropertyChangeListener;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import javax.swing.AbstractButton;
import javax.swing.Action;
import javax.swing.ActionMap;
import javax.swing.ButtonGroup;
import javax.swing.Icon;
import javax.swing.JButton;
import javax.swing.JCheckBoxMenuItem;
import javax.swing.JComponent;
import javax.swing.JMenu;
import javax.swing.JMenuBar;
import javax.swing.JMenuItem;
import javax.swing.JPopupMenu;
import javax.swing.JRadioButtonMenuItem;
import javax.swing.JToggleButton;
import javax.swing.JToolBar;

/**
 * Creates user interface elements based on action ids and lists of action ids.
 * All action ids must represent actions managed by the ActionManager.
 * 
 * 

Action Lists

* Use the createXXX(List) methods to construct containers of actions like menu * bars, menus, popups and toolbars from actions represented as action ids in a * java.util.List. Each element in the action-list can be one of 3 types: *
    *
  • action id: corresponds to an action managed by the ActionManager *
  • null: indicates a separator should be inserted. *
  • java.util.List: represents a submenu. See the note below which describes * the configuration of menus. *
  • *
* The order of elements in an action-list determines the arrangement of the ui * components which are constructed from the action-list. *

* For a menu or submenu, the first element in the action-list represents a menu * and subsequent elements represent menu items or separators (if null). *

* This class can be used as a general component factory which will construct * components from Actions if the create<comp>(Action,...) * methods are used. * * @see ActionManager */ public class ActionContainerFactory { /** * Standard margin for toolbar buttons to improve their look */ private static Insets TOOLBAR_BUTTON_MARGIN = new Insets(1, 1, 1, 1); private ActionMap manager; // Map between group id + component and the ButtonGroup private Map groupMap; /** * Constructs an container factory which uses the default ActionManager. * */ public ActionContainerFactory() { } /** * Constructs an container factory which uses managed actions. * * @param manager use the actions managed with this manager for * constructing ui componenents. */ public ActionContainerFactory(ActionMap manager) { setActionManager(manager); } /** * Gets the ActionManager instance. * If the ActionManager has not been explicitly set then * the default ActionManager instance will be used. * * @return the ActionManager used by the ActionContainerFactory. * @see #setActionManager */ public ActionMap getActionManager() { if (manager == null) { manager = ActionManager.getInstance(); } return manager; } /** * Sets the ActionManager instance that will be used by this ActionContainerFactory * * @param manager the ActionManager */ public void setActionManager(ActionMap manager) { this.manager = manager; } /** * Constructs a toolbar from an action-list id. * By convention, the identifier of the main toolbar should be "main-toolbar" * * @param list a list of action ids used to construct the toolbar. * @return the toolbar or null */ public JToolBar createToolBar(Object[] list) { return createToolBar(Arrays.asList(list)); } /** * Constructs a toolbar from an action-list id. By convention, * the identifier of the main toolbar should be "main-toolbar" * * @param list a list of action ids used to construct the toolbar. * @return the toolbar or null */ public JToolBar createToolBar(List list) { JToolBar toolbar = new JToolBar(); Iterator iter = list.iterator(); while(iter.hasNext()) { Object element = iter.next(); if (element == null) { toolbar.addSeparator(); } else { AbstractButton button = createButton(element, toolbar); // toolbar buttons shouldn't steal focus button.setFocusable(false); /* * TODO * The next two lines improve the default look of the buttons. * This code should be changed to retrieve the default look * from some UIDefaults object. */ button.setMargin(TOOLBAR_BUTTON_MARGIN); button.setBorderPainted(false); toolbar.add(button); } } return toolbar; } /** * Constructs a popup menu from an array of action ids. * * @param list an array of action ids used to construct the popup. * @return the popup or null */ public JPopupMenu createPopup(Object[] list) { return createPopup(Arrays.asList(list)); } /** * Constructs a popup menu from a list of action ids. * * @param list a list of action ids used to construct the popup. * @return the popup or null */ public JPopupMenu createPopup(List list) { JPopupMenu popup = new JPopupMenu(); Iterator iter = list.iterator(); while(iter.hasNext()) { Object element = iter.next(); if (element == null) { popup.addSeparator(); } else if (element instanceof List) { JMenu newMenu= createMenu((List)element); if (newMenu!= null) { popup.add(newMenu); } } else { popup.add(createMenuItem(element, popup)); } } return popup; } /** * Constructs a menu tree from a list of actions or lists of lists or actions. * * @param actionIds an array which represents the root item. * @return a menu bar which represents the menu bar tree */ public JMenuBar createMenuBar(Object[] actionIds) { return createMenuBar(Arrays.asList(actionIds)); } /** * Constructs a menu tree from a list of actions or lists of lists or actions. * * @param list a list which represents the root item. * @return a menu bar which represents the menu bar tree */ public JMenuBar createMenuBar(List list) { final JMenuBar menubar = new JMenuBar(); for (Object element : list) { if (element == null) { continue; } JMenuItem menu; if (element instanceof Object[]) { menu = createMenu((Object[]) element); } else if (element instanceof List) { menu = createMenu((List) element); } else { menu = createMenuItem(element, menubar); } if (menu != null) { menubar.add(menu); } } return menubar; } /** * Creates and returns a menu from a List which represents actions, separators and sub-menus. * The menu constructed will have the attributes from the first action in the List. * Subsequent actions in the list represent menu items. * * @param actionIds an array of action ids used to construct the menu and menu items. * the first element represents the action used for the menu, * @return the constructed JMenu or null */ public JMenu createMenu(Object[] actionIds) { return createMenu(Arrays.asList(actionIds)); } /** * Creates and returns a menu from a List which represents actions, separators and sub-menus. * The menu constructed will have the attributes from the first action in the List. * Subsequent actions in the list represent menu items. * * @param list a list of action ids used to construct the menu and menu items. * the first element represents the action used for the menu, * @return the constructed JMenu or null */ public JMenu createMenu(List list) { // The first item will be the action for the JMenu Action action = getAction(list.get(0)); if (action == null) { return null; } JMenu menu = new JMenu(action); // The rest of the items represent the menu items. for (Object element : list.subList(1, list.size())) { if (element == null) { menu.addSeparator(); } else { JMenuItem newMenu; // submenu if (element instanceof Object[]) { newMenu = createMenu((Object[]) element); } else if (element instanceof List) { newMenu = createMenu((List) element); } else { // item newMenu = createMenuItem(element, menu); } if (newMenu != null) { menu.add(newMenu); } } } return menu; } /** * Convenience method to get the action from an ActionManager. */ private Action getAction(Object id) { return getActionManager().get(id); } /** * Returns the button group corresponding to the groupid * * @param groupid the value of the groupid attribute for the action element * @param container a container which will further identify the ButtonGroup */ private ButtonGroup getGroup(String groupid, JComponent container) { if (groupMap == null) { groupMap = new HashMap(); } int intCode = groupid.hashCode(); if (container != null) { intCode ^= container.hashCode(); } Integer hashCode = intCode; ButtonGroup group = groupMap.get(hashCode); if (group == null) { group = new ButtonGroup(); groupMap.put(hashCode, group); } return group; } /** * Creates a menu item based on the attributes of the action element. * Will return a JMenuItem, JRadioButtonMenuItem or a JCheckBoxMenuItem * depending on the context of the Action. * * @return a JMenuItem or subclass depending on type. */ private JMenuItem createMenuItem(Object id, JComponent container) { return createMenuItem(getAction(id), container); } /** * Creates a menu item based on the attributes of the action element. * Will return a JMenuItem, JRadioButtonMenuItem or a JCheckBoxMenuItem * depending on the context of the Action. * * @param action a managed Action * @param container the parent container may be null for non-group actions. * @return a JMenuItem or subclass depending on type. */ private JMenuItem createMenuItem(Action action, JComponent container) { JMenuItem menuItem = null; if (action instanceof AbstractActionExt) { AbstractActionExt ta = (AbstractActionExt)action; if (ta.isStateAction()) { String groupid = (String)ta.getGroup(); if (groupid != null) { // If this action has a groupid attribute then it's a // GroupAction menuItem = createRadioButtonMenuItem(getGroup(groupid, container), (AbstractActionExt)action); } else { menuItem = createCheckBoxMenuItem((AbstractActionExt)action); } } } if (menuItem == null) { menuItem= new JMenuItem(action); configureMenuItemFromExtActionProperties(menuItem, action); } return menuItem; } /** * Creates a menu item based on the attributes of the action. * Will return a JMenuItem, JRadioButtonMenuItem or a JCheckBoxMenuItem * depending on the context of the Action. * * @param action an action used to create the menu item * @return a JMenuItem or subclass depending on type. */ public JMenuItem createMenuItem(Action action) { return createMenuItem(action, null); } /** * Creates, configures and returns an AbstractButton. * * The attributes of the action element * registered with the ActionManger by the given id. * Will return a JButton or a JToggleButton. * * @param id the identifier * @param container the JComponent which parents the group, if any. * @return an AbstractButton based on the */ public AbstractButton createButton(Object id, JComponent container) { return createButton(getAction(id), container); } /** * Creates a button based on the attributes of the action. If the container * parameter is non-null then it will be used to uniquely identify * the returned component within a ButtonGroup. If the action doesn't * represent a grouped component then this value can be null. * * @param action an action used to create the button * @param container the parent container to uniquely identify * grouped components or null * @return will return a JButton or a JToggleButton. */ public AbstractButton createButton(Action action, JComponent container) { if (action == null) { return null; } AbstractButton button = null; if (action instanceof AbstractActionExt) { // Check to see if we should create a toggle button AbstractActionExt ta = (AbstractActionExt)action; if (ta.isStateAction()) { // If this action has a groupid attribute then it's a // GroupAction String groupid = (String)ta.getGroup(); if (groupid == null) { button = createToggleButton(ta); } else { button = createToggleButton(ta, getGroup(groupid, container)); } } } if (button == null) { // Create a regular button button = new JButton(action); configureButtonFromExtActionProperties(button, action); } return button; } /** * Creates a button based on the attributes of the action. * * @param action an action used to create the button * @return will return a JButton or a JToggleButton. */ public AbstractButton createButton(Action action) { return createButton(action, null); } /** * Adds and configures a toggle button. * @param a an abstraction of a toggle action. */ private JToggleButton createToggleButton(AbstractActionExt a) { return createToggleButton(a, null); } /** * Adds and configures a toggle button. * @param a an abstraction of a toggle action. * @param group the group to add the toggle button or null */ private JToggleButton createToggleButton(AbstractActionExt a, ButtonGroup group) { JToggleButton button = new JToggleButton(); configureButton(button, a, group); return button; } /** * method to configure a button from the given AbstractActionExt. * * @param button * @param a action * @param group the button should be added to. */ public void configureButton(JToggleButton button, AbstractActionExt a, ButtonGroup group) { configureSelectableButton(button, a, group); configureButtonFromExtActionProperties(button, a); } /** * method to configure a "selectable" button from the given AbstractActionExt. * As there is some un-/wiring involved to support synch of the selected property between * the action and the button, all config and unconfig (== setting a null action!) * should be passed through this method.

* * It's up to the client to only pass in button's where selected and/or the * group property makes sense. * * PENDING: the group properties are yet untested. * PENDING: think about automated unconfig. * * @param button where selected makes sense * @param a action * @param group the button should be added to. * @throws IllegalArgumentException if the given action doesn't have the state flag set. * */ public void configureSelectableButton(AbstractButton button, AbstractActionExt a, ButtonGroup group){ if ((a != null) && !a.isStateAction()) throw new IllegalArgumentException("the Action must be a stateAction"); // we assume that all button configuration is done exclusively through this method!! if (button.getAction() == a) return; // unconfigure if the old Action is a state AbstractActionExt // PENDING JW: automate unconfigure via a PCL that is listening to // the button's action property? Think about memory leak implications! Action oldAction = button.getAction(); if (oldAction instanceof AbstractActionExt) { AbstractActionExt actionExt = (AbstractActionExt) oldAction; // remove as itemListener button.removeItemListener(actionExt); // remove the button related PCL from the old actionExt PropertyChangeListener[] l = actionExt.getPropertyChangeListeners(); for (int i = l.length - 1; i >= 0; i--) { if (l[i] instanceof ToggleActionPropertyChangeListener) { ToggleActionPropertyChangeListener togglePCL = (ToggleActionPropertyChangeListener) l[i]; if (togglePCL.isToggling(button)) { actionExt.removePropertyChangeListener(togglePCL); } } } } button.setAction(a); if (group != null) { group.add(button); } if (a != null) { button.addItemListener(a); // JW: move the initial config into the PCL?? button.setSelected(a.isSelected()); new ToggleActionPropertyChangeListener(a, button); // new ToggleActionPCL(button, a); } } /** * This method will be called after buttons created from an action. Override * for custom configuration. * * @param button the button to be configured * @param action the action used to construct the menu item. */ protected void configureButtonFromExtActionProperties(AbstractButton button, Action action) { if (action.getValue(Action.SHORT_DESCRIPTION) == null) { button.setToolTipText((String)action.getValue(Action.NAME)); } // Use the large icon for toolbar buttons. if (action.getValue(AbstractActionExt.LARGE_ICON) != null) { button.setIcon((Icon)action.getValue(AbstractActionExt.LARGE_ICON)); } // Don't show the text under the toolbar buttons if they have an icon if (button.getIcon() != null) { button.setText(""); } } /** * This method will be called after menu items are created. * Override for custom configuration. * * @param menuItem the menu item to be configured * @param action the action used to construct the menu item. */ protected void configureMenuItemFromExtActionProperties(JMenuItem menuItem, Action action) { } /** * Helper method to add a checkbox menu item. */ private JCheckBoxMenuItem createCheckBoxMenuItem(AbstractActionExt a) { JCheckBoxMenuItem mi = new JCheckBoxMenuItem(); configureSelectableButton(mi, a, null); configureMenuItemFromExtActionProperties(mi, a); return mi; } /** * Helper method to add a radio button menu item. */ private JRadioButtonMenuItem createRadioButtonMenuItem(ButtonGroup group, AbstractActionExt a) { JRadioButtonMenuItem mi = new JRadioButtonMenuItem(); configureSelectableButton(mi, a, group); configureMenuItemFromExtActionProperties(mi, a); return mi; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy