/** * This software is released as part of the Pumpernickel project. * * All com.pump resources in the Pumpernickel project are distributed under the * MIT License: * https://raw.githubusercontent.com/mickleness/pumpernickel/master/License.txt * * More information about the Pumpernickel project is available here: * https://mickleness.github.io/pumpernickel/ */ package com.pump.plaf; import java.awt.Color; import java.awt.Component; import java.awt.Component.BaselineResizeBehavior; import java.awt.Container; import java.awt.Dimension; import java.awt.Font; import java.awt.GradientPaint; import java.awt.Graphics; import java.awt.Graphics2D; import java.awt.GridBagConstraints; import java.awt.GridBagLayout; import java.awt.Insets; import java.awt.LayoutManager; import java.awt.Paint; import java.awt.Rectangle; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.ComponentAdapter; import java.awt.event.ComponentEvent; import java.awt.event.ComponentListener; import java.awt.event.ContainerAdapter; import java.awt.event.ContainerEvent; import java.awt.event.ContainerListener; import java.awt.event.MouseEvent; import java.awt.event.MouseListener; import java.awt.event.MouseMotionListener; import java.beans.PropertyChangeEvent; import java.beans.PropertyChangeListener; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; import java.util.HashMap; import java.util.List; import java.util.Map; import javax.swing.AbstractButton; import javax.swing.Icon; import javax.swing.JButton; import javax.swing.JComponent; import javax.swing.JLabel; import javax.swing.JPanel; import javax.swing.JTabbedPane; import javax.swing.JToggleButton.ToggleButtonModel; import javax.swing.SwingConstants; import javax.swing.SwingUtilities; import javax.swing.UIManager; import javax.swing.border.Border; import javax.swing.border.CompoundBorder; import javax.swing.border.EmptyBorder; import javax.swing.event.ChangeEvent; import javax.swing.event.ChangeListener; import javax.swing.plaf.TabbedPaneUI; import javax.swing.plaf.UIResource; import javax.swing.plaf.basic.BasicButtonListener; import javax.swing.plaf.basic.BasicButtonUI; import com.pump.awt.DescendantListener; import com.pump.awt.SplayedLayout; import com.pump.icon.button.MinimalDuoToneCloseIcon; import com.pump.plaf.button.QButtonUI; import com.pump.plaf.button.QButtonUI.HorizontalPosition; import com.pump.plaf.button.QButtonUI.VerticalPosition; import com.pump.swing.PartialLineBorder; /** * This is modeled after Safari's tab model. There is one row of tabs (and other * possible controls) that is always stretch to fill the available width (or * height for vertical tabs). * * Features include: * <ul> * <li>The option to hide the tab row when only one tab is present (the content * stays visible, but the tab is hidden). * <li>The option to automatically include a close button on all tabs. * <li>The option to add custom controls before/after the tabs. * </ul> * * TODO: it would be nice to add drag-and-drop reordering * * <p> * This UI does not support the tab layout policy of the JTabbedPane; tabs are * only presented in a single continuous scrollable row. So this will only * support one or zero runs of tabs.<br> * <h3>Context / Design</h3> * <p> * I would suggest this UI is appropriate for tabbed document interfaces (TDIs), * which are a specific implementation of multiple document interfaces (MDIs). * (This is all in contrast to single document interfaces (SDIs).) * <p> * There is a lot of related reading on this online and in well-researched books * like <em>About Face</em>. To sum up what I've read so far: * <ul> * <li>MDIs were most popular in the 90s and early 2000s. This is partly because * of resource limitations at the time -- there was no alternative. Some UX * designers regard them with a stigma and will suggest SDIs are the better way * to go. * <li>Still TDIs don't appear to be going away anytime soon. Browsers are the * most obvious example. Microsoft Excel is another. Apple, who is notorious for * UX scrutiny, recently added tabs to Finder windows. But my favorite example * is Notepad vs Notepad++. Notepad is a fine tool, and maybe it's the go-to * choice for many users, but Notepad++ (or a similar TDI tool like Atom) is * where power-users gravitate to. * </ul> * <p> * If you're interested in displaying a fixed set of tabs (such as in a complex * properties or preferences dialog): you should try to make sure the tabs are * always visible. Depending on the number of tabs you want to display: this UI * may not be a good fit in that case. Horizontal scrolling is relatively subtle * (even if you include an indicator like "+3" to indicate 3 more tabs are * out-of-sight, some users won't register that). * <p> * <h3>Implementation</h3> * <p> * This section describes how parts of this class are implemented, which may be * useful if you want to customize either a {@link Style} object or extend the * {@link DefaultTab} class. * <p> * (This section is written as if using the default tab placement; the same * components are arranged in subtly different ways for other tab placements.) * <p> * Each BoxTabbedPaneUI has a <code>controlRow</code> panel that stretches the * width of the <code>JTabbedPane</code>. * <p> * If you have defined leading components (see * {@link #PROPERTY_LEADING_COMPONENTS}), then they are anchored to the left * side of the <code>controlRow</code>. If you have defined trailing components * (see {@link #PROPERTY_TRAILING_COMPONENTS}), then they are anchored on the * right side of the <code>controlRow</code>. All of these components are given * their preferred size, so their preferred size should be modest. Note it is * your responsibility to format these controls. (For example, you may want your * components to match the border/fill of your BoxTabbedPaneUI.) * <p> * The <code>tabContainer</code> panel is given the remaining width. This width * is divided up into a series of <code>TabContainer</code> panels with one * panel per tab. The <code>TabContainer</code> class is actually a * <code>AbstractButton</code>: if you click it you are selecting that tab. You * can also transfer keyboard focus to it and press the spacebar. * <p> * The <code>TabContainer</code> is actually blank, and it contains the * component used to render/control the actual tab. By default this component * will be the <code>DefaultTab</code> object, but if you have defined custom * components using {@link JTabbedPane#setTabComponentAt(int, Component)} then * that component is used. * <p> * The <code>DefaultTab</code> contains a close button that is visible when * {@link #PROPERTY_CLOSEABLE_TABS} is enabled. (That also means: this property * has no automatic effect if you have supplied your own custom tab components.) * <p> * <img src= * "" * alt="diagram of BoxTabbedPaneUI components"> */ public class BoxTabbedPaneUI extends TabbedPaneUI { public interface Style { public void formatControlRow(JTabbedPane tabs, JPanel tabsContainer); /** * * @param tabs * @param tabContainer * @param tabIndex * the tab index being formatted, or -1 if this button is * part of the control row but does not affect a tab. */ public void formatControlRowButton(JTabbedPane tabs, AbstractButton tabContainer, int tabIndex); public void formatCloseButton(JTabbedPane tabs, JButton closeButton); public void formatTabContent(JTabbedPane tabs, JComponent c); } public static DefaultStyle STYLE_DEFAULT = new DefaultStyle(); public static Style getStyle(JTabbedPane tabs) { Style style = (Style) tabs.getClientProperty(PROPERTY_STYLE); if (style == null) style = STYLE_DEFAULT; return style; } public static class DefaultStyle implements Style { public class TabButtonUI extends BasicButtonUI { @Override public void paint(Graphics g0, JComponent c) { Graphics2D g = (Graphics2D) g0.create(); boolean isSelected = ((AbstractButton) c).isSelected(); g.setPaint(createContentGradient(c, isSelected)); g.fillRect(0, 0, c.getWidth(), c.getHeight()); if (c.isFocusOwner()) { int x = 0; int y = 0; int w = c.getWidth() - 1; int h = c.getHeight() - 1; Border b = c.getBorder(); if (b != null) { Insets i = b.getBorderInsets(c); x += i.left; y += i.top; w -= i.left + i.right; h -= i.top + i.bottom; } Rectangle r = new Rectangle(x, y, w, h); PlafPaintUtils.paintFocus(g, r, 1, PlafPaintUtils.getFocusRingColor()); } g.dispose(); super.paint(g0, c); } } protected Color borderNormalLight = new Color(0xa1a0a1); protected Color borderNormalDark = new Color(0x9e9c9e); protected Color borderOuterSelected = new Color(0xbab9ba); protected Color borderOuterUnselected = new Color(0xa1a0a1); protected Color contentOuterNormal = new Color(0xbdbbbd); protected Color contentInnerNormal = new Color(0xb5b3b5); protected Color contentOuterSelected = new Color(0xd9d7d9); protected Color contentInnerSelected = new Color(0xd1cfd1); private boolean outerBorder, contentBorder; protected GradientPaint createContentGradient(JComponent c, boolean isSelected) { JTabbedPane tabs = getTabbedPaneParent(c); int placement = tabs == null ? SwingConstants.TOP : tabs .getTabPlacement(); Color outer = isSelected ? contentOuterSelected : contentOuterNormal; Color inner = isSelected ? contentInnerSelected : contentInnerNormal; if (placement == SwingConstants.LEFT) { return (new GradientPaint(0, 0, outer, c.getWidth(), 0, inner)); } else if (placement == SwingConstants.RIGHT) { return (new GradientPaint(0, 0, inner, c.getWidth(), 0, outer)); } else if (placement == SwingConstants.BOTTOM) { return (new GradientPaint(0, 0, inner, 0, c.getHeight(), outer)); } else { return (new GradientPaint(0, 0, outer, 0, c.getHeight(), inner)); } } private JTabbedPane getTabbedPaneParent(Container c) { while (c != null) { if (c instanceof JTabbedPane) return (JTabbedPane) c; c = c.getParent(); } return null; } public DefaultStyle() { this(true, true); } public DefaultStyle(boolean outerBorder, boolean contentBorder) { this.outerBorder = outerBorder; this.contentBorder = contentBorder; } private GradientPaint createBorderGradient(int tabPlacement) { if (tabPlacement == SwingConstants.BOTTOM) { return new GradientPaint(0, 0, borderNormalDark, 0, 25, borderNormalLight); } else if (tabPlacement == SwingConstants.LEFT) { return new GradientPaint(0, 0, borderNormalLight, 25, 0, borderNormalDark); } else if (tabPlacement == SwingConstants.RIGHT) { return new GradientPaint(0, 0, borderNormalDark, 25, 0, borderNormalLight); } else { return new GradientPaint(0, 0, borderNormalLight, 0, 25, borderNormalDark); } } @Override public void formatControlRow(JTabbedPane tabs, JPanel tabsContainer) { Border border; Paint paint = createBorderGradient(tabs.getTabPlacement()); if (tabs.getTabPlacement() == SwingConstants.BOTTOM) { border = new PartialLineBorder(paint, true, true, tabs.getTabCount() == 0, true); } else if (tabs.getTabPlacement() == SwingConstants.LEFT) { border = new PartialLineBorder(paint, true, tabs.getTabCount() == 0, true, true); } else if (tabs.getTabPlacement() == SwingConstants.RIGHT) { border = new PartialLineBorder(paint, true, true, true, tabs.getTabCount() == 0); } else { border = new PartialLineBorder(paint, tabs.getTabCount() == 0, true, true, true); } tabsContainer.setUI(new GradientPanelUI(createContentGradient(tabs, false))); tabsContainer.setBorder(border); } @Override public void formatControlRowButton(JTabbedPane tabs, AbstractButton tabContainer, int tabIndex) { if (tabIndex >= 0) { Border border; int p = tabs.getTabPlacement(); if (p == SwingConstants.LEFT) { border = new PartialLineBorder(createBorderGradient(p), tabIndex != 0, false, false, false); if (isOuterBorder()) { Paint paint = tabIndex == tabs.getSelectedIndex() ? borderOuterSelected : borderOuterUnselected; Border inner = new PartialLineBorder(paint, false, true, false, false); border = new CompoundBorder(border, inner); } } else if (p == SwingConstants.RIGHT) { border = new PartialLineBorder(createBorderGradient(p), tabIndex != 0, false, false, false); if (isOuterBorder()) { Paint paint = tabIndex == tabs.getSelectedIndex() ? borderOuterSelected : borderOuterUnselected; Border inner = new PartialLineBorder(paint, false, false, false, true); border = new CompoundBorder(border, inner); } } else { border = new PartialLineBorder(createBorderGradient(p), false, tabIndex != 0, false, false); if (isOuterBorder()) { Paint paint = tabIndex == tabs.getSelectedIndex() ? borderOuterSelected : borderOuterUnselected; Border inner = new PartialLineBorder(paint, p == SwingConstants.TOP, false, p == SwingConstants.BOTTOM, false); border = new CompoundBorder(border, inner); } } tabContainer.setBorder(border); } if (!(tabContainer.getUI() instanceof TabButtonUI)) { tabContainer.setUI(new TabButtonUI()); tabContainer.putClientProperty( QButtonUI.PROPERTY_HORIZONTAL_POSITION, HorizontalPosition.MIDDLE); tabContainer.putClientProperty( QButtonUI.PROPERTY_VERTICAL_POSITION, VerticalPosition.MIDDLE); tabContainer.putClientProperty( QButtonUI.PROPERTY_STROKE_PAINTED, Boolean.FALSE); } } public boolean isOuterBorder() { return outerBorder; } @Override public void formatCloseButton(JTabbedPane tabs, JButton closeButton) { if (!(closeButton.getIcon() instanceof MinimalDuoToneCloseIcon)) { MinimalDuoToneCloseIcon icon = new MinimalDuoToneCloseIcon( closeButton); closeButton.setIcon(icon); DescendantMouseListener.installForParentOf(closeButton, icon.getParentRollover()); } closeButton.setMargin(new Insets(0, 0, 0, 0)); closeButton.setBorder(new EmptyBorder(0, 0, 0, 0)); } @Override public void formatTabContent(JTabbedPane tabs, JComponent c) { if (contentBorder) { Border border; if (tabs.getTabPlacement() == SwingConstants.LEFT) { border = new PartialLineBorder(borderNormalDark, true, false, true, true); } else if (tabs.getTabPlacement() == SwingConstants.BOTTOM) { border = new PartialLineBorder(borderNormalDark, true, true, false, true); } else if (tabs.getTabPlacement() == SwingConstants.RIGHT) { border = new PartialLineBorder(borderNormalDark, true, true, true, false); } else { border = new PartialLineBorder(borderNormalDark, false, true, true, true); } c.setBorder(border); } } } /** * An optional client property for a <code>java.util.List</code> of * JComponents to always place on the trailing (right) side of the tabs. */ public static final String PROPERTY_TRAILING_COMPONENTS = BoxTabbedPaneUI.class .getName() + "#trailingComponents"; /** * An optional client property for a <code>java.util.List</code> of * JComponents to always place on the leading (left) side of the tabs. */ public static final String PROPERTY_LEADING_COMPONENTS = BoxTabbedPaneUI.class .getName() + "#leadingComponents"; /** * This client property on tabs maps to the Integer index of that tab * relative to the JTabbedPane. */ private static final String PROPERTY_TAB_INDEX = BoxTabbedPaneUI.class .getName() + "#tabIndex"; /** * This client property on JTabbedPane resolves to an internal Data object. */ private static final String PROPERTY_DATA = BoxTabbedPaneUI.class.getName() + "#data"; /** * This optional property on JTabbedPane resolves to a Boolean used to * indicate whether tabs should be closeable by default. Note if you have * defined a custom tab component using * {@link JTabbedPane#setTabComponentAt(int, Component)} then that component * is used and this property is not automatically consulted. This value is * assumed to be false by default. */ public static final String PROPERTY_CLOSEABLE_TABS = BoxTabbedPaneUI.class .getName() + "#closeableTabs"; /** * This optional property on JTabbedPane resolves to a Style object used to * help format colors, borders, inner component UIs. */ public static final String PROPERTY_STYLE = BoxTabbedPaneUI.class.getName() + "#style"; /** * This optional property on JTabbedPane resolves to a Boolean used to * indicate whether the tab UI controls should be hidden if there is only * one tab. This value is assumed to be false by default. */ public static final String PROPERTY_HIDE_SINGLE_TAB = BoxTabbedPaneUI.class .getName() + "#hideSingleTab"; /** * This caches a TabContainer on each Tab object. If we recreate the * TabContainer with every refresh: the icons/animations may flicker a * little. */ private static final String PROPERTY_TAB_CONTAINER = BoxTabbedPaneUI.class .getName() + "#tabContainer"; private static class UIResourcePanel extends JPanel implements UIResource { private static final long serialVersionUID = 1L; UIResourcePanel(LayoutManager layoutManager) { super(layoutManager); } } private static class BoxTabbedPaneUILayoutManager implements LayoutManager { @Override public void addLayoutComponent(String name, Component comp) { } @Override public void removeLayoutComponent(Component comp) { } @Override public Dimension preferredLayoutSize(Container parent) { return getLayoutSize(parent, true); } @Override public Dimension minimumLayoutSize(Container parent) { return getLayoutSize(parent, false); } private Dimension getLayoutSize(Container parent, boolean preferred) { int tabPlacement = ((JTabbedPane) parent).getTabPlacement(); boolean verticalPlacement = tabPlacement == JTabbedPane.TOP || tabPlacement == JTabbedPane.BOTTOM; Dimension additional = new Dimension(0, 0); Dimension max = new Dimension(0, 0); for (int a = 0; a < parent.getComponentCount(); a++) { Component c = parent.getComponent(a); Dimension d = preferred ? c.getPreferredSize() : c .getMinimumSize(); if (c instanceof UIResourcePanel) { if (verticalPlacement) { additional.height += d.height; additional.width = Math.max(additional.width, d.width); } else { additional.width += d.width; additional.height = Math.max(additional.height, d.height); } } else { max.width = Math.max(max.width, d.width); max.height = Math.max(max.height, d.height); } } if (verticalPlacement) { return new Dimension(Math.max(additional.width, max.width), additional.height + max.height); } return new Dimension(additional.width + max.width, Math.max( additional.height, max.height)); } @Override public void layoutContainer(Container parent) { int y = 0; int x = 0; int width = parent.getWidth(); int height = parent.getHeight(); int tabPlacement = ((JTabbedPane) parent).getTabPlacement(); for (int a = 0; a < parent.getComponentCount(); a++) { Component c = parent.getComponent(a); if (c instanceof UIResourcePanel && c.isVisible()) { Dimension d = c.getPreferredSize(); if (tabPlacement == JTabbedPane.TOP) { c.setBounds(x, y, width, d.height); y += d.height; height -= d.height; } else if (tabPlacement == JTabbedPane.BOTTOM) { c.setBounds(x, height - d.height, width, d.height); height -= d.height; } else if (tabPlacement == JTabbedPane.LEFT) { c.setBounds(x, y, d.width, height); x += d.width; width -= d.width; } else if (tabPlacement == JTabbedPane.RIGHT) { c.setBounds(width - d.width, y, d.width, height); width -= d.width; } } } for (int a = 0; a < parent.getComponentCount(); a++) { Component c = parent.getComponent(a); if (!(c instanceof UIResourcePanel)) { c.setBounds(x, y, width, height); } } } } private static final String[] REFRESH_PROPERTIES = new String[] { "mnemonicAt", "displayedMnemonicIndexAt", "indexForTitle", "tabLayoutPolicy", "opaque", "background", "indexForTabComponent", "indexForNullComponent", "font", PROPERTY_HIDE_SINGLE_TAB, PROPERTY_CLOSEABLE_TABS }; /** * This is a cluster of data related to a specific JTabbedPane. * <p> * In theory a TabbedPaneUI can be applied to multiple JTabbedPanes. If that * happens each pane gets its own Data object to control it. */ class Data { JPanel controlRow = new UIResourcePanel(new GridBagLayout()); JPanel leadingComponents = new JPanel(new GridBagLayout()); JPanel tabsContainer = new JPanel(); JPanel trailingComponents = new JPanel(new GridBagLayout()); JTabbedPane tabs; ChangeListener refreshChangeListener = new ChangeListener() { @Override public void stateChanged(ChangeEvent e) { refreshTabStates(); } }; PropertyChangeListener refreshPropertyListener = new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { refreshTabStates(); } }; PropertyChangeListener refreshExtraComponentsListener = new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { refreshExtraComponents(); } }; PropertyChangeListener refreshStyleListener = new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { refreshStyle(); refreshTabStates(); } }; PropertyChangeListener refreshContentBorderListener = new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { refreshContentBorder(); } }; PropertyChangeListener tabPlacementPropertyListener = new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { relayoutControlRow(); refreshExtraComponents(); refreshTabStates(); refreshStyle(); refreshContentBorder(); } }; ContainerListener containerListener = new ContainerAdapter() { @Override public void componentAdded(ContainerEvent e) { refreshTabStates(); refreshContentBorder(); refreshStyle(); } @Override public void componentRemoved(ContainerEvent e) { refreshTabStates(); refreshContentBorder(); refreshStyle(); } }; public Data(JTabbedPane tabs) { this.tabs = tabs; controlRow.setOpaque(false); leadingComponents.setOpaque(false); tabsContainer.setOpaque(false); trailingComponents.setOpaque(false); } public void install() { tabs.setLayout(new BoxTabbedPaneUILayoutManager()); removeNonUIResources(tabs); tabs.add(controlRow); relayoutControlRow(); for (String property : new String[] { PROPERTY_STYLE }) { tabs.addPropertyChangeListener(property, refreshStyleListener); } for (String property : new String[] { PROPERTY_TRAILING_COMPONENTS, PROPERTY_LEADING_COMPONENTS }) { tabs.addPropertyChangeListener(property, refreshExtraComponentsListener); } tabs.addPropertyChangeListener("tabPlacement", tabPlacementPropertyListener); for (String property : REFRESH_PROPERTIES) { tabs.addPropertyChangeListener(property, refreshPropertyListener); } tabs.addContainerListener(containerListener); tabs.getModel().addChangeListener(refreshChangeListener); refreshExtraComponents(); refreshTabStates(); refreshContentBorder(); refreshStyle(); } private void refreshContentBorder() { for (int a = 0; a < tabs.getComponentCount(); a++) { Component c = tabs.getComponent(a); if (!(c instanceof UIResource) && (c instanceof JComponent)) { getStyle(tabs).formatTabContent(tabs, (JComponent) c); } } } private void refreshStyle() { getStyle(tabs).formatControlRow(tabs, tabsContainer); } private void relayoutControlRow() { controlRow.removeAll(); GridBagConstraints c = new GridBagConstraints(); c.gridx = 0; c.gridy = 0; c.weightx = 0; c.weighty = 0; c.fill = GridBagConstraints.BOTH; if (tabs.getTabPlacement() == SwingConstants.LEFT) { controlRow.add(trailingComponents, c); c.gridy++; c.weighty = 1; controlRow.add(tabsContainer, c); c.gridy++; c.weighty = 0; controlRow.add(leadingComponents, c); } else if (tabs.getTabPlacement() == SwingConstants.RIGHT) { controlRow.add(leadingComponents, c); c.gridy++; c.weighty = 1; controlRow.add(tabsContainer, c); c.gridy++; c.weighty = 0; controlRow.add(trailingComponents, c); } else { controlRow.add(leadingComponents, c); c.gridx++; c.weightx = 1; controlRow.add(tabsContainer, c); c.gridx++; c.weightx = 0; controlRow.add(trailingComponents, c); } controlRow.revalidate(); } int lastTabPlacement = -1; @SuppressWarnings("unchecked") private void refreshExtraComponents() { List<JComponent> newLeadingComponents = (List<JComponent>) tabs .getClientProperty(PROPERTY_LEADING_COMPONENTS); List<JComponent> newTrailingComponents = (List<JComponent>) tabs .getClientProperty(PROPERTY_TRAILING_COMPONENTS); int tabPlacement = tabs.getTabPlacement(); boolean forceReinstall = tabPlacement != lastTabPlacement; lastTabPlacement = tabPlacement; installExtraComponents(leadingComponents, newLeadingComponents, forceReinstall); installExtraComponents(trailingComponents, newTrailingComponents, forceReinstall); } ComponentListener extraComponentListener = new ComponentAdapter() { @Override public void componentShown(ComponentEvent e) { refreshExtraContainerVisibility(); } @Override public void componentHidden(ComponentEvent e) { refreshExtraContainerVisibility(); } }; private void installExtraComponents(Container container, List<JComponent> components, boolean forceReinstall) { if (components == null) components = new ArrayList<>(); Component[] oldComponents = container.getComponents(); if (!Arrays.asList(oldComponents).equals(components)) { forceReinstall = true; } if (forceReinstall) { container.removeAll(); GridBagConstraints c = new GridBagConstraints(); c.gridx = 0; c.gridy = 100; c.weightx = 1; c.weighty = 1; c.fill = GridBagConstraints.BOTH; for (JComponent jc : components) { container.add(jc, c); if (tabs.getTabPlacement() == SwingConstants.LEFT) { c.gridy--; } else if (tabs.getTabPlacement() == SwingConstants.RIGHT) { c.gridy++; } else { c.gridx++; } jc.removeComponentListener(extraComponentListener); jc.addComponentListener(extraComponentListener); for (Component oldComponent : oldComponents) { if (components.contains(oldComponent)) { oldComponent .removeComponentListener(extraComponentListener); } } } container.revalidate(); } refreshExtraContainerVisibility(); } private void refreshExtraContainerVisibility() { leadingComponents .setVisible(containsVisibleComponent(leadingComponents)); trailingComponents .setVisible(containsVisibleComponent(trailingComponents)); } private boolean containsVisibleComponent(Container container) { for (Component c : container.getComponents()) { if (c.isVisible()) return true; } return false; } public void uninstall() { tabs.remove(controlRow); for (String property : new String[] { PROPERTY_STYLE }) { tabs.removePropertyChangeListener(property, refreshStyleListener); } for (String property : new String[] { PROPERTY_TRAILING_COMPONENTS, PROPERTY_LEADING_COMPONENTS }) { tabs.removePropertyChangeListener(property, refreshExtraComponentsListener); } tabs.removePropertyChangeListener("tabPlacement", tabPlacementPropertyListener); for (String property : REFRESH_PROPERTIES) { tabs.removePropertyChangeListener(property, refreshPropertyListener); } tabs.removeContainerListener(containerListener); tabs.getModel().removeChangeListener(refreshChangeListener); } private void removeNonUIResources(Container c) { for (Component comp : c.getComponents()) { if (!(comp instanceof UIResource)) { c.remove(comp); } } } protected void refreshTabStates() { List<Component> newTabs = new ArrayList<>(); for (int a = 0; a < tabs.getTabCount(); a++) { JComponent tab = (JComponent) tabs.getTabComponentAt(a); if (tab == null) tab = getDefaultTab(a); TabContainer tabContainer = (TabContainer) tab .getClientProperty(PROPERTY_TAB_CONTAINER); if (tabContainer == null) { tabContainer = new TabContainer(tabs, a, tab); tab.putClientProperty(PROPERTY_TAB_CONTAINER, tabContainer); } newTabs.add(tabContainer); tab.putClientProperty(PROPERTY_TAB_INDEX, a); getStyle(tabs).formatControlRowButton(tabs, tabContainer, a); } Boolean hideSingleTab = (Boolean) tabs .getClientProperty(PROPERTY_HIDE_SINGLE_TAB); if (hideSingleTab == null) hideSingleTab = Boolean.FALSE; controlRow.setVisible(!(tabs.getTabCount() <= 1 && hideSingleTab)); if (!(tabsContainer.getLayout() instanceof SplayedLayout)) { SplayedLayout l = new SplayedLayout(true) { @Override protected Collection<JComponent> getEmphasizedComponents( JComponent container) { Collection<JComponent> returnValue = super .getEmphasizedComponents(container); for (Component c : container.getComponents()) { if (c instanceof AbstractButton) { AbstractButton ab = (AbstractButton) c; if (ab.isSelected()) returnValue.add(ab); } } return returnValue; } }; tabsContainer.setLayout(l); } int orientation = tabs.getTabPlacement() == SwingConstants.LEFT || tabs.getTabPlacement() == SwingConstants.RIGHT ? SwingConstants.VERTICAL : SwingConstants.HORIZONTAL; ((SplayedLayout) tabsContainer.getLayout()).setOrientation(null, orientation); setComponents(tabsContainer, newTabs); tabsContainer.revalidate(); } /** * This is basically * <code>java.awt.Container.setComponents(Component[])</code>. This is * functionally the same as removing all of a container's children and * then adding them back again, but this method makes individual changes * to minimize the number of container/hierarchy changes that listeners * hear. */ private void setComponents(Container container, List<Component> components) { Component[] oldComponents = container.getComponents(); for (int a = 0; a < oldComponents.length; a++) { if (!components.contains(oldComponents[a])) { container.remove(oldComponents[a]); } } oldComponents = container.getComponents(); int oldCtr = 0; int newCtr = 0; while (newCtr < components.size()) { Component oldComponent = oldCtr < oldComponents.length ? oldComponents[oldCtr] : null; if (oldComponent != components.get(newCtr)) { container.add(components.get(newCtr), oldCtr); } else { oldCtr++; } newCtr++; } } Map<Integer, DefaultTab> tabMap = new HashMap<>(); private DefaultTab getDefaultTab(int a) { DefaultTab returnValue = tabMap.get(a); if (returnValue == null) { returnValue = createDefaultTab(tabs, a); tabMap.put(a, returnValue); } else { returnValue.decorateLabel(a); } return returnValue; } public int getTabRunCount() { return 1; } public Rectangle getTabBounds(int index) { for (int a = 0; a < controlRow.getComponentCount(); a++) { Component c = controlRow.getComponent(a); if (c instanceof JComponent) { JComponent jc = (JComponent) c; Integer i = (Integer) jc .getClientProperty(PROPERTY_TAB_INDEX); if (i != null && i == index) return SwingUtilities.convertRectangle(controlRow, jc.getBounds(), tabs); } } return null; } public int getTabForCoordinate(int x, int y) { Component c = SwingUtilities.getDeepestComponentAt(tabs, x, y); while (c != tabs) { if (c instanceof JComponent) { JComponent jc = (JComponent) c; Integer i = (Integer) jc .getClientProperty(PROPERTY_TAB_INDEX); if (i != null) return i; } c = c.getParent(); } return -1; } } static class TabContainer extends AbstractButton { /** * This converts MouseEvent sources to an AbstractButton. */ private static class MyBasicButtonListener extends BasicButtonListener { AbstractButton button; public MyBasicButtonListener(AbstractButton b) { super(b); button = b; } @Override public void mouseMoved(MouseEvent e) { super.mouseMoved(convert(e)); } @Override public void mouseDragged(MouseEvent e) { super.mouseDragged(convert(e)); } @Override public void mouseClicked(MouseEvent e) { super.mouseClicked(convert(e)); } @Override public void mousePressed(MouseEvent e) { super.mousePressed(convert(e)); } @Override public void mouseReleased(MouseEvent e) { super.mouseReleased(convert(e)); } @Override public void mouseEntered(MouseEvent e) { super.mouseEntered(convert(e)); } @Override public void mouseExited(MouseEvent e) { super.mouseExited(convert(e)); } private MouseEvent convert(MouseEvent e) { if (e.getSource() == button) return e; return SwingUtilities.convertMouseEvent(e.getComponent(), e, button); } } JTabbedPane tabs; int tabIndex; public TabContainer(JTabbedPane tabs, int tabIndex, JComponent tab) { this.tabs = tabs; this.tabIndex = tabIndex; setLayout(new GridBagLayout()); GridBagConstraints c = new GridBagConstraints(); c.gridx = 0; c.gridy = 0; c.weightx = 1; c.weighty = 1; c.fill = GridBagConstraints.BOTH; add(tab, c); setFocusable(true); setFocusPainted(true); setModel(new ToggleButtonModel()); refreshSelectedState(); getModel().addChangeListener(new ChangeListener() { @Override public void stateChanged(ChangeEvent e) { refreshSelectedState(); } }); tabs.addChangeListener(new ChangeListener() { @Override public void stateChanged(ChangeEvent e) { refreshSelectedState(); } }); BasicButtonListener buttonListener = new MyBasicButtonListener(this); buttonListener.installKeyboardActions(this); addFocusListener(buttonListener); DescendantListener .addMouseListener(this, (MouseListener) buttonListener, false, AbstractButton.class); DescendantListener.addMouseListener(this, (MouseMotionListener) buttonListener, false, AbstractButton.class); addActionListener(new ActionListener() { @Override public void actionPerformed(ActionEvent e) { TabContainer.this.tabs .setSelectedIndex(TabContainer.this.tabIndex); } }); setFocusable(true); setRequestFocusEnabled(false); } private void refreshSelectedState() { getModel().setSelected(tabs.getSelectedIndex() == tabIndex); } private static final long serialVersionUID = 1L; } public static class DefaultTab extends JPanel { private static final long serialVersionUID = 1L; protected final JTabbedPane tabs; private JLabel label = new JLabel(); private JButton closeButton = new JButton(); private int tabIndex = -1; private PropertyChangeListener closeableTabsListener = new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { refreshCloseableButton(); } }; public DefaultTab(JTabbedPane tabs, int tabIndex) { setLayout(new GridBagLayout()); this.tabs = tabs; closeButton.setContentAreaFilled(false); closeButton.setBorderPainted(false); label.setHorizontalAlignment(SwingConstants.CENTER); Font font = UIManager.getFont("TabbedPane.smallFont"); if (font != null) label.setFont(font); setOpaque(false); decorateLabel(tabIndex); closeButton.addActionListener(new ActionListener() { @Override public void actionPerformed(ActionEvent e) { doCloseTab(DefaultTab.this.tabIndex); } }); tabs.addPropertyChangeListener(PROPERTY_CLOSEABLE_TABS, closeableTabsListener); refreshCloseableButton(); } /** * Remove a tab at a given index. * <p> * The default implementation simply calls * <code>tabs.removeTabAt(tabIndex)</code>, but subclasses can override * this as needed. For example, you may need to prompt the user to * confirm discarding unsaved changes. * * @param tabIndex * the index of the tab to close. */ protected void doCloseTab(int tabIndex) { tabs.removeTabAt(tabIndex); } private void refreshCloseableButton() { Boolean b = (Boolean) tabs .getClientProperty(PROPERTY_CLOSEABLE_TABS); if (b == null) b = Boolean.FALSE; closeButton.setVisible(b); int i; if (b) { Dimension d = closeButton.getPreferredSize(); i = Math.max(d.width, d.height); } else { i = 0; } i += 2; label.setBorder(new EmptyBorder(0, i, 0, i)); getStyle(tabs).formatCloseButton(tabs, closeButton); } protected JLabel getLabel() { return label; } protected JButton getCloseButton() { return closeButton; } protected GridBagConstraints createLabelConstraints() { GridBagConstraints labelConstraints = new GridBagConstraints(); labelConstraints.gridx = 0; labelConstraints.gridy = 0; labelConstraints.weightx = 1; labelConstraints.weighty = 1; labelConstraints.fill = GridBagConstraints.BOTH; labelConstraints.insets = new Insets(3, 3, 3, 3); return labelConstraints; } protected GridBagConstraints createCloseButtonConstraints() { GridBagConstraints closeButtonConstraints = new GridBagConstraints(); closeButtonConstraints.gridx = 0; closeButtonConstraints.gridy = 0; closeButtonConstraints.weightx = 1; closeButtonConstraints.weighty = 1; closeButtonConstraints.fill = GridBagConstraints.NONE; if (tabs.getTabPlacement() == SwingConstants.LEFT) { closeButtonConstraints.anchor = GridBagConstraints.SOUTH; } else if (tabs.getTabPlacement() == SwingConstants.RIGHT) { closeButtonConstraints.anchor = GridBagConstraints.NORTH; } else { closeButtonConstraints.anchor = GridBagConstraints.WEST; } if (tabs.getTabPlacement() == SwingConstants.LEFT) { closeButtonConstraints.insets = new Insets(0, 0, 3, 0); } else if (tabs.getTabPlacement() == SwingConstants.RIGHT) { closeButtonConstraints.insets = new Insets(3, 0, 0, 0); } else { closeButtonConstraints.insets = new Insets(0, 3, 0, 0); } return closeButtonConstraints; } public void decorateLabel(int index) { tabIndex = index; Color background = tabs.getBackgroundAt(index); setBackground(background); Icon disabledIcon = tabs.getDisabledIconAt(index); Icon icon = tabs.getIconAt(index); if (!tabs.isEnabledAt(index) || !tabs.isEnabled()) { if (disabledIcon != null) icon = disabledIcon; } getLabel().setIcon(icon); // TODO: we don't current support mnemonics tabs.getDisplayedMnemonicIndexAt(index); tabs.getMnemonicAt(index); Color foreground = tabs.getForegroundAt(index); if (foreground == null) foreground = UIManager.getColor("Label.foreground"); getLabel().setForeground(foreground); setToolTipText(tabs.getToolTipTextAt(index)); String title = tabs.getTitleAt(index); if (title == null) title = ""; getLabel().setText(title); getStyle(tabs).formatCloseButton(tabs, closeButton); addChild(tabs.getTabPlacement()); } /** * Add the label to this DefaultTab; if necessary wrap it inside a * RotatedComponent. * * @param tabPlacement * the tab placement (SwingConstants.TOP, * SwingConstants.LEFT, SwingConstants.BOTTOM, * SwingConstants.RIGHT) */ protected void addChild(int tabPlacement) { RotatedPanel.Rotation rotation = RotatedPanel.Rotation.NONE; if (tabPlacement == SwingConstants.LEFT) { rotation = RotatedPanel.Rotation.COUNTER_CLOCKWISE; } else if (tabPlacement == SwingConstants.RIGHT) { rotation = RotatedPanel.Rotation.CLOCKWISE; } Component child = getComponentCount() == 0 ? null : getComponent(0); if (rotation == RotatedPanel.Rotation.NONE) { if (child == getLabel()) { return; } else { removeAll(); add(getCloseButton(), createCloseButtonConstraints()); add(getLabel(), createLabelConstraints()); revalidate(); } } else { if (child instanceof RotatedPanel) { RotatedPanel rc = (RotatedPanel) child; rc.setRotation(rotation); } else { removeAll(); add(getCloseButton(), createCloseButtonConstraints()); add(new RotatedPanel(getLabel(), rotation), createLabelConstraints()); revalidate(); } } } } protected Data getData(JTabbedPane tabs) { Data d = (Data) tabs.getClientProperty(PROPERTY_DATA); if (d == null) { d = new Data(tabs); tabs.putClientProperty(PROPERTY_DATA, d); } return d; } public DefaultTab createDefaultTab(JTabbedPane tabs, int a) { return new DefaultTab(tabs, a); } @Override public void installUI(JComponent c) { super.installUI(c); Data data = getData((JTabbedPane) c); data.install(); } @Override public void uninstallUI(JComponent c) { super.uninstallUI(c); Data data = getData((JTabbedPane) c); data.uninstall(); } @Override public boolean contains(JComponent c, int x, int y) { return x >= 0 && x < c.getWidth() && y >= 0 && y < c.getHeight(); } @Override public int getBaseline(JComponent c, int width, int height) { // TODO return -1; } @Override public BaselineResizeBehavior getBaselineResizeBehavior(JComponent c) { return BaselineResizeBehavior.OTHER; } @Override public int tabForCoordinate(JTabbedPane pane, int x, int y) { return getData(pane).getTabForCoordinate(x, y); } @Override public Rectangle getTabBounds(JTabbedPane pane, int index) { return getData(pane).getTabBounds(index); } @Override public int getTabRunCount(JTabbedPane pane) { return getData(pane).getTabRunCount(); } }