Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.alee.laf.combobox.WebComboBoxUI Maven / Gradle / Ivy
/*
* This file is part of WebLookAndFeel library.
*
* WebLookAndFeel library is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* WebLookAndFeel library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with WebLookAndFeel library. If not, see .
*/
package com.alee.laf.combobox;
import com.alee.api.annotations.NotNull;
import com.alee.api.annotations.Nullable;
import com.alee.api.jdk.Consumer;
import com.alee.api.jdk.Objects;
import com.alee.extended.layout.AbstractLayoutManager;
import com.alee.laf.WebLookAndFeel;
import com.alee.laf.button.WebButton;
import com.alee.laf.list.WebList;
import com.alee.laf.scroll.WebScrollPane;
import com.alee.laf.separator.WebSeparator;
import com.alee.managers.style.*;
import com.alee.painter.DefaultPainter;
import com.alee.painter.Painter;
import com.alee.painter.PainterSupport;
import com.alee.painter.decoration.DecorationState;
import com.alee.painter.decoration.DecorationUtils;
import com.alee.painter.decoration.Stateful;
import com.alee.utils.CoreSwingUtils;
import com.alee.utils.swing.EditabilityListener;
import com.alee.utils.swing.VisibilityListener;
import javax.swing.*;
import javax.swing.event.EventListenerList;
import javax.swing.plaf.ComponentUI;
import javax.swing.plaf.basic.BasicComboPopup;
import javax.swing.plaf.basic.ComboPopup;
import java.awt.*;
import java.awt.event.FocusAdapter;
import java.awt.event.FocusEvent;
import java.awt.event.MouseEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.ArrayList;
import java.util.List;
/**
* Custom UI for {@link JComboBox} component.
*
* @author Mikle Garin
*/
public class WebComboBoxUI extends WComboBoxUI implements ShapeSupport, MarginSupport, PaddingSupport
{
/**
* Client property key for referencing {@link JComboBox} within its popup {@link JList}.
*/
protected static final Object COMBOBOX_INSTANCE = "JComboBox.instance";
/**
* Default combobox renderer.
*/
protected static ListCellRenderer DEFAULT_RENDERER;
/**
* Whether or not wide popup is allowed.
* Setting this to {@code true} will allow combobox popup to stretch to larger than combobox widths.
*/
protected Boolean widePopup;
/**
* Component painter.
*/
@DefaultPainter ( ComboBoxPainter.class )
protected IComboBoxPainter painter;
/**
* Listeners.
*/
protected transient PropertyChangeListener visibilityListener;
protected transient EventListenerList listenerList;
protected transient PropertyChangeListener editorChangeListener;
/**
* Runtime variables.
*/
protected transient JSeparator separator;
/**
* Returns an instance of the {@link WebComboBoxUI} for the specified component.
* This tricky method is used by {@link UIManager} to create component UIs when needed.
*
* @param c component that will use UI instance
* @return instance of the {@link WebComboBoxUI}
*/
public static ComponentUI createUI ( final JComponent c )
{
return new WebComboBoxUI ();
}
@Override
public void installUI ( final JComponent c )
{
// Creating listener list before any operations
listenerList = new EventListenerList ();
super.installUI ( c );
// Listeners
if ( popup instanceof JComponent )
{
visibilityListener = new PropertyChangeListener ()
{
@Override
public void propertyChange ( final PropertyChangeEvent evt )
{
if ( Objects.notEquals ( evt.getOldValue (), evt.getNewValue () ) )
{
firePopupVisibilityChanged ( ( Boolean ) evt.getNewValue () );
}
}
};
( ( JComponent ) popup ).addPropertyChangeListener ( WebLookAndFeel.VISIBLE_PROPERTY, visibilityListener );
}
editorChangeListener = new PropertyChangeListener ()
{
@Override
public void propertyChange ( final PropertyChangeEvent evt )
{
if ( Objects.equals ( evt.getPropertyName (), WebLookAndFeel.EDITOR_PROPERTY ) )
{
updateEditor ( comboBox.getEditor () );
}
}
};
comboBox.addPropertyChangeListener ( editorChangeListener );
// Applying skin
StyleManager.installSkin ( comboBox );
}
@Override
public void uninstallUI ( final JComponent c )
{
// Uninstalling applied skin
StyleManager.uninstallSkin ( comboBox );
// Listeners
comboBox.removePropertyChangeListener ( editorChangeListener );
if ( popup instanceof JComponent )
{
( ( JComponent ) popup ).removePropertyChangeListener ( WebLookAndFeel.VISIBLE_PROPERTY, visibilityListener );
}
super.uninstallUI ( c );
// Erasing listener list after all uninstalls
listenerList = null;
}
@Override
protected void installComponents ()
{
// Removing all previous components
comboBox.removeAll ();
// Updating layout
comboBox.setLayout ( createLayoutManager () );
// Value and button separator
addSeparator ();
// Arrow button
addArrowButton ();
// Value editor
if ( comboBox.isEditable () )
{
addEditor ();
}
// Value renderer pane
comboBox.add ( currentValuePane );
}
@Override
protected void uninstallComponents ()
{
// Removing all components
removeArrowButton ();
removeSeparator ();
removeEditor ();
comboBox.removeAll ();
}
/**
* Adding separator onto combobox.
*/
protected void addSeparator ()
{
separator = createSeparator ();
if ( separator instanceof ComboBoxSeparator )
{
( ( ComboBoxSeparator ) separator ).install ();
}
comboBox.add ( separator );
}
/**
* Removing separator from combobox.
*/
protected void removeSeparator ()
{
if ( separator != null )
{
comboBox.remove ( separator );
if ( separator instanceof ComboBoxSeparator )
{
( ( ComboBoxSeparator ) separator ).uninstall ();
}
separator = null;
}
}
/**
* Adding arrow button onto combobox.
*/
protected void addArrowButton ()
{
arrowButton = createArrowButton ();
if ( arrowButton instanceof ComboBoxButton )
{
( ( ComboBoxButton ) arrowButton ).install ();
}
configureArrowButton ();
comboBox.add ( arrowButton );
}
/**
* Removing arrow button from combobox.
*/
protected void removeArrowButton ()
{
if ( arrowButton != null )
{
comboBox.remove ( arrowButton );
unconfigureArrowButton ();
if ( arrowButton instanceof ComboBoxButton )
{
( ( ComboBoxButton ) arrowButton ).uninstall ();
}
arrowButton = null;
}
}
@Override
public Shape getShape ()
{
return PainterSupport.getShape ( comboBox, painter );
}
@Override
public boolean isShapeDetectionEnabled ()
{
return PainterSupport.isShapeDetectionEnabled ( comboBox, painter );
}
@Override
public void setShapeDetectionEnabled ( final boolean enabled )
{
PainterSupport.setShapeDetectionEnabled ( comboBox, painter, enabled );
}
@Override
public Insets getMargin ()
{
return PainterSupport.getMargin ( comboBox );
}
@Override
public void setMargin ( final Insets margin )
{
PainterSupport.setMargin ( comboBox, margin );
}
@Override
public Insets getPadding ()
{
return PainterSupport.getPadding ( comboBox );
}
@Override
public void setPadding ( final Insets padding )
{
PainterSupport.setPadding ( comboBox, padding );
}
/**
* Returns combobox painter.
*
* @return combobox painter
*/
public Painter getPainter ()
{
return PainterSupport.getPainter ( painter );
}
/**
* Sets combobox painter.
* Pass null to remove combobox painter.
*
* @param painter new combobox painter
*/
public void setPainter ( final Painter painter )
{
comboBox.hidePopup ();
resetRendererSize ();
PainterSupport.setPainter ( comboBox, new Consumer ()
{
@Override
public void accept ( final IComboBoxPainter newPainter )
{
WebComboBoxUI.this.painter = newPainter;
}
}, this.painter, painter, IComboBoxPainter.class, AdaptiveComboBoxPainter.class );
}
@Override
protected ListCellRenderer createRenderer ()
{
return new WebComboBoxRenderer.UIResource> ();
}
@Override
protected ComboBoxEditor createEditor ()
{
final ComboBoxEditor editor = new WebComboBoxEditor.UIResource ();
updateEditor ( editor );
return editor;
}
/**
* Forces combobox editor style update.
* todo Properly remove focus listener?
*
* @param editor editor {@link Component}
*/
private void updateEditor ( final ComboBoxEditor editor )
{
if ( editor != null )
{
final Component editorComponent = editor.getEditorComponent ();
editorComponent.addFocusListener ( new FocusAdapter ()
{
@Override
public void focusGained ( final FocusEvent e )
{
comboBox.repaint ();
}
@Override
public void focusLost ( final FocusEvent e )
{
comboBox.repaint ();
}
} );
if ( editorComponent instanceof JTextField )
{
StyleId.comboboxEditor.at ( comboBox ).set ( ( JTextField ) editorComponent );
}
}
}
/**
* Constructs and returns separator to be placed between renderer/editor and button.
*
* @return separator to be placed between renderer/editor and button
*/
protected JSeparator createSeparator ()
{
return new ComboBoxSeparator ();
}
@Override
protected JButton createArrowButton ()
{
return new ComboBoxButton ();
}
@Override
protected ComboPopup createPopup ()
{
return new BasicComboPopup ( comboBox )
{
@Override
protected JList createList ()
{
final WebList list = new WebList ( comboBox.getModel () )
{
@Override
public void processMouseEvent ( MouseEvent e )
{
if ( CoreSwingUtils.isMenuShortcutKeyDown ( e ) )
{
/**
* Fix for 4234053. Filter out the Control Key from the list.
* ie., don't allow CTRL key deselection.
*/
final Toolkit toolkit = Toolkit.getDefaultToolkit ();
e = new MouseEvent ( ( Component ) e.getSource (), e.getID (), e.getWhen (),
e.getModifiers () ^ toolkit.getMenuShortcutKeyMask (),
e.getX (), e.getY (),
e.getXOnScreen (), e.getYOnScreen (),
e.getClickCount (),
e.isPopupTrigger (),
MouseEvent.NOBUTTON );
}
super.processMouseEvent ( e );
}
};
// Adding combobox reference for internal usage
list.putClientProperty ( COMBOBOX_INSTANCE, comboBox );
// todo Handle inside of the popup painter
// Custom listener to update popup menu dropdown corner
// list.addListSelectionListener ( new ListSelectionListener ()
// {
// @Override
// public void valueChanged ( final ListSelectionEvent e )
// {
// // Checking that popup is still displaying on screen
// if ( isShowing () && getUI () instanceof WebPopupMenuUI )
// {
// // Only do additional repaints for dropdown-styled menu
// final WebPopupMenuUI ui = ( WebPopupMenuUI ) getUI ();
// if ( ui.getPopupStyle () == PopupStyle.dropdown )
// {
// // Retrieving menu and combobox position on screen and deciding which side to repaint
// final int py = getLocationOnScreen ().y;
// final int cbi = comboBox.getLocationOnScreen ().y;
// final Insets pi = getInsets ();
// if ( py > cbi )
// {
// // Repainting top corner area
// repaint ( 0, 0, getWidth (), pi.top );
// }
// else
// {
// // Repainting bottom corner area
// repaint ( 0, getHeight () - pi.bottom, getWidth (), pi.bottom );
// }
// }
// }
// }
// } );
return list;
}
@Override
protected void configureList ()
{
list.setFocusable ( false );
list.setFont ( comboBox.getFont () );
list.setCellRenderer ( comboBox.getRenderer () );
list.setSelectionMode ( ListSelectionModel.SINGLE_SELECTION );
setListSelection ( comboBox.getSelectedIndex () );
installListListeners ();
}
@Override
protected JScrollPane createScroller ()
{
// todo This should be better implemented later on in a custom popup
StyleId.comboboxPopup.at ( comboBox ).set ( this );
final WebScrollPane scroll = new WebScrollPane ( StyleId.comboboxPopupScrollPane.at ( this ), list );
scroll.setHorizontalScrollBar ( null );
// Custom list styling
StyleId.comboboxPopupList.at ( scroll ).set ( list );
return scroll;
}
@Override
protected void configureScroller ()
{
scroller.setFocusable ( false );
scroller.getVerticalScrollBar ().setFocusable ( false );
}
@Override
protected void configurePopup ()
{
setOpaque ( false );
setDoubleBuffered ( true );
setFocusable ( false );
setLayout ( new BoxLayout ( this, BoxLayout.Y_AXIS ) );
add ( scroller );
}
@Override
public void show ()
{
comboBox.firePopupMenuWillBecomeVisible ();
setListSelection ( comboBox.getSelectedIndex () );
final Point location = getPopupLocation ();
show ( comboBox, location.x, location.y );
}
/**
* Sets list selection to the specified index.
*
* @param index selected index
*/
private void setListSelection ( final int index )
{
if ( index == -1 )
{
list.clearSelection ();
}
else
{
list.setSelectedIndex ( index );
list.ensureIndexIsVisible ( index );
}
}
/**
* Returns combobox popup location.
*
* @return combobox popup location
*/
private Point getPopupLocation ()
{
final Dimension comboSize = comboBox.getSize ();
if ( isWidePopup () )
{
final Dimension prefSize = comboBox.getPreferredSize ();
if ( prefSize.width > comboSize.width )
{
comboSize.width = prefSize.width;
}
}
comboSize.setSize ( comboSize.width - 2, getPopupHeightForRowCount ( comboBox.getMaximumRowCount () ) );
final Rectangle popupBounds = computePopupBounds ( 0, comboBox.getBounds ().height, comboSize.width, comboSize.height );
final Dimension scrollSize = popupBounds.getSize ();
scroller.setMaximumSize ( scrollSize );
scroller.setPreferredSize ( scrollSize );
scroller.setMinimumSize ( scrollSize );
list.revalidate ();
return popupBounds.getLocation ();
}
};
}
@Override
public boolean isWidePopup ()
{
return widePopup != null && widePopup;
}
@Override
public void setWidePopup ( final boolean wide )
{
this.widePopup = wide;
}
@Override
public JList getListBox ()
{
return listBox;
}
@Override
public boolean contains ( final JComponent c, final int x, final int y )
{
return PainterSupport.contains ( c, this, painter, x, y );
}
@Override
public void paint ( final Graphics g, final JComponent c )
{
if ( painter != null )
{
painter.prepareToPaint ( currentValuePane );
painter.paint ( ( Graphics2D ) g, c, this, new Bounds ( c ) );
}
}
@Override
protected LayoutManager createLayoutManager ()
{
return new WebComboBoxLayout ();
}
@Override
public Dimension getMinimumSize ( final JComponent c )
{
if ( !isMinimumSizeDirty )
{
return new Dimension ( cachedMinimumSize );
}
final Dimension size = getDisplaySize ();
final Insets insets = getInsets ();
// Insets sizes
size.height += insets.top + insets.bottom;
size.width += insets.left + insets.right;
// Arrow button width
if ( arrowButton != null )
{
size.width += arrowButton.getPreferredSize ().width;
// Separator width
if ( separator != null )
{
size.width += separator.getPreferredSize ().width;
}
}
// Saving resulting size
cachedMinimumSize.setSize ( size.width, size.height );
isMinimumSizeDirty = false;
return new Dimension ( size );
}
@Override
protected Dimension getDefaultSize ()
{
// todo Use current renderer instead?
// Calculates the height and width using the default text renderer
return getSizeForComponent ( getDefaultListCellRenderer ().getListCellRendererComponent ( listBox, " ", -1, false, false ) );
}
/**
* Returns renderer component preferred size.
*
* @param c renderer component
* @return renderer component preferred size
*/
protected Dimension getSizeForComponent ( final Component c )
{
currentValuePane.add ( c );
c.setFont ( comboBox.getFont () );
final Dimension d = c.getPreferredSize ();
currentValuePane.remove ( c );
return d;
}
@Override
public Dimension getPreferredSize ( final JComponent c )
{
return PainterSupport.getPreferredSize ( c, super.getPreferredSize ( c ), painter, true );
}
@Override
protected Rectangle rectangleForCurrentValue ()
{
return getValueBounds ();
}
@Override
public Rectangle getValueBounds ()
{
final int width = comboBox.getWidth ();
final int height = comboBox.getHeight ();
final Insets i = comboBox.getInsets ();
final boolean ltr = comboBox.getComponentOrientation ().isLeftToRight ();
int side = 0;
if ( arrowButton != null )
{
side += arrowButton.getPreferredSize ().width;
if ( separator != null )
{
side += separator.getPreferredSize ().width;
}
}
return new Rectangle ( ltr ? i.left : i.left + side, i.top, width - i.left - i.right - side, height - i.top - i.bottom );
}
/**
* Returns the area that is reserved for drawing separator between currently selected item and arrow button.
*
* @return area that is reserved for drawing between currently selected item and arrow button
*/
public Rectangle getSeparatorBounds ()
{
final int width = comboBox.getWidth ();
final int height = comboBox.getHeight ();
final Insets i = comboBox.getInsets ();
final boolean ltr = comboBox.getComponentOrientation ().isLeftToRight ();
final int sep = separator.getPreferredSize ().width;
int button = 0;
if ( arrowButton != null )
{
button += arrowButton.getPreferredSize ().width;
}
return new Rectangle ( ltr ? width - i.right - button - sep : i.left + button, i.top, sep, height - i.top - i.bottom );
}
/**
* Returns the area that is reserved for drawing arrow button.
*
* @return area that is reserved for drawing arrow button
*/
public Rectangle getArrowButtonBounds ()
{
final int width = comboBox.getWidth ();
final int height = comboBox.getHeight ();
final Insets i = comboBox.getInsets ();
final boolean ltr = comboBox.getComponentOrientation ().isLeftToRight ();
final int button = arrowButton.getPreferredSize ().width;
return new Rectangle ( ltr ? width - i.right - button : i.left, i.top, button, height - i.top - i.bottom );
}
@Override
public void addEditor ()
{
super.addEditor ();
// Inform about editability change
// This is thrown here instead of property change listener to ensure fires order is preserved
fireEditabilityChanged ( true );
}
@Override
protected void configureEditor ()
{
super.configureEditor ();
// Applying proper editor orientation on display
editor.applyComponentOrientation ( comboBox.getComponentOrientation () );
}
@Override
public void removeEditor ()
{
super.removeEditor ();
// Inform about editability change
// This is thrown here instead of property change listener to ensure fires order is preserved
fireEditabilityChanged ( false );
}
@Override
public void addEditabilityListener ( final EditabilityListener listener )
{
listenerList.add ( EditabilityListener.class, listener );
}
@Override
public void removeEditabilityListener ( final EditabilityListener listener )
{
listenerList.remove ( EditabilityListener.class, listener );
}
/**
* Informs about combobox editability change.
*
* @param editable whether or not combobox is editable.
*/
public void fireEditabilityChanged ( final boolean editable )
{
for ( final EditabilityListener listener : listenerList.getListeners ( EditabilityListener.class ) )
{
listener.editabilityChanged ( editable );
}
}
@Override
public void addPopupVisibilityListener ( final VisibilityListener listener )
{
listenerList.add ( VisibilityListener.class, listener );
}
@Override
public void removePopupVisibilityListener ( final VisibilityListener listener )
{
listenerList.remove ( VisibilityListener.class, listener );
}
/**
* Informs about combobox popup visibility change.
*
* @param visible whether or not combobox popup is visible
*/
public void firePopupVisibilityChanged ( final boolean visible )
{
for ( final VisibilityListener listener : listenerList.getListeners ( VisibilityListener.class ) )
{
listener.visibilityChanged ( visible );
}
}
/**
* Custom layout manager for WebComboBoxUI.
*/
protected class WebComboBoxLayout extends AbstractLayoutManager
{
@Override
public void layoutContainer ( @NotNull final Container container )
{
// Arrow button
if ( arrowButton != null )
{
arrowButton.setBounds ( getArrowButtonBounds () );
// Separator
if ( separator != null )
{
separator.setBounds ( getSeparatorBounds () );
}
}
// Value editor
if ( editor != null )
{
editor.setBounds ( getValueBounds () );
}
}
@Override
public Dimension minimumLayoutSize ( @NotNull final Container container )
{
return container.getMinimumSize ();
}
@Override
public Dimension preferredLayoutSize ( @NotNull final Container container )
{
return container.getPreferredSize ();
}
}
/**
* Custom combobox separator placed between renderer/editor and button.
*/
protected class ComboBoxSeparator extends WebSeparator implements Stateful, EditabilityListener, VisibilityListener
{
/**
* Constructs new combobox separator.
*/
public ComboBoxSeparator ()
{
super ( StyleId.comboboxSeparator.at ( comboBox ), HORIZONTAL );
}
/**
* Installs custom combobox listeners.
*/
public void install ()
{
addEditabilityListener ( this );
addPopupVisibilityListener ( this );
}
/**
* Uninstalls custom combobox listeners.
*/
public void uninstall ()
{
removePopupVisibilityListener ( this );
removeEditabilityListener ( this );
}
@Override
public void visibilityChanged ( final boolean visible )
{
DecorationUtils.fireStatesChanged ( this );
}
@Override
public void editabilityChanged ( final boolean editable )
{
DecorationUtils.fireStatesChanged ( this );
}
@Nullable
@Override
public List getStates ()
{
final List states = new ArrayList ( 1 );
if ( comboBox.isEditable () )
{
states.add ( DecorationState.editable );
}
states.add ( isPopupVisible ( comboBox ) ? DecorationState.expanded : DecorationState.collapsed );
return states;
}
}
/**
* Custom combobox button used to display popup menu arrow.
*/
protected class ComboBoxButton extends WebButton implements Stateful, EditabilityListener, VisibilityListener
{
/**
* Constructs new combobox button.
*/
public ComboBoxButton ()
{
super ( StyleId.comboboxArrowButton.at ( comboBox ) );
setName ( "ComboBox.arrowButton" );
}
/**
* Installs custom combobox listeners.
*/
public void install ()
{
addEditabilityListener ( this );
addPopupVisibilityListener ( this );
}
/**
* Uninstalls custom combobox listeners.
*/
public void uninstall ()
{
removePopupVisibilityListener ( this );
removeEditabilityListener ( this );
}
@Override
public void setFocusable ( final boolean focusable )
{
// Workaround to completely disable focusability of this button
super.setFocusable ( false );
}
@Override
public void visibilityChanged ( final boolean visible )
{
DecorationUtils.fireStatesChanged ( this );
}
@Override
public void editabilityChanged ( final boolean editable )
{
DecorationUtils.fireStatesChanged ( this );
}
@Nullable
@Override
public List getStates ()
{
final List states = new ArrayList ( 1 );
if ( comboBox.isEditable () )
{
states.add ( DecorationState.editable );
}
states.add ( isPopupVisible ( comboBox ) ? DecorationState.expanded : DecorationState.collapsed );
return states;
}
}
/**
* Returns default list cell renderer instance.
*
* @return default list cell renderer instance
*/
protected static ListCellRenderer getDefaultListCellRenderer ()
{
if ( DEFAULT_RENDERER == null )
{
DEFAULT_RENDERER = new WebComboBoxRenderer.UIResource> ();
}
return DEFAULT_RENDERER;
}
}