Package com.jidesoft.dialog

Source Code of com.jidesoft.dialog.MultiplePageDialog

/*
* @(#)MultiplePageDialog.java
*
* Copyright 2002 - 2003 JIDE Software. All rights reserved.
*/
package com.jidesoft.dialog;

import com.jidesoft.swing.JideButton;
import com.jidesoft.swing.JideScrollPane;
import com.jidesoft.swing.JideSwingUtilities;

import javax.swing.*;
import javax.swing.event.*;
import javax.swing.tree.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.*;

/**
* MultiplePageDialog is a StandardDialog which can have multiple AbstractDialogPages. You can choose one from four
* predefined styles of how to change the page visibility. Those four styles are TAB_STYLE, ICON_STYLE, LIST_STYLE and
* TREE_STYLE.
* <p/>
* To use this class, just create a PageList of AbstractDialogPage and call setPageList() to set to this dialog. Based
* on the style, the class will automatically layout those pages correctly and hook up actions to switch based on user
* selection.
* <p/>
* As AbstractDialogPage extends AbstractPage, so you can always use PageListener to decide what to do when page is
* opened, closing, closed, activated or deactivated.
* <p/>
* We automatically create a button panel which have three button - OK, Cancel and Apply. The ButtonPanel listens to
* ButtonEvent from all the pages. You can simply fireButtonEvent in the page to change the state of those buttons. Or
* if you want to create your own button panel, just override createButtonPanel() method.
* <p/>
* If you choose LIST_STYLE and TREE_STYLE, you can set your own ListCellRenderer and TreeCellRenderer. Just call
* setListCellRenderer() and setTreeCellRenderer(). The value passed in the renderer is an instance of
* AbstractDialogPage associated with that list row or tree node.
*/
public class MultiplePageDialog extends StandardDialog {
    private static final long serialVersionUID = 4915689214157425081L;

    /**
     * Predefined style of multiple page dialog.
     */
    public static final int TAB_STYLE = 0;

    /**
     * Predefined style of multiple page dialog.
     */
    public static final int TREE_STYLE = 1;

    /**
     * Predefined style of multiple page dialog.
     */
    public static final int LIST_STYLE = 2;

    /**
     * Predefined style of multiple page dialog.
     */
    public static final int ICON_STYLE = 3;

    private int _style;

    private PageList _pageList;

    /**
     * The left pane to show the icon, list etc. It's an index area to choose which page.
     */
    private JComponent _indexPanel;

    /**
     * The panel contains all the pages. In TAB_STYLE, it is a tabbed pane and in other styles, it's a panel with
     * CardLayout.
     */
    private JComponent _pagesPanel;
    private CardLayout _cardLayout;

    /**
     * Map that maps from page full title to tree node. It provides a fast access from page full title to the tree node
     * in TREE_STYLE.
     */
    private Map<String, MutableTreeNode> _titleNodeMap;

    private JButton _okButton;
    private JButton _cancelButton;
    private JButton _applyButton;

    private TreeCellRenderer _treeCellRenderer;

    private ListCellRenderer _listCellRenderer;
    private JTabbedPane _tabbedPane;

    private String _initialPageTitle;
    public JTree _tree;

    /**
     * Creates a non-modal MultiplePageDialog without a title and without a specified <code>Frame</code> owner.  A
     * shared, hidden frame will be set as the owner of the dialog. By default TAB_STYLE is used.
     *
     * @throws HeadlessException if the page does not support mouse or key events.
     */
    public MultiplePageDialog() throws HeadlessException {
        this((Frame) null);
    }

    /**
     * Creates a non-modal MultiplePageDialog without a title with the specified <code>Frame</code> as its owner.  If
     * <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the owner of the dialog. By
     * default TAB_STYLE is used.
     *
     * @param owner the owner of the dialog
     * @throws HeadlessException if the page does not support mouse or key events.
     */
    public MultiplePageDialog(Frame owner) throws HeadlessException {
        this(owner, false);
    }

    /**
     * Creates a modal or non-modal MultiplePageDialog without a title and with the specified owner <code>Frame</code>.
     * If <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the owner of the dialog. By
     * default TAB_STYLE is used.
     *
     * @param owner the <code>Frame</code> from which the dialog is displayed
     * @param modal true for a modal dialog, false for one that allows others windows to be active at the same time
     * @throws HeadlessException if the page does not support mouse or key events.
     */
    public MultiplePageDialog(Frame owner, boolean modal) throws HeadlessException {
        this(owner, "", modal);
    }

    /**
     * Creates a non-modal MultiplePageDialog with the specified title and with the specified owner frame.  If
     * <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the owner of the dialog.
     *
     * @param owner the <code>Frame</code> from which the dialog is displayed
     * @param title the <code>String</code> to display in the dialog's title bar
     * @throws HeadlessException if GraphicsEnvironment.isHeadless() returns true.
     * @see java.awt.GraphicsEnvironment#isHeadless
     * @see JComponent#getDefaultLocale
     */
    public MultiplePageDialog(Frame owner, String title) throws HeadlessException {
        this(owner, title, true);
    }

    /**
     * Creates a modal or non-modal dialog with the specified title and the specified owner <code>Frame</code>.  If
     * <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the owner of this dialog.
     *
     * @param owner the <code>Frame</code> from which the dialog is displayed
     * @param title the <code>String</code> to display in the dialog's title bar
     * @param modal true for a modal dialog, false for one that allows other windows to be active at the same time
     * @throws HeadlessException if GraphicsEnvironment.isHeadless() returns true.
     * @see java.awt.GraphicsEnvironment#isHeadless
     * @see JComponent#getDefaultLocale
     */
    public MultiplePageDialog(Frame owner, String title, boolean modal) throws HeadlessException {
        this(owner, title, modal, TAB_STYLE);
    }

    public MultiplePageDialog(Frame owner, String title, boolean modal, GraphicsConfiguration gc) {
        super(owner, title, modal, gc);
        setStyle(TAB_STYLE);
    }

    /**
     * Creates a modal or non-modal MultiplePageDialog with the specified style, the specified title and the specified
     * owner <code>Frame</code>.  If <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the
     * owner of this dialog.  All constructors defer to this one.
     *
     * @param owner the <code>Frame</code> from which the dialog is displayed
     * @param title the <code>String</code> to display in the dialog's title bar
     * @param modal true for a modal dialog, false for one that allows other windows to be active at the same time
     * @param style the style. It must be one of the following: TAB_STYLE, ICON_STYLE, LIST_STYLE or TREE_STYLE.
     * @throws HeadlessException if GraphicsEnvironment.isHeadless() returns true.
     * @see java.awt.GraphicsEnvironment#isHeadless
     * @see JComponent#getDefaultLocale
     */
    public MultiplePageDialog(Frame owner, String title, boolean modal, int style) throws HeadlessException {
        super(owner, title, modal);
        setStyle(style);
    }

    public MultiplePageDialog(Frame owner, String title, boolean modal, GraphicsConfiguration gc, int style) {
        super(owner, title, modal, gc);
        setStyle(style);
    }

    public MultiplePageDialog(Dialog owner, int style) {
        super(owner);
        setStyle(style);
    }

    public MultiplePageDialog(Dialog owner, boolean modal, int style) {
        super(owner, modal);
        setStyle(style);
    }

    public MultiplePageDialog(Dialog owner, String title, int style) {
        super(owner, title);
        setStyle(style);
    }

    /**
     * Creates a non-modal MultiplePageDialog without a title with the specified <code>Dialog</code> as its owner.  If
     * <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the owner of the dialog. By
     * default TAB_STYLE is used.
     *
     * @param owner the owner of the dialog
     * @throws HeadlessException if the page does not support mouse or key events.
     */
    public MultiplePageDialog(Dialog owner) throws HeadlessException {
        this(owner, false);
    }

    /**
     * Creates a modal or non-modal MultiplePageDialog without a title and with the specified owner <code>Dialog</code>.
     * If <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the owner of the dialog. By
     * default TAB_STYLE is used.
     *
     * @param owner the <code>Frame</code> from which the dialog is displayed
     * @param modal true for a modal dialog, false for one that allows others windows to be active at the same time
     * @throws HeadlessException if the page does not support mouse or key events.
     */
    public MultiplePageDialog(Dialog owner, boolean modal) throws HeadlessException {
        this(owner, "", modal);
    }

    /**
     * Creates a non-modal MultiplePageDialog with the specified title and with the specified owner frame.  If
     * <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the owner of the dialog.
     *
     * @param owner the <code>Frame</code> from which the dialog is displayed
     * @param title the <code>String</code> to display in the dialog's title bar
     * @throws HeadlessException if GraphicsEnvironment.isHeadless() returns true.
     * @see java.awt.GraphicsEnvironment#isHeadless
     * @see JComponent#getDefaultLocale
     */
    public MultiplePageDialog(Dialog owner, String title) throws HeadlessException {
        this(owner, title, true);
    }

    /**
     * Creates a modal or non-modal dialog with the specified title and the specified owner <code>Dialog</code>.  If
     * <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the owner of this dialog.
     *
     * @param owner the <code>Dialog</code> from which the dialog is displayed
     * @param title the <code>String</code> to display in the dialog's title bar
     * @param modal true for a modal dialog, false for one that allows other windows to be active at the same time
     * @throws HeadlessException if GraphicsEnvironment.isHeadless() returns true.
     * @see java.awt.GraphicsEnvironment#isHeadless
     * @see JComponent#getDefaultLocale
     */
    public MultiplePageDialog(Dialog owner, String title, boolean modal) throws HeadlessException {
        this(owner, title, modal, TAB_STYLE);
    }

    public MultiplePageDialog(Dialog owner, String title, boolean modal, GraphicsConfiguration gc) {
        super(owner, title, modal, gc);
        setStyle(TAB_STYLE);
    }

    public MultiplePageDialog(Window owner) {
        super(owner);
        setStyle(TAB_STYLE);
    }

    public MultiplePageDialog(Window owner, ModalityType modalityType) {
        super(owner, modalityType);
        setStyle(TAB_STYLE);
    }

    public MultiplePageDialog(Window owner, String title) {
        super(owner, title);
        setStyle(TAB_STYLE);
    }

    public MultiplePageDialog(Window owner, String title, ModalityType modalityType) {
        super(owner, title, modalityType);
        setStyle(TAB_STYLE);
    }

    public MultiplePageDialog(Window owner, String title, ModalityType modalityType, GraphicsConfiguration gc) {
        super(owner, title, modalityType, gc);
        setStyle(TAB_STYLE);
    }

    /**
     * Creates a modal or non-modal MultiplePageDialog with the specified style, the specified title and the specified
     * owner <code>Dialog</code>.  If <code>owner</code> is <code>null</code>, a shared, hidden frame will be set as the
     * owner of this dialog.  All constructors defer to this one.
     *
     * @param owner the <code>Dialog</code> from which the dialog is displayed
     * @param title the <code>String</code> to display in the dialog's title bar
     * @param modal true for a modal dialog, false for one that allows other windows to be active at the same time
     * @param style the style. It must be one of the following: TAB_STYLE, ICON_STYLE, LIST_STYLE or TREE_STYLE.
     * @throws HeadlessException if GraphicsEnvironment.isHeadless() returns true.
     * @see java.awt.GraphicsEnvironment#isHeadless
     * @see JComponent#getDefaultLocale
     */
    public MultiplePageDialog(Dialog owner, String title, boolean modal, int style) throws HeadlessException {
        super(owner, title, modal);
        setStyle(style);
    }

    public MultiplePageDialog(Dialog owner, String title, boolean modal, GraphicsConfiguration gc, int style) {
        super(owner, title, modal, gc);
        setStyle(style);
    }

    public MultiplePageDialog(Window owner, int style) {
        super(owner);
        setStyle(style);
    }

    public MultiplePageDialog(Window owner, ModalityType modalityType, int style) {
        super(owner, modalityType);
        setStyle(style);
    }

    public MultiplePageDialog(Window owner, String title, int style) {
        super(owner, title);
        setStyle(style);
    }

    public MultiplePageDialog(Window owner, String title, ModalityType modalityType, int style) {
        super(owner, title, modalityType);
        setStyle(style);
    }

    public MultiplePageDialog(Window owner, String title, ModalityType modalityType, GraphicsConfiguration gc, int style) {
        super(owner, title, modalityType, gc);
        setStyle(style);
    }

    public MultiplePageDialog(int style) {
        setStyle(style);
    }

    public MultiplePageDialog(Frame owner, int style) {
        super(owner);
        setStyle(style);
    }

    public MultiplePageDialog(Frame owner, boolean modal, int style) {
        super(owner, modal);
        setStyle(style);
    }

    public MultiplePageDialog(Frame owner, String title, int style) {
        super(owner, title);
        setStyle(style);
    }

    /**
     * Implements the method in StandardDialog. You can override this method to create a BannerPanel.
     *
     * @return the BannerPanel
     */
    @Override
    public JComponent createBannerPanel() {
        return null;
    }

    /**
     * Implements the method in StandardDialog. You can override this method to create a ContentPanel. By default, a
     * JPanel with BorderLayout is created. IndexPanel is added to WEST and PagesPanel is added to CENTER.
     *
     * @return the ContentPanel
     */
    @Override
    public JComponent createContentPanel() {
        _indexPanel = createIndexPanel();
        _pagesPanel = createPagesPanel();
        if (_pageList.getPageCount() > 0) {
            if (getInitialPageTitle() != null) {
                setCurrentPage(getInitialPageTitle());
            }
            else {
                setCurrentPage(_pageList.getPage(0));
            }
        }
        return setupContentPanel(_indexPanel, _pagesPanel);
    }

    /**
     * Setups the content panel. It will use the index panel and the pages panel created earlier and put it into another
     * panel.
     *
     * @param indexPanel the index panel. It has the nagivation control to control which page to show.
     * @param pagesPanel the pages panel. It contains all the pages of this dialog.
     * @return the panel that contains both index panel and pages panel.
     */
    protected JComponent setupContentPanel(JComponent indexPanel, JComponent pagesPanel) {
        JPanel middlePanel = new JPanel(new BorderLayout(10, 10));
        if (indexPanel != null) {
            middlePanel.add(indexPanel, BorderLayout.BEFORE_LINE_BEGINS);
        }
        if (pagesPanel != null) {
            middlePanel.add(pagesPanel, BorderLayout.CENTER);
        }
        return middlePanel;
    }

    /**
     * Creates the button panel. It has three buttons - OK, Cancel and Apply. If you want to create your own button
     * panel, just override this method.
     *
     * @return button panel
     */
    @Override
    public ButtonPanel createButtonPanel() {
        ButtonPanel buttonPanel = new ButtonPanel();
        _okButton = new JButton();
        _cancelButton = new JButton();
        _applyButton = new JButton();
        _okButton.setName(OK);
        _cancelButton.setName(CANCEL);
        _applyButton.setName(APPLY);
        buttonPanel.addButton(_okButton, ButtonPanel.AFFIRMATIVE_BUTTON);
        buttonPanel.addButton(_cancelButton, ButtonPanel.CANCEL_BUTTON);
        buttonPanel.addButton(_applyButton, ButtonPanel.OTHER_BUTTON);

        Locale l = getLocale();
        _okButton.setAction(new AbstractAction(JideSwingUtilities.getOKString(l)) {
            private static final long serialVersionUID = 7761238902525319363L;

            public void actionPerformed(ActionEvent e) {
                setDialogResult(RESULT_AFFIRMED);
                setVisible(false);
                dispose();
            }
        });
        _cancelButton.setAction(new AbstractAction(JideSwingUtilities.getCancelString(l)) {
            private static final long serialVersionUID = 2671605366801733356L;

            public void actionPerformed(ActionEvent e) {
                setDialogResult(RESULT_CANCELLED);
                setVisible(false);
                dispose();
            }
        });
        _applyButton.setAction(new AbstractAction(ButtonResources.getResourceBundle(Locale.getDefault()).getString("Button.apply")) {
            private static final long serialVersionUID = -7553895212164069062L;

            public void actionPerformed(ActionEvent e) {
                if (getCurrentPage() != null) {
                    getCurrentPage().fireButtonEvent(ButtonEvent.DISABLE_BUTTON, APPLY);
                }
            }
        });
        _applyButton.setMnemonic(ButtonResources.getResourceBundle(Locale.getDefault()).getString("Button.apply.mnemonic").charAt(0));
        _applyButton.setEnabled(false);

        setDefaultCancelAction(_cancelButton.getAction());
        setDefaultAction(_okButton.getAction());
        getRootPane().setDefaultButton(_okButton);
        return buttonPanel;
    }

    /**
     * Gets the OK Button only if you didn't override the createButtonPanel() and remove the OK button.
     *
     * @return the OK Button
     */
    public JButton getOkButton() {
        return _okButton;
    }

    /**
     * Gets the cancel button. only if you didn't override the createButtonPanel() and remove the cancel button.
     *
     * @return the cancel button.
     */
    public JButton getCancelButton() {
        return _cancelButton;
    }

    /**
     * Gets the apply button. only if you didn't override the createButtonPanel() and remove the apply button.
     *
     * @return the apply button.
     */
    public JButton getApplyButton() {
        return _applyButton;
    }

    /**
     * Creates the pages panel. If it's TAB_STYLE, a tabbed pane will be created. If it's any other styles, a JPanel
     * with CardLayout will be created.
     *
     * @return a panel containing all the pages.
     */
    protected JComponent createPagesPanel() {
        if (_style == TAB_STYLE) {
            _tabbedPane = createTabbedPane();
            _tabbedPane.addChangeListener(new ChangeListener() {
                public void stateChanged(ChangeEvent e) {
                    Component selectedComponent = _tabbedPane.getSelectedComponent();
                    if (selectedComponent instanceof AbstractDialogPage) {
                        setCurrentPage((AbstractDialogPage) selectedComponent, _tabbedPane);
                    }
                }
            });
            for (int i = 0; i < _pageList.getPageCount(); i++) {
                AbstractDialogPage page = _pageList.getPage(i);
                page.addButtonListener(getButtonPanel());
                _tabbedPane.addTab(page.getTitle(), page.getIcon(), page, page.getDescription());
                _tabbedPane.setEnabledAt(i, page.isPageEnabled());
                final int index = i;
                page.addPropertyChangeListener(new PropertyChangeListener() {
                    public void propertyChange(PropertyChangeEvent evt) {
                        if (AbstractDialogPage.PROPERTY_PAGE_ENABLED.equals(evt.getPropertyName())) {
                            _tabbedPane.setEnabledAt(index, Boolean.TRUE.equals(evt.getNewValue()));
                        }
                        else if (AbstractDialogPage.ICON_PROPERTY.equals(evt.getPropertyName())) {
                            _tabbedPane.setIconAt(index, (Icon) evt.getNewValue());
                        }
                        else if (AbstractDialogPage.TITLE_PROPERTY.equals(evt.getPropertyName())) {
                            _tabbedPane.setTitleAt(index, (String) evt.getNewValue());
                        }
                        else if (AbstractDialogPage.DESCRIPTION_PROPERTY.equals(evt.getPropertyName())) {
                            _tabbedPane.setToolTipTextAt(index, (String) evt.getNewValue());
                        }
                    }
                });
            }
            _pageList.addListDataListener(new ListDataListener() {
                public void intervalAdded(ListDataEvent e) {
                    for (int i = e.getIndex0(); i <= e.getIndex1(); i++) {
                        AbstractDialogPage page = _pageList.getPage(i);
                        _tabbedPane.insertTab(page.getTitle(), page.getIcon(), page, page.getDescription(), i);
                    }
                }

                public void intervalRemoved(ListDataEvent e) {
                    for (int i = e.getIndex1(); i >= e.getIndex0(); i--) {
                        _tabbedPane.removeTabAt(i);
                    }
                }

                public void contentsChanged(ListDataEvent e) {
                }
            });
            return _tabbedPane;
        }
        else {
            final JPanel pagesPanel = new JPanel();
            _cardLayout = new CardLayout();
            pagesPanel.setLayout(_cardLayout);

            for (int i = 0; i < _pageList.getPageCount(); i++) {
                AbstractDialogPage page = _pageList.getPage(i);
                page.addButtonListener(getButtonPanel());
                page.setName(page.getFullTitle());
                page.addPropertyChangeListener(new PropertyChangeListener() {
                    public void propertyChange(PropertyChangeEvent evt) {
                        if (AbstractDialogPage.TITLE_PROPERTY.equals(evt.getPropertyName())) {
                            for (int j = 0; j < pagesPanel.getComponentCount(); j++) {
                                Component c = pagesPanel.getComponent(j);
                                boolean wasVisible = c.isVisible();
                                Object source = evt.getSource();
                                if (source instanceof AbstractDialogPage && c == source) {
                                    pagesPanel.remove(j);
                                    String fullTitle = ((AbstractDialogPage) source).getFullTitle();
                                    pagesPanel.add((AbstractDialogPage) source, fullTitle, j);
                                    ((AbstractDialogPage) source).setName(fullTitle);
                                    getIndexPanel().repaint();
                                    if (wasVisible) {
                                        _cardLayout.show(pagesPanel, fullTitle);
                                    }
                                    break;
                                }
                            }
                        }
                    }
                });

                pagesPanel.add(page, page.getFullTitle());
            }

            _pageList.addListDataListener(new ListDataListener() {
                public void intervalAdded(ListDataEvent e) {
                    for (int i = e.getIndex0(); i <= e.getIndex1(); i++) {
                        AbstractDialogPage page = _pageList.getPage(i);
                        page.setName(page.getFullTitle());
                        pagesPanel.add(page, page.getFullTitle(), i);
                    }
                }

                public void intervalRemoved(ListDataEvent e) {
                    for (int i = e.getIndex1(); i >= e.getIndex0(); i--) {
                        pagesPanel.remove(i);
                    }
                }

/*
                private void dumpPagesPanel() {
                    for (int i = 0; i < pagesPanel.getComponentCount(); i++) {
                        System.out.println("" + i + ": " + pagesPanel.getComponent(i).getName());
                    }
                }
*/

                public void contentsChanged(ListDataEvent e) {
                    if (e.getSource() instanceof PageList) {
                        Object o = ((PageList) e.getSource()).getSelectedItem();
                        if (o instanceof AbstractDialogPage && ((AbstractDialogPage) o).isPageEnabled()) {
                            setCurrentPage((AbstractDialogPage) o);
                        }
                    }
                }
            });
            return pagesPanel;
        }
    }

    /**
     * Creates the JTabbedPane used by TAB_STYLE dialog.
     *
     * @return a JTabbedPane
     */
    protected JTabbedPane createTabbedPane() {
        return new JTabbedPane(JTabbedPane.TOP);
    }

    /**
     * Creates the index panel based on the style.
     *
     * @return the index panel.
     */
    public JComponent createIndexPanel() {
        switch (_style) {
            case ICON_STYLE:
                return createIconPanel();
            case LIST_STYLE:
                return createListPanel();
            case TREE_STYLE:
                return createTreePanel();
            case TAB_STYLE:
            default:
                return null;
        }
    }

    /**
     * Sets the page list of this dialog. User must call this method before the dialog is set visible.
     *
     * @param pageList the page list
     */
    public void setPageList(PageList pageList) {
        _pageList = pageList;
    }

    /**
     * Gets the page list of this dialog.
     *
     * @return the page list.
     */
    public PageList getPageList() {
        return _pageList;
    }

    /**
     * Gets the current selected page.
     *
     * @return the current selected page.
     */
    public AbstractDialogPage getCurrentPage() {
        return _pageList.getCurrentPage();
    }

    protected void setCurrentPage(String pageTitle) {
        if (_pageList != null) {
            setCurrentPage(_pageList.getPageByFullTitle(pageTitle));
        }
    }

    protected void setCurrentPage(AbstractDialogPage currentPage) {
        setCurrentPage(currentPage, null);
    }

    protected void setCurrentPage(AbstractDialogPage currentPage, Object source) {
        if (!_pageList.setCurrentPage(currentPage, source)) {
            return;
        }

        if (currentPage != null) {
            showCurrentPage(currentPage);
        }
    }

    /**
     * Displays the current page. If it is TAB_STYLE, this method will simply select the tab that has the current page.
     * If it is any of the other styles, this method will show the page that is already added in a CardLayout in
     * createPagePanel method.
     *
     * @param currentPage the current page
     */
    protected void showCurrentPage(AbstractDialogPage currentPage) {
        if (currentPage != null) {
            if (getStyle() == TAB_STYLE) {
                _tabbedPane.setSelectedComponent(currentPage);
            }
            else {
                _cardLayout.show(_pagesPanel, currentPage.getFullTitle());
            }
            currentPage.focusDefaultFocusComponent();
        }
    }

    private JComponent createTreePanel() {
        final DefaultMutableTreeNode root = new DefaultMutableTreeNode("", true);

        _titleNodeMap = new HashMap<String, MutableTreeNode>((int) (_pageList.getPageCount() * 0.75));
        for (int i = 0; i < _pageList.getPageCount(); i++) {
            AbstractDialogPage dialogPage = _pageList.getPage(i);
            addPage(dialogPage, root, false);
        }

        _tree = createTree(root);
        configureTree(_tree);
        _pageList.addListDataListener(new ListDataListener() {
            public void intervalAdded(ListDataEvent e) {
                for (int i = e.getIndex0(); i <= e.getIndex1(); i++) {
                    AbstractDialogPage dialogPage = _pageList.getPage(i);
                    addPage(dialogPage, (DefaultMutableTreeNode) _tree.getModel().getRoot(), true);
                }
            }

            public void intervalRemoved(ListDataEvent e) {
                // compare PageList with TitleNodeMap to find out what is missing
                Set<String> set = _titleNodeMap.keySet();
                Vector<String> toBeRemoved = new Vector<String>();
                for (String title : set) {
                    if (_pageList.getPageByFullTitle(title) == null) {
                        DefaultMutableTreeNode treeNode = (DefaultMutableTreeNode) _titleNodeMap.get(title);
                        if (treeNode != null) {
                            toBeRemoved.add(title);
                            DefaultMutableTreeNode parentNode = (DefaultMutableTreeNode) treeNode.getParent();
                            if (parentNode != null) {
                                int index = parentNode.getIndex(treeNode);
                                parentNode.remove(treeNode);
                                ((DefaultTreeModel) _tree.getModel()).nodesWereRemoved(parentNode, new int[]{index}, new Object[]{treeNode});
                            }
                        }
                    }
                }
                for (String o : toBeRemoved) {
                    _titleNodeMap.remove(o);
                }
            }

            public void contentsChanged(ListDataEvent e) {
                if (e.getIndex0() == -1 && e.getIndex1() == -1 && e.getType() == ListDataEvent.CONTENTS_CHANGED) {
                    if (_titleNodeMap != null && _pageList.getCurrentPage() != null) {
                        TreeNode node = _titleNodeMap.get(_pageList.getCurrentPage().getFullTitle());
                        if (node != null) {
                            ArrayList<TreeNode> list = new ArrayList<TreeNode>();
                            while (node != null) {
                                list.add(0, node);
                                node = node.getParent();
                            }
                            TreePath treePath = new TreePath(list.toArray(new TreeNode[list.size()]));
                            _tree.getSelectionModel().setSelectionPath(treePath);
                        }
                    }
                }
            }
        });

        JComponent indexPanel = new JPanel(new BorderLayout());
        indexPanel.add(new JScrollPane(_tree), BorderLayout.CENTER);
        return indexPanel;
    }

    /**
     * Creates tree that is used in TREE_STYLE dialog's index panel. Below is the code we used. If you just want to have
     * a different cell renderer, you can just call {@link #setTreeCellRenderer(javax.swing.tree.TreeCellRenderer)} to
     * set a new one.
     * <pre><code>
     * UIManager.put("Tree.hash", Color.white);
     * return new JTree(root);
     * </code></pre>
     *
     * @param root the root of the tree
     * @return tree the created JTree instance
     */
    protected JTree createTree(DefaultMutableTreeNode root) {
        UIManager.put("Tree.hash", Color.white);
        return new JTree(root);
    }

    /**
     * Configure the JTree used in TREE_STYLE dialog. Subclass can override this method to configure the JTree to the
     * way you want. Below is the default implementation of this method.
     * <code><pre>
     * tree.setToggleClickCount(1);
     * tree.setCellRenderer(createTreeCellRenderer());
     * tree.setRootVisible(false);
     * tree.setShowsRootHandles(false);
     * tree.addTreeSelectionListener(new TreeSelectionListener() {
     *     public void valueChanged(TreeSelectionEvent e) {
     *         if (tree.getSelectionPath() == null) {
     *             return;
     *         }
     * <p/>
     *         DefaultMutableTreeNode treeNode = (DefaultMutableTreeNode)
     * tree.getSelectionPath().getLastPathComponent();
     *         // comment this while block if you want the parent page shows its own page instead
     * of
     * showing its first child page.
     *         while (!treeNode.isLeaf()) {
     *             final DefaultMutableTreeNode tn = treeNode;
     *             Runnable runnable = new Runnable() {
     *                 public void run() {
     *                     tree.expandPath(new TreePath(tn.getPath()));
     *                 }
     *             };
     *             SwingUtilities.invokeLater(runnable);
     *             treeNode = (DefaultMutableTreeNode) treeNode.getChildAt(0);
     *         }
     * <p/>
     *         if (treeNode != null) {
     *             Object userObject = treeNode.getUserObject();
     *             if (userObject instanceof AbstractDialogPage) {
     *                 setCurrentPage((AbstractDialogPage) userObject, tree);
     *             }
     *         }
     *     }
     * });
     * </pre></code>
     *
     * @param tree the tree to configure
     */
    protected void configureTree(final JTree tree) {
        tree.setToggleClickCount(1);
        tree.setCellRenderer(createTreeCellRenderer());
        tree.setRootVisible(false);
        tree.setShowsRootHandles(false);
        tree.addTreeSelectionListener(new TreeSelectionListener() {
            public void valueChanged(TreeSelectionEvent e) {
                if (tree.getSelectionPath() == null) {
                    return;
                }

                DefaultMutableTreeNode treeNode = (DefaultMutableTreeNode) tree.getSelectionPath().getLastPathComponent();

                // comment this while block if you want the parent page shows its own page instead of showing its first child page.
                while (!treeNode.isLeaf()) {
                    final DefaultMutableTreeNode tn = treeNode;
                    Runnable runnable = new Runnable() {
                        public void run() {
                            tree.expandPath(new TreePath(tn.getPath()));
                        }
                    };
                    SwingUtilities.invokeLater(runnable);
                    treeNode = (DefaultMutableTreeNode) treeNode.getChildAt(0);
                }

                Object userObject = treeNode.getUserObject();
                if (userObject instanceof AbstractDialogPage && !userObject.equals(getCurrentPage()) && ((AbstractDialogPage) userObject).isPageEnabled()) {
                    setCurrentPage((AbstractDialogPage) userObject, tree);
                    if (getCurrentPage() != userObject) {
                        // TODO select the old path.
                    }
                }
            }
        });
    }

    private void addPage(final AbstractDialogPage dialogPage, final DefaultMutableTreeNode root, boolean fireEvent) {
        if (dialogPage == null) {
            return;
        }

        final MutableTreeNode treeNode = createTreeNode(dialogPage);
        if (treeNode instanceof MutableTreeNodeEx) {
            ((MutableTreeNodeEx) treeNode).setEnabled(dialogPage.isPageEnabled());
        }
        if (dialogPage.getParentPage() == null) {
            _titleNodeMap.put(dialogPage.getFullTitle(), treeNode);
            root.add(treeNode);
            if (fireEvent) {
                ((DefaultTreeModel) _tree.getModel()).nodesWereInserted(root, new int[]{root.getIndex(treeNode)});
            }
        }
        else {
            _titleNodeMap.put(dialogPage.getFullTitle(), treeNode);
            DefaultMutableTreeNode parentNode = (DefaultMutableTreeNode) _titleNodeMap.get(dialogPage.getParentPage().getFullTitle());
            if (parentNode != null) {
                parentNode.add(treeNode);
                if (fireEvent) {
                    ((DefaultTreeModel) _tree.getModel()).nodesWereInserted(parentNode, new int[]{parentNode.getIndex(treeNode)});
                }
            }
        }
        dialogPage.addPropertyChangeListener(new PropertyChangeListener() {
            public void propertyChange(PropertyChangeEvent evt) {
                if (AbstractDialogPage.ICON_PROPERTY.equals(evt.getPropertyName()) && treeNode instanceof MutableTreeNodeEx) {
                    ((MutableTreeNodeEx) treeNode).setEnabled(dialogPage.isPageEnabled());
                }
            }
        });
    }

    /**
     * Create tree node for TREE_STYLE pages.
     *
     * @param dialogPage the corresponding dialog page.
     * @return the tree node.
     */
    protected MutableTreeNode createTreeNode(AbstractDialogPage dialogPage) {
        return new MutableTreeNodeEx(dialogPage);
    }

/*
    private void removePage(AbstractDialogPage dialogPage, final DefaultMutableTreeNode root, boolean fireEvent) {
        if (dialogPage == null) {
            return;
        }

        DefaultMutableTreeNode treeNode = (DefaultMutableTreeNode) _titleNodeMap.get(dialogPage.getFullTitle());

        if (treeNode == null) {
            return;
        }

        if (treeNode.getChildCount() > 0) {
            throw new IllegalArgumentException("Please remove all children pages before removing parent page \"" + dialogPage.getFullTitle() + "\"");
        }
        _titleNodeMap.remove(dialogPage.getFullTitle());
        if (dialogPage.getParentPage() == null) {
            int index = root.getIndex(treeNode);
            root.remove(treeNode);
            if (fireEvent) {
                ((DefaultTreeModel) _tree.getModel()).nodesWereRemoved(root, new int[]{index}, new Object[]{treeNode});
            }
        }
        else {
            DefaultMutableTreeNode parentNode = (DefaultMutableTreeNode) _titleNodeMap.get(dialogPage.getParentPage().getFullTitle());
            if (parentNode != null) {
                int index = parentNode.getIndex(treeNode);
                parentNode.remove(treeNode);
                if (fireEvent) {
                    ((DefaultTreeModel) _tree.getModel()).nodesWereRemoved(parentNode, new int[]{index}, new Object[]{treeNode});
                }
            }
        }
    }
*/

    private JComponent createListPanel() {
        final DefaultListModel listModel = new DefaultListModel();
        for (int i = 0; i < _pageList.getPageCount(); i++) {
            AbstractDialogPage optionsPanel = _pageList.getPage(i);
            listModel.addElement(optionsPanel);
        }

        final JList list = createList(listModel);
        if (list.getModel().getSize() > 0) {
            list.setSelectedIndex(0);
        }
        list.addListSelectionListener(new ListSelectionListener() {
            public void valueChanged(ListSelectionEvent e) {
                if (list.getSelectedValue() == getCurrentPage()) {
                    return;
                }
                if (!e.getValueIsAdjusting()) {
                    AbstractDialogPage page = (AbstractDialogPage) list.getSelectedValue();
                    if (page != null) {
                        setCurrentPage(page, list);
                        if (getCurrentPage() != page) {
                            list.setSelectedValue(getCurrentPage(), true);
                        }
                    }
                    else {
                        list.setSelectedIndex(e.getLastIndex());
                    }
                }
            }
        });
        list.setBorder(BorderFactory.createEmptyBorder(2, 2, 2, 10));

        _pageList.addListDataListener(new ListDataListener() {
            public void intervalAdded(ListDataEvent e) {
                for (int i = e.getIndex0(); i <= e.getIndex1(); i++) {
                    AbstractDialogPage optionsPanel = _pageList.getPage(i);
                    listModel.add(i, optionsPanel);
                }
            }

            public void intervalRemoved(ListDataEvent e) {
                for (int i = e.getIndex1(); i >= e.getIndex0(); i--) {
                    listModel.remove(i);
                }
            }

            public void contentsChanged(ListDataEvent e) {
                if (e.getIndex0() == -1 && e.getIndex1() == -1 && e.getType() == ListDataEvent.CONTENTS_CHANGED) {
                    if (_pageList.getCurrentPage() != null) {
                        int index = _pageList.getPageIndexByFullTitle(_pageList.getCurrentPage().getFullTitle());
                        list.setSelectedIndex(index);
                    }
                }
            }
        });

        JComponent indexPanel = new JPanel(new BorderLayout(4, 4));
        indexPanel.add(new JideScrollPane(list), BorderLayout.CENTER);
        indexPanel.setOpaque(false);
        return indexPanel;
    }

    /**
     * Creates list that is used in LIST_STYLE dialog's index panel. Below is the code we used. If you just want to have
     * a different cell renderer, you can just call {@link #setListCellRenderer(javax.swing.ListCellRenderer)} to set a
     * new one.
     * <pre><code>
     * JList list = new JList(listModel);
     * list.setCellRenderer(createListCellRenderer());
     * return list;
     * </code></pre>
     *
     * @param listModel the list model
     * @return list.
     */
    protected JList createList(DefaultListModel listModel) {
        JList list = new JList(listModel);
        list.setCellRenderer(createListCellRenderer());
        list.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
        return list;
    }

    /**
     * Creates the panel that contains several icons. Each icon represents for a page. This is only used for
     * ICON_STYLE.
     *
     * @return a panel that contains several icons.
     */
    protected JComponent createIconPanel() {
        final ButtonPanel buttonsPanel = createIconButtonPanel();
        buttonsPanel.setGroupGap(0);
        buttonsPanel.setButtonGap(0);

        final ButtonGroup group = new ButtonGroup();
        for (int i = 0; i < _pageList.getPageCount(); i++) {
            final AbstractDialogPage optionsPanel = _pageList.getPage(i);
            final JideButton button = createIconButton(optionsPanel.getTitle(), optionsPanel.getIcon());
            button.setToolTipText(optionsPanel.getDescription());
            button.setEnabled(optionsPanel.isPageEnabled());
            button.addActionListener(new AbstractAction() {
                private static final long serialVersionUID = 4451059166068761678L;

                public void actionPerformed(ActionEvent e) {
                    setCurrentPage(optionsPanel, buttonsPanel);
                    if (getCurrentPage() == optionsPanel) {
                        group.setSelected(button.getModel(), true);
                    }
                }
            });
            optionsPanel.addPropertyChangeListener(new PropertyChangeListener() {
                public void propertyChange(PropertyChangeEvent evt) {
                    if (AbstractDialogPage.PROPERTY_PAGE_ENABLED.equals(evt.getPropertyName())) {
                        button.setEnabled(Boolean.TRUE.equals(evt.getNewValue()));
                    }
                    else if (AbstractDialogPage.ICON_PROPERTY.equals(evt.getPropertyName())) {
                        button.setIcon((Icon) evt.getNewValue());
                    }
                    else if (AbstractDialogPage.TITLE_PROPERTY.equals(evt.getPropertyName())) {
                        button.setText((String) evt.getNewValue());
                    }
                    else if (AbstractDialogPage.DESCRIPTION_PROPERTY.equals(evt.getPropertyName())) {
                        button.setToolTipText((String) evt.getNewValue());
                    }
                }
            });
            buttonsPanel.addButton(button);
            group.add(button);
            if (_pageList.getPageCount() > 0) {
                if (getInitialPageTitle() != null && getInitialPageTitle().equals(optionsPanel.getFullTitle())) {
                    group.setSelected(button.getModel(), true);
                }
                else if (getInitialPageTitle() == null && i == 0) {
                    group.setSelected(button.getModel(), true);
                }
            }
        }

        buttonsPanel.setOpaque(false);
        buttonsPanel.setBorder(BorderFactory.createEmptyBorder(10, 10, 10, 10));

        final JScrollPane pane = new JScrollPane(buttonsPanel) {
            private static final long serialVersionUID = -5872376661587310320L;

            @Override
            public Dimension getPreferredSize() {
                if (buttonsPanel.getAlignment() == SwingConstants.TOP || buttonsPanel.getAlignment() == SwingConstants.BOTTOM)
                    return new Dimension(buttonsPanel.getPreferredSize().width + getVerticalScrollBar().getPreferredSize().width, 5);
                else
                    return new Dimension(5, buttonsPanel.getPreferredSize().height + getHorizontalScrollBar().getPreferredSize().height);
            }

            @Override
            public Dimension getMinimumSize() {
                return getPreferredSize();
            }
        };

        if (buttonsPanel.getAlignment() == SwingConstants.TOP || buttonsPanel.getAlignment() == SwingConstants.BOTTOM)
            pane.setHorizontalScrollBarPolicy(JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
        else
            pane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_NEVER);

        buttonsPanel.setOpaque(false);

        _pageList.addListDataListener(new ListDataListener() {
            public void intervalAdded(ListDataEvent e) {
                for (int i = e.getIndex0(); i <= e.getIndex1(); i++) {
                    addPage(i, group, buttonsPanel);
                }
                buttonsPanel.invalidate();
                buttonsPanel.doLayout();
            }

            public void intervalRemoved(ListDataEvent e) {
                for (int i = e.getIndex1(); i >= e.getIndex0(); i--) {
                    AbstractButton button = (AbstractButton) buttonsPanel.getComponent(i);
                    buttonsPanel.remove(button);
                    group.remove(button);
                }
                buttonsPanel.invalidate();
                buttonsPanel.doLayout();
            }

            public void contentsChanged(ListDataEvent e) {
                if (e.getIndex0() == -1 && e.getIndex1() == -1 && e.getType() == ListDataEvent.CONTENTS_CHANGED) {
                    AbstractButton button = (AbstractButton) buttonsPanel.getButtonByName(_pageList.getCurrentPage().getTitle());
                    if (button != null) {
                        group.setSelected(button.getModel(), true);
                    }
                }
            }
        });

        pane.getViewport().setOpaque(false);
        return pane;
    }

    /**
     * Creates the ButtonPanel used by IconPanel. By default, we create it using <code>new
     * ScrollableButtonPanel(SwingConstants.TOP, ButtonPanel.SAME_SIZE)</code>.
     *
     * @return the ButtonPanel.
     */
    protected ButtonPanel createIconButtonPanel() {
        return new ScrollableButtonPanel(SwingConstants.TOP, ButtonPanel.SAME_SIZE);
    }

    private JideButton addPage(int i, final ButtonGroup group, final ButtonPanel buttonsPanel) {
        AbstractDialogPage optionsPanel = _pageList.getPage(i);
        final JideButton button = createIconButton(optionsPanel.getTitle(), optionsPanel.getIcon());
        button.addActionListener(new AbstractAction(optionsPanel.getTitle(), optionsPanel.getIcon()) {
            private static final long serialVersionUID = 5987367362274303556L;

            public void actionPerformed(ActionEvent e) {
                group.setSelected(button.getModel(), true);
                setCurrentPage(_pageList.getPageByFullTitle(e.getActionCommand()), buttonsPanel);
            }
        });
        buttonsPanel.addButton(button, i);
        group.add(button);
        return button;
    }

    /**
     * Creates the button for each icon.
     *
     * @param title the button title
     * @param icon  the button icon
     * @return the button
     */
    protected JideButton createIconButton(String title, Icon icon) {
        final JideButton button = new JideButton(title, icon);
        button.setName(title);
        button.setHorizontalAlignment(SwingConstants.CENTER);
        button.setVerticalTextPosition(SwingConstants.BOTTOM);
        button.setHorizontalTextPosition(SwingConstants.CENTER);
        button.setRequestFocusEnabled(false);
        button.setFocusable(false);
        return button;
    }

    /**
     * Gets the style of this dialog.
     *
     * @return the style. It can be TAB_STYLE, ICON_STYLE, LIST_STYLE or TREE_STYLE.
     */
    public int getStyle() {
        return _style;
    }

    /**
     * Sets the style of this dialog. This class doesn't support change style on fly. You can only change style before
     * the dialog is set to visible.
     *
     * @param style It must be one of the following: TAB_STYLE, ICON_STYLE, LIST_STYLE or TREE_STYLE.
     */
    public void setStyle(int style) {
        if (style == TAB_STYLE || style == LIST_STYLE || style == ICON_STYLE || style == TREE_STYLE) {
            _style = style;
        }
        else {
            throw new IllegalArgumentException("The value of style must be one of the following - TAB_STYLE, ICON_STYLE, LIST_STYLE or TREE_STYLE");
        }
    }

    /**
     * Gets the index panel.
     *
     * @return the index panel.
     */
    public JComponent getIndexPanel() {
        return _indexPanel;
    }

    /**
     * Gets the pages panel.
     *
     * @return the pages panel.
     */
    public JComponent getPagesPanel() {
        return _pagesPanel;
    }

    /**
     * Gets the cell renderer used by the tree. It's used only when the style is TREE_STYLE.
     *
     * @return the tree cell renderer.
     */
    protected TreeCellRenderer getTreeCellRenderer() {
        return _treeCellRenderer;
    }

    /**
     * Sets the tree cell renderer that will be used by JTree when the style is TREE_STYLE.
     *
     * @param treeCellRenderer the tree cell renderer
     */
    public void setTreeCellRenderer(TreeCellRenderer treeCellRenderer) {
        _treeCellRenderer = treeCellRenderer;
    }

    /**
     * Gets the cell renderer used by the list. It's used only when the style is LIST_STYLE.
     *
     * @return the list cell renderer.
     */
    protected ListCellRenderer getListCellRenderer() {
        return _listCellRenderer;
    }

    /**
     * Sets the list cell renderer that will be used by JList when the style is LIST_STYLE.
     *
     * @param listCellRenderer the list cell renderer
     */
    public void setListCellRenderer(ListCellRenderer listCellRenderer) {
        _listCellRenderer = listCellRenderer;
    }

    /**
     * Creates a list cell renderer used by list in LIST_STYLE dialog's index panel.
     *
     * @return the list cell renderer.
     */
    protected ListCellRenderer createListCellRenderer() {
        if (getListCellRenderer() == null) {
            setListCellRenderer(new DialogPageListCellRenderer());
        }
        return getListCellRenderer();
    }

    /**
     * Creates the tree cell renderer used by tree in TREE_STYLE dialog's index panel.
     *
     * @return the tree cell renderer.
     */
    protected TreeCellRenderer createTreeCellRenderer() {
        if (getTreeCellRenderer() == null) {
            setTreeCellRenderer(new DialogPageTreeCellRenderer());
        }
        return getTreeCellRenderer();
    }

    /**
     * Gets the initial page title. Initial page is the page that will be selected when the dialog is just opened.
     * Please note the title is the full title. In most case it's just the title of the page. Only in TREE_STYLE, it
     * should be a list of titles that concats with '.'.
     *
     * @return the initial page title.
     */
    public String getInitialPageTitle() {
        return _initialPageTitle;
    }

    /**
     * Sets the initial page title. Initial page is the page that will be selected when the dialog.
     *
     * @param initialPageTitle the initial page title
     */
    public void setInitialPageTitle(String initialPageTitle) {
        _initialPageTitle = initialPageTitle;
    }
}
TOP

Related Classes of com.jidesoft.dialog.MultiplePageDialog

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.