JCardPane
Aus Byte-Welt Wiki
Version vom 16. März 2011, 11:32 Uhr von AndreUhres (Diskussion | Beiträge)
* JCardPane.java * wraps a JTabbedPane, hiding the tabs. * The main method shows an example. */
import java.awt.*; import java.awt.event.*; import java.beans.*; import java.util.*; import java.util.logging.*; import javax.swing.*; import javax.swing.event.*; import javax.swing.plaf.*; import javax.swing.plaf.basic.*;
public class JCardPane extends JComponent {
private JTabbedPane pane; private CardAction nextAction; private CardAction previousAction; private TabFocusHandler tabFocusHandler; private TabbedPaneUI uiNoTabs = new BasicTabbedPaneUI() {
@Override protected int calculateTabAreaHeight(int tabPlacement, int horizRunCount, int maxTabHeight) { return 0; }
@Override protected Insets getContentBorderInsets(int tabPlacement) { return new Insets(0, 0, 0, 0); }
@Override protected MouseListener createMouseListener() { return null; }
@Override protected void installKeyboardActions() { }
@Override protected int calculateTabWidth(int tabPlacement, int tabIndex, FontMetrics metrics) { return 0; } };
/** * Creates an empty cardpane */ public JCardPane() { pane = new JTabbedPane(); tabFocusHandler = new TabFocusHandler(pane); hideTabs(); pane.addChangeListener(new ChangeListener() {
@Override public void stateChanged(final ChangeEvent e) { if (nextAction != null) { nextAction.setEnabled(hasNext()); } if (previousAction != null) { previousAction.setEnabled(hasPrevious()); } } }); setLayout(new BorderLayout()); super.add(pane); }
/** * Flips to the first card of the cardpane. * @return 0 if this cardpane has a card * or -1 if this cardpane has no card */ public int first() { return show(0); }
/** * Flips to the last card of the cardpane. * @return 0 if this cardpane has a card * or -1 if this cardpane has no card */ public int last() { return show(-1); }
/** * Flips to the next card of the cardpane. * If the currently visible card is the last one, * this method flips to the first card of the cardpane. * @return the index of the card * or -1 if this cardpane has no card */ public int next() { return show(pane.getSelectedIndex() + 1); }
/** * Flips to the previous card of the cardpane. * If the currently visible card is the first one, * this method flips to the last card of the cardpane. * @return the index of the card * or -1 if this cardpane has no card */ public int previous() { return show(pane.getSelectedIndex() - 1); }
/** * Flips to the card at the specified index. * If the index is positive but does not exist, * this method flips to the first card of the cardpane. * If the index is negative, * this method flips to the last card of the cardpane. * @return the index of the card * or -1 if this cardpane has no card */ public int show(final int index) { if (pane.getTabCount() == 0) { return -1; } if (index >= pane.getTabCount()) { pane.setSelectedIndex(0); return 0;
} if (index < 0) { pane.setSelectedIndex(pane.getTabCount() - 1); return pane.getTabCount() - 1;
} pane.setSelectedIndex(index); return index; }
/** * Flips to the card with the specified name. * If the name does not exist, nothing happens. * @return the index of the card * or -1 if the name does not exist */ public int show(final String name) { int index = pane.indexOfTab(name); if (index >= 0) { pane.setSelectedIndex(index); } return index; }
/** * Flips to the specified card. * If the card does not exist in the cardpane, it is added to it. * @return the index of the specified card */ public int show(final JComponent card) { int index = pane.indexOfComponent(card); if (index < 0) { index = pane.getTabCount(); pane.addTab(card.getName(), card); } pane.setSelectedIndex(index); return index; }
/** * Flips to the specified card. * If the card does not exist in the cardpane, it is added to it. * @return the index of the specified card */ public int add(final JComponent card) { return show(card); }
/** * Inserts the specified card at the sepcified index and * flips to the specified card. */ public void insertAt(JComponent card, int index){ pane.insertTab(null, null, card, null, index); pane.setSelectedIndex(index); }
/** * @return true if there is a card after the visible one */ public boolean hasNext() { return pane.getSelectedIndex() < pane.getComponentCount() - 1; }
/** * @return true if there is a card before the visible one */ public boolean hasPrevious() { return pane.getSelectedIndex() > 0; }
/** * @return the visible card */ public Component getCurrent() { return pane.getSelectedComponent(); }
/**
* Adds a ChangeListener
to this cardpane.
*/
public void addChangeListener(final ChangeListener l) {
pane.addChangeListener(l);
}
/**
* Removes a ChangeListener
from this cardpane.
*/
public void removeChangeListener(final ChangeListener l) {
pane.removeChangeListener(l);
}
/** * gets the action to be used for a "Next" button */ public Action getNextAction(final String name) { if (nextAction == null) { nextAction = new CardAction(name, true); nextAction.putValue(Action.MNEMONIC_KEY, (int) name.charAt(0)); nextAction.setEnabled(hasNext()); } return nextAction; }
/** * gets the action to be used for a "Previous" button */ public Action getPreviousAction(final String name) { if (previousAction == null) { previousAction = new CardAction(name, false); previousAction.putValue(Action.MNEMONIC_KEY, (int) name.charAt(0)); previousAction.setEnabled(hasPrevious()); } return previousAction; }
/** * * retain: if true, the component focused within a card will remain focused * each time it becomes visible, otherwise the first component of each card is focused */ public void retainFocus(final boolean retain) { int focusPolicy = TabFocusHandler.RESET_FOCUS; if (retain) { focusPolicy = TabFocusHandler.RETAIN_FOCUS; } tabFocusHandler.uninstall(); tabFocusHandler = new TabFocusHandler(pane, focusPolicy); }
private void hideTabs() { pane.setUI(uiNoTabs); }
private class CardAction extends AbstractAction {
private boolean isNext;
CardAction(final String text, final boolean isNext) { super(text); this.isNext = isNext; putValue(Action.SHORT_DESCRIPTION, getValue(Action.NAME)); }
@Override public void actionPerformed(final ActionEvent e) { if (isNext) { next(); } else { previous(); } } }
/** * Manage the focus when a new tab is selected. You can select a focus policy: * a) Reset Focus - focus is reset to the first focusable component on the tab * b) Retain Focus - focus returns to the last component with focus on the tab * * In addition you add tabs that you want to exclude from the focus policy, * in which case the other policy will be in effect. */ private class TabFocusHandler {
public final static int RESET_FOCUS = 0; public final static int RETAIN_FOCUS = 1; private HashMap<Component, Component> tabFocus = new HashMap<Component, Component>(16); private HashSet<Component> exceptions; private JTabbedPane tabbedPane; private int focusPolicy; private final KeyboardFocusManager focusManager; private final ChangeListener changeListener; private final PropertyChangeListener propertyChangeListener;
/** * Create with the default Retain Focus policy */ TabFocusHandler(final JTabbedPane tabbedPane) { this(tabbedPane, RETAIN_FOCUS); }
/** * Create using the specified focus policy */ TabFocusHandler(final JTabbedPane tabbedPane, final int focusPolicy) { if (focusPolicy != RESET_FOCUS && focusPolicy != RETAIN_FOCUS) { throw new IllegalArgumentException("Invalid focus policy"); } this.tabbedPane = tabbedPane; this.focusPolicy = focusPolicy; // Add listeners to manage a tab change changeListener = new TabChangeListener(); propertyChangeListener = new TabPropertyChangeListener(); tabbedPane.addChangeListener(changeListener); focusManager = KeyboardFocusManager.getCurrentKeyboardFocusManager(); focusManager.addPropertyChangeListener("permanentFocusOwner", propertyChangeListener); }
public void uninstall() { tabbedPane.removeChangeListener(changeListener); focusManager.removePropertyChangeListener(propertyChangeListener); }
/** * Specify a tab with an exception to the focus policy rule */ public void addException(final int index) { if (exceptions == null) { exceptions = new HashSet<Component>(16); } Component key = tabbedPane.getComponentAt(index); exceptions.add(key); }
private class TabPropertyChangeListener implements PropertyChangeListener {
TabPropertyChangeListener() { }
/** * Track focus changes and update the current focus component * for the current tab */ @Override public void propertyChange(final PropertyChangeEvent e) { // No need to track focus change if (exceptions == null && focusPolicy == RESET_FOCUS) { return; } Component key = tabbedPane.getComponentAt(tabbedPane.getSelectedIndex()); if (exceptions != null) { if (focusPolicy == RESET_FOCUS && !exceptions.contains(key)) { return; } if (focusPolicy == RETAIN_FOCUS && exceptions.contains(key)) { return; } } Component value = (Component) e.getNewValue(); if (value != null && SwingUtilities.isDescendingFrom(value, key)) { tabFocus.put(key, value); } } }
private class TabChangeListener implements ChangeListener {
TabChangeListener() { }
/** * Tab has changed. Focus on saved component for the given tab. * When there is no saved component, focus on the first component. */ @Override public void stateChanged(final ChangeEvent e) { Component key = tabbedPane.getComponentAt(tabbedPane.getSelectedIndex()); if (key == null) { return; } Component value = tabFocus.get(key); // First time selecting this tab or focus policy is RESET_FOCUS if (value == null) { key.transferFocus(); tabFocus.put(key, value); } else // Use the saved component for focusing { value.requestFocusInWindow(); } } } }
//for testing only: public static void main(final String... args) { try { UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()); } catch (ClassNotFoundException ex) { Logger.getLogger(JCardPane.class.getName()).log(Level.SEVERE, null, ex); } catch (InstantiationException ex) { Logger.getLogger(JCardPane.class.getName()).log(Level.SEVERE, null, ex); } catch (IllegalAccessException ex) { Logger.getLogger(JCardPane.class.getName()).log(Level.SEVERE, null, ex); } catch (UnsupportedLookAndFeelException ex) { Logger.getLogger(JCardPane.class.getName()).log(Level.SEVERE, null, ex); } Runnable gui = new Runnable() {
@Override public void run() { final JFrame f = new JFrame("JCardPane Demo"); f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); f.setSize(400, 150); f.setLocationRelativeTo(null); final JCardPane cardPane = new JCardPane(); f.add(cardPane, BorderLayout.CENTER); class Card extends JPanel {
Card(final String name) { JTextField tf = new JTextField(20); tf.setText(name); add(tf); tf = new JTextField(20); tf.setText(name); add(tf); setName(name); } } for (int i = 0; i < 5; i++) { cardPane.show(new Card(String.valueOf(i + 1))); } cardPane.first(); JPanel control = new JPanel(); control.add(new JButton(cardPane.getPreviousAction("Previous"))); control.add(new JButton(cardPane.getNextAction("Next"))); f.add(control, BorderLayout.SOUTH); f.setVisible(true); } }; //GUI must start on EventDispatchThread: SwingUtilities.invokeLater(gui); }
} </code=java>