JCardPane: Unterschied zwischen den Versionen
Aus Byte-Welt Wiki
Zur Navigation springenZur Suche springenZeile 2: | Zeile 2: | ||
/* | /* | ||
* JCardPane.java | * JCardPane.java | ||
− | * wraps a JTabbedPane, | + | * wraps a JTabbedPane, hiding the tabs. |
* The main method shows an example. | * The main method shows an example. | ||
*/ | */ | ||
Zeile 21: | Zeile 21: | ||
private CardAction previousAction; | private CardAction previousAction; | ||
private TabFocusHandler tabFocusHandler; | private TabFocusHandler tabFocusHandler; | ||
− | |||
private TabbedPaneUI uiNoTabs = new BasicTabbedPaneUI() { | private TabbedPaneUI uiNoTabs = new BasicTabbedPaneUI() { | ||
Zeile 50: | Zeile 49: | ||
}; | }; | ||
+ | /** | ||
+ | * Creates an empty cardpane | ||
+ | */ | ||
public JCardPane() { | public JCardPane() { | ||
− | |||
− | |||
− | |||
− | |||
pane = new JTabbedPane(); | pane = new JTabbedPane(); | ||
tabFocusHandler = new TabFocusHandler(pane); | tabFocusHandler = new TabFocusHandler(pane); | ||
− | + | hideTabs(); | |
− | |||
− | |||
− | |||
pane.addChangeListener(new ChangeListener() { | pane.addChangeListener(new ChangeListener() { | ||
Zeile 76: | Zeile 71: | ||
} | } | ||
+ | /** | ||
+ | * 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() { | public int first() { | ||
return show(0); | 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() { | public int last() { | ||
return show(-1); | 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() { | public int next() { | ||
return show(pane.getSelectedIndex() + 1); | 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() { | public int previous() { | ||
return show(pane.getSelectedIndex() - 1); | 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) { | public int show(final int index) { | ||
if (pane.getTabCount() == 0) { | if (pane.getTabCount() == 0) { | ||
Zeile 110: | Zeile 138: | ||
} | } | ||
+ | /** | ||
+ | * 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) { | public int show(final String name) { | ||
int index = pane.indexOfTab(name); | int index = pane.indexOfTab(name); | ||
Zeile 118: | Zeile 152: | ||
} | } | ||
+ | /** | ||
+ | * 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) { | public int show(final JComponent card) { | ||
int index = pane.indexOfComponent(card); | int index = pane.indexOfComponent(card); | ||
Zeile 128: | Zeile 167: | ||
} | } | ||
+ | /** | ||
+ | * @return true if there is a card after the visible one | ||
+ | */ | ||
public boolean hasNext() { | public boolean hasNext() { | ||
return pane.getSelectedIndex() < pane.getComponentCount() - 1; | return pane.getSelectedIndex() < pane.getComponentCount() - 1; | ||
} | } | ||
+ | /** | ||
+ | * @return true if there is a card before the visible one | ||
+ | */ | ||
public boolean hasPrevious() { | public boolean hasPrevious() { | ||
return pane.getSelectedIndex() > 0; | return pane.getSelectedIndex() > 0; | ||
} | } | ||
+ | /** | ||
+ | * @return the visible card | ||
+ | */ | ||
public Component getCurrent() { | public Component getCurrent() { | ||
return pane.getSelectedComponent(); | return pane.getSelectedComponent(); | ||
} | } | ||
+ | /** | ||
+ | * Adds a <code>ChangeListener</code> to this cardpane. | ||
+ | */ | ||
public void addChangeListener(ChangeListener l) { | public void addChangeListener(ChangeListener l) { | ||
pane.addChangeListener(l); | pane.addChangeListener(l); | ||
} | } | ||
+ | /** | ||
+ | * Removes a <code>ChangeListener</code> from this cardpane. | ||
+ | */ | ||
public void removeChangeListener(ChangeListener l) { | public void removeChangeListener(ChangeListener l) { | ||
pane.removeChangeListener(l); | pane.removeChangeListener(l); | ||
} | } | ||
− | + | /** | |
− | + | * gets the action to be used for a "Next" button | |
− | + | */ | |
− | |||
− | |||
− | |||
− | |||
− | |||
public Action getNextAction() { | public Action getNextAction() { | ||
if (nextAction == null) { | if (nextAction == null) { | ||
Zeile 162: | Zeile 211: | ||
nextAction.setEnabled(hasNext()); | nextAction.setEnabled(hasNext()); | ||
} | } | ||
− | |||
return nextAction; | return nextAction; | ||
} | } | ||
+ | /** | ||
+ | * gets the action to be used for a "Previous" button | ||
+ | */ | ||
public Action getPreviousAction() { | public Action getPreviousAction() { | ||
if (previousAction == null) { | if (previousAction == null) { | ||
Zeile 172: | Zeile 223: | ||
previousAction.setEnabled(hasPrevious()); | previousAction.setEnabled(hasPrevious()); | ||
} | } | ||
− | |||
return previousAction; | 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(boolean retain) { | public void retainFocus(boolean retain) { | ||
int focusPolicy = TabFocusHandler.RESET_FOCUS; | int focusPolicy = TabFocusHandler.RESET_FOCUS; | ||
Zeile 185: | Zeile 240: | ||
} | } | ||
− | class CardAction extends AbstractAction { | + | private void hideTabs() { |
+ | pane.setUI(uiNoTabs); | ||
+ | } | ||
+ | |||
+ | private class CardAction extends AbstractAction { | ||
private boolean isNext; | private boolean isNext; | ||
Zeile 204: | Zeile 263: | ||
} | } | ||
− | /* | + | /** |
* Manage the focus when a new tab is selected. You can select a focus policy: | * 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 | * a) Reset Focus - focus is reset to the first focusable component on the tab | ||
Zeile 212: | Zeile 271: | ||
* in which case the other policy will be in effect. | * in which case the other policy will be in effect. | ||
*/ | */ | ||
− | class TabFocusHandler implements ChangeListener, PropertyChangeListener { | + | private class TabFocusHandler implements ChangeListener, PropertyChangeListener { |
public final static int RESET_FOCUS = 0; | public final static int RESET_FOCUS = 0; | ||
Zeile 222: | Zeile 281: | ||
private final KeyboardFocusManager focusManager; | private final KeyboardFocusManager focusManager; | ||
− | /* | + | /** |
* Create with the default Retain Focus policy | * Create with the default Retain Focus policy | ||
*/ | */ | ||
Zeile 229: | Zeile 288: | ||
} | } | ||
− | /* | + | /** |
* Create using the specified focus policy | * Create using the specified focus policy | ||
*/ | */ | ||
Zeile 248: | Zeile 307: | ||
focusManager.removePropertyChangeListener(this); | focusManager.removePropertyChangeListener(this); | ||
} | } | ||
− | /* | + | |
+ | /** | ||
* Specify a tab with an exception to the focus policy rule | * Specify a tab with an exception to the focus policy rule | ||
*/ | */ | ||
− | |||
public void addException(int index) { | public void addException(int index) { | ||
if (exceptions == null) { | if (exceptions == null) { | ||
Zeile 260: | Zeile 319: | ||
} | } | ||
− | /* | + | /** |
− | * Tab has changed. Focus on saved component for the given tab. | + | * Tab has changed. Focus on saved component for the given tab. |
− | * | + | * When there is no saved component, focus on the first component. |
*/ | */ | ||
public void stateChanged(ChangeEvent e) { | public void stateChanged(ChangeEvent e) { | ||
Zeile 280: | Zeile 339: | ||
} | } | ||
− | /* | + | /** |
* Track focus changes and update the current focus component | * Track focus changes and update the current focus component | ||
* for the current tab | * for the current tab | ||
Zeile 343: | Zeile 402: | ||
control.add(new JButton(cardPane.getPreviousAction())); | control.add(new JButton(cardPane.getPreviousAction())); | ||
control.add(new JButton(cardPane.getNextAction())); | control.add(new JButton(cardPane.getNextAction())); | ||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
f.add(control, BorderLayout.SOUTH); | f.add(control, BorderLayout.SOUTH); | ||
f.setVisible(true); | f.setVisible(true); |
Version vom 16. Oktober 2009, 21:49 Uhr
<code=java> /*
* 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 javax.swing.*; import javax.swing.event.*; import javax.swing.plaf.TabbedPaneUI; import javax.swing.plaf.basic.BasicTabbedPaneUI;
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() {
public void stateChanged(ChangeEvent e) { if (nextAction != null) { nextAction.setEnabled(hasNext()); } if (previousAction != null) { previousAction.setEnabled(hasPrevious()); } } }); setLayout(new BorderLayout()); 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; }
/** * @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(ChangeListener l) {
pane.addChangeListener(l);
}
/**
* Removes a ChangeListener
from this cardpane.
*/
public void removeChangeListener(ChangeListener l) {
pane.removeChangeListener(l);
}
/** * gets the action to be used for a "Next" button */ public Action getNextAction() { if (nextAction == null) { nextAction = new CardAction("Next", true); nextAction.putValue(Action.MNEMONIC_KEY, KeyEvent.VK_N); nextAction.setEnabled(hasNext()); } return nextAction; }
/** * gets the action to be used for a "Previous" button */ public Action getPreviousAction() { if (previousAction == null) { previousAction = new CardAction("Previous", false); previousAction.putValue(Action.MNEMONIC_KEY, KeyEvent.VK_P); 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(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;
public CardAction(String text, boolean isNext) { super(text); this.isNext = isNext; putValue(Action.SHORT_DESCRIPTION, getValue(Action.NAME)); }
public void actionPerformed(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 implements ChangeListener, PropertyChangeListener {
public final static int RESET_FOCUS = 0; public final static int RETAIN_FOCUS = 1; private HashMap<Component, Component> tabFocus = new HashMap<Component, Component>(); private HashSet<Component> exceptions; private JTabbedPane tabbedPane; private int focusPolicy; private final KeyboardFocusManager focusManager;
/** * Create with the default Retain Focus policy */ public TabFocusHandler(JTabbedPane tabbedPane) { this(tabbedPane, RETAIN_FOCUS); }
/** * Create using the specified focus policy */ public TabFocusHandler(JTabbedPane tabbedPane, 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 tabbedPane.addChangeListener(this); focusManager = KeyboardFocusManager.getCurrentKeyboardFocusManager(); focusManager.addPropertyChangeListener("permanentFocusOwner", this); }
public void uninstall() { tabbedPane.removeChangeListener(this); focusManager.removePropertyChangeListener(this); }
/** * Specify a tab with an exception to the focus policy rule */ public void addException(int index) { if (exceptions == null) { exceptions = new HashSet<Component>(); } Component key = tabbedPane.getComponentAt(index); exceptions.add(key); }
/** * Tab has changed. Focus on saved component for the given tab. * When there is no saved component, focus on the first component. */ public void stateChanged(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(); } }
/** * Track focus changes and update the current focus component * for the current tab */ public void propertyChange(PropertyChangeEvent e) { // No need to track focus change if (exceptions == null && focusPolicy == RESET_FOCUS) { return; } // Check for exceptions to the focus policy exist 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; } } // Track focus changes for the tab Component value = (Component) e.getNewValue(); if (value != null && SwingUtilities.isDescendingFrom(value, key)) { tabFocus.put(key, value); } } }
//for testing only: public static void main(final String[] args) { try { UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()); } catch (Exception ex) { ex.printStackTrace(); } 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 {
public 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())); control.add(new JButton(cardPane.getNextAction())); f.add(control, BorderLayout.SOUTH); f.setVisible(true); } }; //GUI must start on EventDispatchThread: SwingUtilities.invokeLater(gui); }
} </code=java>