JCardPane: Unterschied zwischen den Versionen
Aus Byte-Welt Wiki
Keine Bearbeitungszusammenfassung |
Keine Bearbeitungszusammenfassung |
||
| Zeile 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>
