001: /* ========================================================================
002: * JCommon : a free general purpose class library for the Java(tm) platform
003: * ========================================================================
004: *
005: * (C) Copyright 2000-2005, by Object Refinery Limited and Contributors.
006: *
007: * Project Info: http://www.jfree.org/jcommon/index.html
008: *
009: * This library is free software; you can redistribute it and/or modify it
010: * under the terms of the GNU Lesser General Public License as published by
011: * the Free Software Foundation; either version 2.1 of the License, or
012: * (at your option) any later version.
013: *
014: * This library is distributed in the hope that it will be useful, but
015: * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
016: * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
017: * License for more details.
018: *
019: * You should have received a copy of the GNU Lesser General Public
020: * License along with this library; if not, write to the Free Software
021: * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
022: * USA.
023: *
024: * [Java is a trademark or registered trademark of Sun Microsystems, Inc.
025: * in the United States and other countries.]
026: *
027: * ---------------------
028: * AbstractTabbedUI.java
029: * ---------------------
030: * (C)opyright 2004, by Thomas Morgner and Contributors.
031: *
032: * Original Author: Thomas Morgner;
033: * Contributor(s): David Gilbert (for Object Refinery Limited);
034: *
035: * $Id: AbstractTabbedUI.java,v 1.9 2005/11/03 09:55:27 mungady Exp $
036: *
037: * Changes
038: * -------------------------
039: * 16-Feb-2004 : Initial version
040: * 07-Jun-2004 : Added standard header (DG);
041: */
042:
043: package org.jfree.ui.tabbedui;
044:
045: import java.awt.BorderLayout;
046: import java.awt.Component;
047: import java.awt.Window;
048: import java.awt.event.ActionEvent;
049: import java.beans.PropertyChangeEvent;
050: import java.beans.PropertyChangeListener;
051: import java.util.ArrayList;
052:
053: import javax.swing.AbstractAction;
054: import javax.swing.Action;
055: import javax.swing.JComponent;
056: import javax.swing.JMenu;
057: import javax.swing.JMenuBar;
058: import javax.swing.JPanel;
059: import javax.swing.JTabbedPane;
060: import javax.swing.SwingConstants;
061: import javax.swing.SwingUtilities;
062: import javax.swing.event.ChangeEvent;
063: import javax.swing.event.ChangeListener;
064:
065: import org.jfree.util.Log;
066:
067: /**
068: * A tabbed GUI. All views on the data are contained in tabs.
069: *
070: * @author Thomas Morgner
071: */
072: public abstract class AbstractTabbedUI extends JComponent {
073:
074: /** The menu bar property key. */
075: public static final String JMENUBAR_PROPERTY = "jMenuBar";
076:
077: /** The global menu property. */
078: public static final String GLOBAL_MENU_PROPERTY = "globalMenu";
079:
080: /**
081: * An exit action.
082: */
083: protected class ExitAction extends AbstractAction {
084:
085: /**
086: * Defines an <code>Action</code> object with a default
087: * description string and default icon.
088: */
089: public ExitAction() {
090: putValue(NAME, "Exit");
091: }
092:
093: /**
094: * Invoked when an action occurs.
095: *
096: * @param e the event.
097: */
098: public void actionPerformed(final ActionEvent e) {
099: attempExit();
100: }
101:
102: }
103:
104: /**
105: * A tab change handler.
106: */
107: private class TabChangeHandler implements ChangeListener {
108:
109: /** The tabbed pane to which this handler is registered. */
110: private final JTabbedPane pane;
111:
112: /**
113: * Creates a new handler.
114: *
115: * @param pane the pane.
116: */
117: public TabChangeHandler(final JTabbedPane pane) {
118: this .pane = pane;
119: }
120:
121: /**
122: * Invoked when the target of the listener has changed its state.
123: *
124: * @param e a ChangeEvent object
125: */
126: public void stateChanged(final ChangeEvent e) {
127: setSelectedEditor(this .pane.getSelectedIndex());
128: }
129: }
130:
131: /**
132: * A tab enable change listener.
133: */
134: private class TabEnableChangeListener implements
135: PropertyChangeListener {
136:
137: /**
138: * Default constructor.
139: */
140: public TabEnableChangeListener() {
141: }
142:
143: /**
144: * This method gets called when a bound property is changed.
145: *
146: * @param evt A PropertyChangeEvent object describing the event source
147: * and the property that has changed.
148: */
149: public void propertyChange(final PropertyChangeEvent evt) {
150: if (evt.getPropertyName().equals("enabled") == false) {
151: Log.debug("PropertyName");
152: return;
153: }
154: if (evt.getSource() instanceof RootEditor == false) {
155: Log.debug("Source");
156: return;
157: }
158: final RootEditor editor = (RootEditor) evt.getSource();
159: updateRootEditorEnabled(editor);
160: }
161: }
162:
163: /** The list of root editors. One for each tab. */
164: private ArrayList rootEditors;
165: /** The tabbed pane filling the content area. */
166: private JTabbedPane tabbedPane;
167: /** The index of the currently selected root editor. */
168: private int selectedRootEditor;
169: /** The current toolbar. */
170: private JComponent currentToolbar;
171: /** The container component for the toolbar. */
172: private JPanel toolbarContainer;
173: /** The close action assigned to this UI. */
174: private Action closeAction;
175: /** The current menu bar. */
176: private JMenuBar jMenuBar;
177: /** Whether the UI should build a global menu from all root editors. */
178: private boolean globalMenu;
179:
180: /**
181: * Default constructor.
182: */
183: public AbstractTabbedUI() {
184: this .selectedRootEditor = -1;
185:
186: this .toolbarContainer = new JPanel();
187: this .toolbarContainer.setLayout(new BorderLayout());
188:
189: this .tabbedPane = new JTabbedPane(SwingConstants.BOTTOM);
190: this .tabbedPane.addChangeListener(new TabChangeHandler(
191: this .tabbedPane));
192:
193: this .rootEditors = new ArrayList();
194:
195: setLayout(new BorderLayout());
196: add(this .toolbarContainer, BorderLayout.NORTH);
197: add(this .tabbedPane, BorderLayout.CENTER);
198:
199: this .closeAction = createCloseAction();
200: }
201:
202: /**
203: * Returns the tabbed pane.
204: *
205: * @return The tabbed pane.
206: */
207: protected JTabbedPane getTabbedPane() {
208: return this .tabbedPane;
209: }
210:
211: /**
212: * Defines whether to use a global unified menu bar, which contains
213: * all menus from all tab-panes or whether to use local menubars.
214: * <p>
215: * From an usability point of view, global menubars should be preferred,
216: * as this way users always see which menus are possibly available and
217: * do not wonder where the menus are disappearing.
218: *
219: * @return true, if global menus should be used, false otherwise.
220: */
221: public boolean isGlobalMenu() {
222: return this .globalMenu;
223: }
224:
225: /**
226: * Sets the global menu flag.
227: *
228: * @param globalMenu the flag.
229: */
230: public void setGlobalMenu(final boolean globalMenu) {
231: this .globalMenu = globalMenu;
232: if (isGlobalMenu()) {
233: setJMenuBar(updateGlobalMenubar());
234: } else {
235: if (getRootEditorCount() > 0) {
236: setJMenuBar(createEditorMenubar(getRootEditor(getSelectedEditor())));
237: }
238: }
239: }
240:
241: /**
242: * Returns the menu bar.
243: *
244: * @return The menu bar.
245: */
246: public JMenuBar getJMenuBar() {
247: return this .jMenuBar;
248: }
249:
250: /**
251: * Sets the menu bar.
252: *
253: * @param menuBar the menu bar.
254: */
255: protected void setJMenuBar(final JMenuBar menuBar) {
256: final JMenuBar oldMenuBar = this .jMenuBar;
257: this .jMenuBar = menuBar;
258: firePropertyChange(JMENUBAR_PROPERTY, oldMenuBar, menuBar);
259: }
260:
261: /**
262: * Creates a close action.
263: *
264: * @return A close action.
265: */
266: protected Action createCloseAction() {
267: return new ExitAction();
268: }
269:
270: /**
271: * Returns the close action.
272: *
273: * @return The close action.
274: */
275: public Action getCloseAction() {
276: return this .closeAction;
277: }
278:
279: /**
280: * Returns the prefix menus.
281: *
282: * @return The prefix menus.
283: */
284: protected abstract JMenu[] getPrefixMenus();
285:
286: /**
287: * The postfix menus.
288: *
289: * @return The postfix menus.
290: */
291: protected abstract JMenu[] getPostfixMenus();
292:
293: /**
294: * Adds menus.
295: *
296: * @param menuBar the menu bar
297: * @param customMenus the menus that should be added.
298: */
299: private void addMenus(final JMenuBar menuBar,
300: final JMenu[] customMenus) {
301: for (int i = 0; i < customMenus.length; i++) {
302: menuBar.add(customMenus[i]);
303: }
304: }
305:
306: /**
307: * Updates the global menu bar.
308: * @return the fully initialized menu bar.
309: */
310: private JMenuBar updateGlobalMenubar() {
311: JMenuBar menuBar = getJMenuBar();
312: if (menuBar == null) {
313: menuBar = new JMenuBar();
314: } else {
315: menuBar.removeAll();
316: }
317:
318: addMenus(menuBar, getPrefixMenus());
319: for (int i = 0; i < this .rootEditors.size(); i++) {
320: final RootEditor editor = (RootEditor) this .rootEditors
321: .get(i);
322: addMenus(menuBar, editor.getMenus());
323: }
324: addMenus(menuBar, getPostfixMenus());
325: return menuBar;
326: }
327:
328: /**
329: * Creates a menu bar.
330: *
331: * @param root
332: * @return A menu bar.
333: */
334: private JMenuBar createEditorMenubar(final RootEditor root) {
335:
336: JMenuBar menuBar = getJMenuBar();
337: if (menuBar == null) {
338: menuBar = new JMenuBar();
339: } else {
340: menuBar.removeAll();
341: }
342:
343: addMenus(menuBar, getPrefixMenus());
344: if (isGlobalMenu()) {
345: for (int i = 0; i < this .rootEditors.size(); i++) {
346: final RootEditor editor = (RootEditor) this .rootEditors
347: .get(i);
348: addMenus(menuBar, editor.getMenus());
349: }
350: } else {
351: addMenus(menuBar, root.getMenus());
352: }
353: addMenus(menuBar, getPostfixMenus());
354: return menuBar;
355: }
356:
357: /**
358: * Adds a root editor.
359: *
360: * @param rootPanel the root panel.
361: */
362: public void addRootEditor(final RootEditor rootPanel) {
363: this .rootEditors.add(rootPanel);
364: this .tabbedPane.add(rootPanel.getEditorName(), rootPanel
365: .getMainPanel());
366: rootPanel.addPropertyChangeListener("enabled",
367: new TabEnableChangeListener());
368: updateRootEditorEnabled(rootPanel);
369: if (getRootEditorCount() == 1) {
370: setSelectedEditor(0);
371: } else if (isGlobalMenu()) {
372: setJMenuBar(updateGlobalMenubar());
373: }
374: }
375:
376: /**
377: * Returns the number of root editors.
378: *
379: * @return The count.
380: */
381: public int getRootEditorCount() {
382: return this .rootEditors.size();
383: }
384:
385: /**
386: * Returns the specified editor.
387: *
388: * @param pos the position index.
389: *
390: * @return The editor at the given position.
391: */
392: public RootEditor getRootEditor(final int pos) {
393: return (RootEditor) this .rootEditors.get(pos);
394: }
395:
396: /**
397: * Returns the selected editor.
398: *
399: * @return The selected editor.
400: */
401: public int getSelectedEditor() {
402: return this .selectedRootEditor;
403: }
404:
405: /**
406: * Sets the selected editor.
407: *
408: * @param selectedEditor the selected editor.
409: */
410: public void setSelectedEditor(final int selectedEditor) {
411: final int oldEditor = this .selectedRootEditor;
412: if (oldEditor == selectedEditor) {
413: // no change - so nothing to do!
414: return;
415: }
416: this .selectedRootEditor = selectedEditor;
417: // make sure that only the selected editor is active.
418: // all other editors will be disabled, if needed and
419: // not touched if they are already in the correct state
420:
421: for (int i = 0; i < this .rootEditors.size(); i++) {
422: final boolean shouldBeActive = (i == selectedEditor);
423: final RootEditor container = (RootEditor) this .rootEditors
424: .get(i);
425: if (container.isActive() && (shouldBeActive == false)) {
426: container.setActive(false);
427: }
428: }
429:
430: if (this .currentToolbar != null) {
431: closeToolbar();
432: this .toolbarContainer.removeAll();
433: this .currentToolbar = null;
434: }
435:
436: for (int i = 0; i < this .rootEditors.size(); i++) {
437: final boolean shouldBeActive = (i == selectedEditor);
438: final RootEditor container = (RootEditor) this .rootEditors
439: .get(i);
440: if ((container.isActive() == false)
441: && (shouldBeActive == true)) {
442: container.setActive(true);
443: setJMenuBar(createEditorMenubar(container));
444: this .currentToolbar = container.getToolbar();
445: if (this .currentToolbar != null) {
446: this .toolbarContainer.add(this .currentToolbar,
447: BorderLayout.CENTER);
448: this .toolbarContainer.setVisible(true);
449: this .currentToolbar.setVisible(true);
450: } else {
451: this .toolbarContainer.setVisible(false);
452: }
453:
454: this .getJMenuBar().repaint();
455: }
456: }
457: }
458:
459: /**
460: * Closes the toolbar.
461: */
462: private void closeToolbar() {
463: if (this .currentToolbar != null) {
464: if (this .currentToolbar.getParent() != this .toolbarContainer) {
465: // ha!, the toolbar is floating ...
466: // Log.debug (currentToolbar.getParent());
467: final Window w = SwingUtilities
468: .windowForComponent(this .currentToolbar);
469: if (w != null) {
470: w.setVisible(false);
471: w.dispose();
472: }
473: }
474: this .currentToolbar.setVisible(false);
475: }
476: }
477:
478: /**
479: * Attempts to exit.
480: */
481: protected abstract void attempExit();
482:
483: /**
484: * Update handler for the enable state of the root editor.
485: *
486: * @param editor the editor.
487: */
488: protected void updateRootEditorEnabled(final RootEditor editor) {
489:
490: final boolean enabled = editor.isEnabled();
491: for (int i = 0; i < this .tabbedPane.getTabCount(); i++) {
492: final Component tab = this.tabbedPane.getComponentAt(i);
493: if (tab == editor.getMainPanel()) {
494: this.tabbedPane.setEnabledAt(i, enabled);
495: return;
496: }
497: }
498: }
499: }
|