blob: 06718a44963e33ff83fc41301d8f9413a1b4764a [file] [log] [blame]
/*
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You under the Apache License, Version 2.0
(the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package org.apache.batik.util.gui.resource;
import java.net.URL;
import java.util.List;
import java.util.MissingResourceException;
import java.util.ResourceBundle;
import javax.swing.AbstractButton;
import javax.swing.Action;
import javax.swing.ButtonGroup;
import javax.swing.ImageIcon;
import javax.swing.JCheckBoxMenuItem;
import javax.swing.JComponent;
import javax.swing.JMenu;
import javax.swing.JMenuBar;
import javax.swing.JMenuItem;
import javax.swing.JRadioButtonMenuItem;
import javax.swing.JSeparator;
import javax.swing.KeyStroke;
import org.apache.batik.util.resources.ResourceFormatException;
import org.apache.batik.util.resources.ResourceManager;
/**
* This class represents a menu factory which builds
* menubars and menus from the content of a resource file. <br>
*
* The resource entries format is (for a menubar named 'MenuBar'):<br>
* <pre>
* MenuBar = Menu1 Menu2 ...
*
* Menu1.type = RADIO | CHECK | MENU | ITEM
* Menu1 = Item1 Item2 - Item3 ...
* Menu1.text = text
* Menu1.icon = icon_name
* Menu1.mnemonic = mnemonic
* Menu1.accelerator = accelerator
* Menu1.action = action_name
* Menu1.selected = true | false
* Menu1.enabled = true | false
* ...
* mnemonic is a single character
* accelerator is of the form described in {@link javax.swing.KeyStroke#getKeyStroke(String)}.
* '-' represents a separator
* </pre>
* All entries are optional except the '.type' entry
* Consecutive RADIO items are put in a ButtonGroup
*
* @author <a href="mailto:stephane@hillion.org">Stephane Hillion</a>
* @version $Id$
*/
public class MenuFactory extends ResourceManager {
// Constants
//
private static final String TYPE_MENU = "MENU";
private static final String TYPE_ITEM = "ITEM";
private static final String TYPE_RADIO = "RADIO";
private static final String TYPE_CHECK = "CHECK";
private static final String SEPARATOR = "-";
private static final String TYPE_SUFFIX = ".type";
private static final String TEXT_SUFFIX = ".text";
private static final String MNEMONIC_SUFFIX = ".mnemonic";
private static final String ACCELERATOR_SUFFIX = ".accelerator";
private static final String ACTION_SUFFIX = ".action";
private static final String SELECTED_SUFFIX = ".selected";
private static final String ENABLED_SUFFIX = ".enabled";
private static final String ICON_SUFFIX = ".icon";
/**
* The table which contains the actions
*/
private ActionMap actions;
/**
* The current radio group
*/
private ButtonGroup buttonGroup;
/**
* Creates a new menu factory
* @param rb the resource bundle that contains the menu bar
* description.
* @param am the actions to add to menu items
*/
public MenuFactory(ResourceBundle rb, ActionMap am) {
super(rb);
actions = am;
buttonGroup = null;
}
/**
* Creates and returns a swing menu bar
* @param name the name of the menu bar in the resource bundle
* @throws MissingResourceException if one of the keys that compose the
* menu is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character and if the accelerator is malformed
* @throws MissingListenerException if an item action is not found in the
* action map
*/
public JMenuBar createJMenuBar(String name)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
return createJMenuBar(name, null);
}
/**
* Creates and returns a swing menu bar
* @param name the name of the menu bar in the resource bundle
* @param specialization the name of the specialization to look for
* @throws MissingResourceException if one of the keys that compose the
* menu is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character and if the accelerator is malformed
* @throws MissingListenerException if an item action is not found in the
* action map
*/
public JMenuBar createJMenuBar(String name, String specialization)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
JMenuBar result = new JMenuBar();
List menus = getSpecializedStringList(name, specialization);
for (Object menu : menus) {
result.add(createJMenuComponent((String) menu, specialization));
}
return result;
}
/**
* Gets a possibly specialized resource string.
* This will first look for
* <code>name + '.' + specialization</code>, and if that resource
* doesn't exist, <code>name</code>.
*/
protected String getSpecializedString(String name, String specialization) {
String s;
try {
s = getString(name + '.' + specialization);
} catch (MissingResourceException mre) {
s = getString(name);
}
return s;
}
/**
* Gets a possibly specialized resource string list.
* This will first look for
* <code>name + '.' + specialization</code>, and if that resource
* doesn't exist, <code>name</code>.
*/
protected List getSpecializedStringList(String name,
String specialization) {
List l;
try {
l = getStringList(name + '.' + specialization);
} catch (MissingResourceException mre) {
l = getStringList(name);
}
return l;
}
/**
* Gets a possibly specialized resource boolean.
* This will first look for
* <code>name + '.' + specialization</code>, and if that resource
* doesn't exist, <code>name</code>.
*/
protected boolean getSpecializedBoolean(String name,
String specialization) {
boolean b;
try {
b = getBoolean(name + '.' + specialization);
} catch (MissingResourceException mre) {
b = getBoolean(name);
}
return b;
}
/**
* Creates and returns a menu item or a separator
* @param name the name of the menu item or "-" to create a separator
* @param specialization the name of the specialization to look for
* @throws MissingResourceException if key is not the name of a menu item.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException in case of malformed entry
* @throws MissingListenerException if an item action is not found in the
* action map
*/
protected JComponent createJMenuComponent(String name,
String specialization)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
if (name.equals(SEPARATOR)) {
buttonGroup = null;
return new JSeparator();
}
String type = getSpecializedString(name + TYPE_SUFFIX,
specialization);
JComponent item = null;
if (type.equals(TYPE_RADIO)) {
if (buttonGroup == null) {
buttonGroup = new ButtonGroup();
}
} else {
buttonGroup = null;
}
if (type.equals(TYPE_MENU)) {
item = createJMenu(name, specialization);
} else if (type.equals(TYPE_ITEM)) {
item = createJMenuItem(name, specialization);
} else if (type.equals(TYPE_RADIO)) {
item = createJRadioButtonMenuItem(name, specialization);
buttonGroup.add((AbstractButton)item);
} else if (type.equals(TYPE_CHECK)) {
item = createJCheckBoxMenuItem(name, specialization);
} else {
throw new ResourceFormatException("Malformed resource",
bundle.getClass().getName(),
name+TYPE_SUFFIX);
}
return item;
}
/**
* Creates and returns a new swing menu
* @param name the name of the menu bar in the resource bundle
* @throws MissingResourceException if one of the keys that compose the
* menu is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if a item action is not found in the
* action map.
*/
public JMenu createJMenu(String name)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
return createJMenu(name, null);
}
/**
* Creates and returns a new swing menu
* @param name the name of the menu bar in the resource bundle
* @param specialization the name of the specialization to look for
* @throws MissingResourceException if one of the keys that compose the
* menu is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if a item action is not found in the
* action map.
*/
public JMenu createJMenu(String name, String specialization)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
JMenu result = new JMenu(getSpecializedString(name + TEXT_SUFFIX,
specialization));
initializeJMenuItem(result, name, specialization);
List items = getSpecializedStringList(name, specialization);
for (Object item : items) {
result.add(createJMenuComponent((String) item, specialization));
}
return result;
}
/**
* Creates and returns a new swing menu item
* @param name the name of the menu item
* @throws MissingResourceException if one of the keys that compose the
* menu item is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if then item action is not found in
* the action map.
*/
public JMenuItem createJMenuItem(String name)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
return createJMenuItem(name, null);
}
/**
* Creates and returns a new swing menu item
* @param name the name of the menu item
* @param specialization the name of the specialization to look for
* @throws MissingResourceException if one of the keys that compose the
* menu item is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if then item action is not found in
* the action map.
*/
public JMenuItem createJMenuItem(String name, String specialization)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
JMenuItem result = new JMenuItem(getSpecializedString(name + TEXT_SUFFIX,
specialization));
initializeJMenuItem(result, name, specialization);
return result;
}
/**
* Creates and returns a new swing radio button menu item
* @param name the name of the menu item
* @throws MissingResourceException if one of the keys that compose the
* menu item is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if then item action is not found in
* the action map.
*/
public JRadioButtonMenuItem createJRadioButtonMenuItem(String name)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
return createJRadioButtonMenuItem(name, null);
}
/**
* Creates and returns a new swing radio button menu item
* @param name the name of the menu item
* @param specialization the name of the specialization to look for
* @throws MissingResourceException if one of the keys that compose the
* menu item is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if then item action is not found in
* the action map.
*/
public JRadioButtonMenuItem createJRadioButtonMenuItem
(String name, String specialization)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
JRadioButtonMenuItem result;
result = new JRadioButtonMenuItem
(getSpecializedString(name + TEXT_SUFFIX, specialization));
initializeJMenuItem(result, name, specialization);
// is the item selected?
try {
result.setSelected(getSpecializedBoolean(name + SELECTED_SUFFIX,
specialization));
} catch (MissingResourceException e) {
}
return result;
}
/**
* Creates and returns a new swing check box menu item
* @param name the name of the menu item
* @throws MissingResourceException if one of the keys that compose the
* menu item is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if then item action is not found in
* the action map.
*/
public JCheckBoxMenuItem createJCheckBoxMenuItem(String name)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
return createJCheckBoxMenuItem(name, null);
}
/**
* Creates and returns a new swing check box menu item
* @param name the name of the menu item
* @param specialization the name of the specialization to look for
* @throws MissingResourceException if one of the keys that compose the
* menu item is missing.
* It is not thrown if the mnemonic, the accelerator and the
* action keys are missing
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if then item action is not found in
* the action map.
*/
public JCheckBoxMenuItem createJCheckBoxMenuItem(String name,
String specialization)
throws MissingResourceException,
ResourceFormatException,
MissingListenerException {
JCheckBoxMenuItem result;
result = new JCheckBoxMenuItem(getSpecializedString(name + TEXT_SUFFIX,
specialization));
initializeJMenuItem(result, name, specialization);
// is the item selected?
try {
result.setSelected(getSpecializedBoolean(name + SELECTED_SUFFIX,
specialization));
} catch (MissingResourceException e) {
}
return result;
}
/**
* Initializes a swing menu item
* @param item the menu item to initialize
* @param name the name of the menu item
* @param specialization the name of the specialization to look for
* @throws ResourceFormatException if the mnemonic is not a single
* character.
* @throws MissingListenerException if then item action is not found in
* the action map.
*/
protected void initializeJMenuItem(JMenuItem item, String name,
String specialization)
throws ResourceFormatException,
MissingListenerException {
// Action
try {
Action a = actions.getAction
(getSpecializedString(name + ACTION_SUFFIX, specialization));
if (a == null) {
throw new MissingListenerException("", "Action",
name+ACTION_SUFFIX);
}
item.setAction(a);
item.setText(getSpecializedString(name + TEXT_SUFFIX,
specialization));
if (a instanceof JComponentModifier) {
((JComponentModifier)a).addJComponent(item);
}
} catch (MissingResourceException e) {
}
// Icon
try {
String s = getSpecializedString(name + ICON_SUFFIX, specialization);
URL url = actions.getClass().getResource(s);
if (url != null) {
item.setIcon(new ImageIcon(url));
}
} catch (MissingResourceException e) {
}
// Mnemonic
try {
String str = getSpecializedString(name + MNEMONIC_SUFFIX,
specialization);
if (str.length() == 1) {
item.setMnemonic(str.charAt(0));
} else {
throw new ResourceFormatException("Malformed mnemonic",
bundle.getClass().getName(),
name+MNEMONIC_SUFFIX);
}
} catch (MissingResourceException e) {
}
// Accelerator
try {
if (!(item instanceof JMenu)) {
String str = getSpecializedString(name + ACCELERATOR_SUFFIX,
specialization);
KeyStroke ks = KeyStroke.getKeyStroke(str);
if (ks != null) {
item.setAccelerator(ks);
} else {
throw new ResourceFormatException
("Malformed accelerator",
bundle.getClass().getName(),
name+ACCELERATOR_SUFFIX);
}
}
} catch (MissingResourceException e) {
}
// is the item enabled?
try {
item.setEnabled(getSpecializedBoolean(name + ENABLED_SUFFIX,
specialization));
} catch (MissingResourceException e) {
}
}
}