All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.formdev.flatlaf.ui.FlatTableHeaderUI Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2019 FormDev Software GmbH
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.formdev.flatlaf.ui;

import java.awt.Color;
import java.awt.Component;
import java.awt.Container;
import java.awt.Cursor;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Insets;
import java.awt.Rectangle;
import java.awt.event.MouseEvent;
import java.awt.geom.Rectangle2D;
import java.beans.PropertyChangeListener;
import java.util.Map;
import javax.swing.CellRendererPane;
import javax.swing.Icon;
import javax.swing.JComponent;
import javax.swing.JLabel;
import javax.swing.JTable;
import javax.swing.SwingConstants;
import javax.swing.UIManager;
import javax.swing.event.MouseInputListener;
import javax.swing.plaf.ComponentUI;
import javax.swing.plaf.basic.BasicTableHeaderUI;
import javax.swing.table.JTableHeader;
import javax.swing.table.TableCellRenderer;
import javax.swing.table.TableColumn;
import javax.swing.table.TableColumnModel;
import com.formdev.flatlaf.ui.FlatStylingSupport.Styleable;
import com.formdev.flatlaf.ui.FlatStylingSupport.StyleableUI;
import com.formdev.flatlaf.util.HiDPIUtils;
import com.formdev.flatlaf.util.LoggingFacade;
import com.formdev.flatlaf.util.UIScale;

/**
 * Provides the Flat LaF UI delegate for {@link javax.swing.table.JTableHeader}.
 *
 * 
 *
 * @uiDefault TableHeader.font					Font
 * @uiDefault TableHeader.background			Color
 * @uiDefault TableHeader.foreground			Color
 *
 * 
 *
 * @uiDefault TableHeader.hoverBackground		Color	optional
 * @uiDefault TableHeader.hoverForeground		Color	optional
 * @uiDefault TableHeader.pressedBackground		Color	optional
 * @uiDefault TableHeader.pressedForeground		Color	optional
 * @uiDefault TableHeader.bottomSeparatorColor	Color
 * @uiDefault TableHeader.height				int
 * @uiDefault TableHeader.sortIconPosition		String	right (default), left, top or bottom
 *
 * 
 *
 * @uiDefault TableHeader.cellMargins			Insets
 * @uiDefault TableHeader.separatorColor		Color
 * @uiDefault TableHeader.bottomSeparatorColor	Color
 * @uiDefault TableHeader.showTrailingVerticalLine	boolean
 *
 * 
 *
 * @uiDefault Component.arrowType				String	chevron (default) or triangle
 * @uiDefault Table.sortIconColor				Color
 *
 * @author Karl Tauber
 */
public class FlatTableHeaderUI
	extends BasicTableHeaderUI
	implements StyleableUI
{
	/** @since 3.1 */ @Styleable protected Color hoverBackground;
	/** @since 3.1 */ @Styleable protected Color hoverForeground;
	/** @since 3.1 */ @Styleable protected Color pressedBackground;
	/** @since 3.1 */ @Styleable protected Color pressedForeground;
	@Styleable protected Color bottomSeparatorColor;
	@Styleable protected int height;
	@Styleable(type=String.class) protected int sortIconPosition;

	// for FlatTableHeaderBorder
	/** @since 2 */ @Styleable protected Insets cellMargins;
	/** @since 2 */ @Styleable protected Color separatorColor;
	/** @since 2 */ @Styleable protected Boolean showTrailingVerticalLine;

	// for FlatAscendingSortIcon and FlatDescendingSortIcon
	// (needs to be public because icon classes are in another package)
	/** @since 2 */ @Styleable public String arrowType;
	/** @since 2 */ @Styleable public Color sortIconColor;

	private PropertyChangeListener propertyChangeListener;
	private Map oldStyleValues;

	public static ComponentUI createUI( JComponent c ) {
		return new FlatTableHeaderUI();
	}

	@Override
	public void installUI( JComponent c ) {
		super.installUI( c );

		// replace cell renderer pane
		header.remove( rendererPane );
		rendererPane = new FlatTableHeaderCellRendererPane();
		header.add( rendererPane );

		installStyle();
	}

	@Override
	protected void installDefaults() {
		super.installDefaults();

		hoverBackground = UIManager.getColor( "TableHeader.hoverBackground" );
		hoverForeground = UIManager.getColor( "TableHeader.hoverForeground" );
		pressedBackground = UIManager.getColor( "TableHeader.pressedBackground" );
		pressedForeground = UIManager.getColor( "TableHeader.pressedForeground" );
		bottomSeparatorColor = UIManager.getColor( "TableHeader.bottomSeparatorColor" );
		height = UIManager.getInt( "TableHeader.height" );
		sortIconPosition = parseSortIconPosition( UIManager.getString( "TableHeader.sortIconPosition" ) );
	}

	@Override
	protected void uninstallDefaults() {
		super.uninstallDefaults();

		hoverBackground = null;
		hoverForeground = null;
		pressedBackground = null;
		pressedForeground = null;
		bottomSeparatorColor = null;

		oldStyleValues = null;
	}

	@Override
	protected void installListeners() {
		super.installListeners();

		propertyChangeListener = FlatStylingSupport.createPropertyChangeListener( header, this::installStyle, null );
		header.addPropertyChangeListener( propertyChangeListener );
	}

	@Override
	protected void uninstallListeners() {
		super.uninstallListeners();

		header.removePropertyChangeListener( propertyChangeListener );
		propertyChangeListener = null;
	}

	/** @since 2 */
	protected void installStyle() {
		try {
			applyStyle( FlatStylingSupport.getResolvedStyle( header, "TableHeader" ) );
		} catch( RuntimeException ex ) {
			LoggingFacade.INSTANCE.logSevere( null, ex );
		}
	}

	/** @since 2 */
	protected void applyStyle( Object style ) {
		oldStyleValues = FlatStylingSupport.parseAndApply( oldStyleValues, style, this::applyStyleProperty );
	}

	/** @since 2 */
	protected Object applyStyleProperty( String key, Object value ) {
		if( key.equals( "sortIconPosition" ) && value instanceof String )
			value = parseSortIconPosition( (String) value );

		return FlatStylingSupport.applyToAnnotatedObjectOrComponent( this, header, key, value );
	}

	/** @since 2 */
	@Override
	public Map> getStyleableInfos( JComponent c ) {
		return FlatStylingSupport.getAnnotatedStyleableInfos( this );
	}

	/** @since 2.5 */
	@Override
	public Object getStyleableValue( JComponent c, String key ) {
		if( key.equals( "sortIconPosition" ) ) {
			switch( sortIconPosition ) {
				default:
				case SwingConstants.RIGHT:	return "right";
				case SwingConstants.LEFT:	return "left";
				case SwingConstants.TOP:	return "top";
				case SwingConstants.BOTTOM:	return "bottom";
			}
		}

		return FlatStylingSupport.getAnnotatedStyleableValue( this, key );
	}

	private static int parseSortIconPosition( String str ) {
		if( str == null )
			str = "";

		switch( str ) {
			default:
			case "right":	return SwingConstants.RIGHT;
			case "left":	return SwingConstants.LEFT;
			case "top":		return SwingConstants.TOP;
			case "bottom":	return SwingConstants.BOTTOM;
		}
	}

	@Override
	protected MouseInputListener createMouseInputListener() {
		return new FlatMouseInputHandler();
	}

	// overridden and made public to allow usage in custom renderers
	@Override
	public int getRolloverColumn() {
		return super.getRolloverColumn();
	}

	@Override
	protected void rolloverColumnUpdated( int oldColumn, int newColumn ) {
		HiDPIUtils.repaint( header, header.getHeaderRect( oldColumn ) );
		HiDPIUtils.repaint( header, header.getHeaderRect( newColumn ) );
	}

	@Override
	public void paint( Graphics g, JComponent c ) {
		fixDraggedAndResizingColumns( header );

		TableColumnModel columnModel = header.getColumnModel();
		if( columnModel.getColumnCount() <= 0 )
			return;

		// compute total width of all columns
		int columnCount = columnModel.getColumnCount();
		int totalWidth = 0;
		for( int i = 0; i < columnCount; i++ )
			totalWidth += columnModel.getColumn( i ).getWidth();

		if( totalWidth < header.getWidth() ) {
			// do not paint bottom separator if JTableHeader.setDefaultRenderer() was used
			TableCellRenderer defaultRenderer = header.getDefaultRenderer();
			boolean paintBottomSeparator = isSystemDefaultRenderer( defaultRenderer );
			if( !paintBottomSeparator && header.getTable() != null ) {
				// check whether the renderer delegates to the system default renderer
				Component rendererComponent = defaultRenderer.getTableCellRendererComponent(
					header.getTable(), "", false, false, -1, 0 );
				paintBottomSeparator = isSystemDefaultRenderer( rendererComponent );
			}

			if( paintBottomSeparator ) {
				int w = c.getWidth() - totalWidth;
				int x = header.getComponentOrientation().isLeftToRight() ? c.getWidth() - w : 0;
				paintBottomSeparator( g, c, x, w );
			}
		}

		// paint header
		super.paint( g, c );
	}

	private boolean isSystemDefaultRenderer( Object headerRenderer ) {
		String rendererClassName = headerRenderer.getClass().getName();
		return rendererClassName.equals( "sun.swing.table.DefaultTableCellHeaderRenderer" ) ||
			   rendererClassName.equals( "sun.swing.FilePane$AlignableTableHeaderRenderer" );
	}

	protected void paintBottomSeparator( Graphics g, JComponent c, int x, int w ) {
		float lineWidth = UIScale.scale( 1f );

		Graphics2D g2 = (Graphics2D) g.create();
		try {
			FlatUIUtils.setRenderingHints( g2 );

			// paint bottom line
			g2.setColor( bottomSeparatorColor );
			g2.fill( new Rectangle2D.Float( x, c.getHeight() - lineWidth, w, lineWidth ) );
		} finally {
			g2.dispose();
		}
	}

	@Override
	public Dimension getPreferredSize( JComponent c ) {
		Dimension size = super.getPreferredSize( c );
		if( size.height > 0 )
			size.height = Math.max( size.height, UIScale.scale( height ) );
		return size;
	}

	static void fixDraggedAndResizingColumns( JTableHeader header ) {
		if( header == null )
			return;

		// Dragged column may be outdated in the case that the table structure
		// was changed from a table header popup menu action. In this case
		// the paint methods in BasicTableHeaderUI and BasicTableUI would throw exceptions.
		TableColumn draggedColumn = header.getDraggedColumn();
		if( draggedColumn != null && !isValidColumn( header.getColumnModel(), draggedColumn ) )
			header.setDraggedColumn( null );

		// also clear outdated resizing column (although this seems not cause exceptions)
		TableColumn resizingColumn = header.getResizingColumn();
		if( resizingColumn != null && !isValidColumn( header.getColumnModel(), resizingColumn ) )
			header.setResizingColumn( null );
	}

	private static boolean isValidColumn( TableColumnModel cm, TableColumn column ) {
		int count = cm.getColumnCount();
		for( int i = 0; i < count; i++ ) {
			if( cm.getColumn( i ) == column )
				return true;
		}
		return false;
	}

	//---- class FlatTableHeaderCellRendererPane ------------------------------

	/**
	 * Cell renderer pane that is used to paint hover and pressed background/foreground
	 * and to paint sort arrows at top, bottom or left position.
	 */
	private class FlatTableHeaderCellRendererPane
		extends CellRendererPane
	{
		private final Icon ascendingSortIcon;
		private final Icon descendingSortIcon;

		public FlatTableHeaderCellRendererPane() {
			ascendingSortIcon = UIManager.getIcon( "Table.ascendingSortIcon" );
			descendingSortIcon = UIManager.getIcon( "Table.descendingSortIcon" );
		}

		@Override
		public void paintComponent( Graphics g, Component c, Container p, int x, int y, int w, int h, boolean shouldValidate ) {
			if( !(c instanceof JLabel) ) {
				super.paintComponent( g, c, p, x, y, w, h, shouldValidate );
				return;
			}

			JLabel l = (JLabel) c;
			Color oldBackground = null;
			Color oldForeground = null;
			boolean oldOpaque = false;
			Icon oldIcon = null;
			int oldHorizontalTextPosition = -1;

			// hover and pressed background/foreground
			TableColumn draggedColumn = header.getDraggedColumn();
			Color background = null;
			Color foreground = null;
			if( draggedColumn != null &&
				header.getTable().convertColumnIndexToView( draggedColumn.getModelIndex() )
					== getColumn( x - header.getDraggedDistance(), w ) )
			{
				background = pressedBackground;
				foreground = pressedForeground;
			} else if( getRolloverColumn() >= 0 && getRolloverColumn() == getColumn( x, w ) ) {
				background = hoverBackground;
				foreground = hoverForeground;
			}
			if( background != null ) {
				oldBackground = l.getBackground();
				oldOpaque = l.isOpaque();
				l.setBackground( FlatUIUtils.deriveColor( background, header.getBackground() ) );
				l.setOpaque( true );
			}
			if( foreground != null ) {
				oldForeground = l.getForeground();
				l.setForeground( FlatUIUtils.deriveColor( foreground, header.getForeground() ) );
			}

			// sort icon position
			Icon icon = l.getIcon();
			boolean isSortIcon = (icon != null && (icon == ascendingSortIcon || icon == descendingSortIcon));
			if( isSortIcon ) {
				if( sortIconPosition == SwingConstants.LEFT ) {
					// left
					oldHorizontalTextPosition = l.getHorizontalTextPosition();
					l.setHorizontalTextPosition( SwingConstants.RIGHT );
				} else if( sortIconPosition == SwingConstants.TOP || sortIconPosition == SwingConstants.BOTTOM ) {
					// top or bottom
					oldIcon = icon;
					l.setIcon( null );
				}
			}

			// paint renderer component
			super.paintComponent( g, c, p, x, y, w, h, shouldValidate );

			// paint top or bottom sort icon
			if( isSortIcon && (sortIconPosition == SwingConstants.TOP || sortIconPosition == SwingConstants.BOTTOM) ) {
				int xi = x + ((w - icon.getIconWidth()) / 2);
				int yi = (sortIconPosition == SwingConstants.TOP)
					? y + UIScale.scale( 1 )
					: y + height - icon.getIconHeight()
						- 1 // for gap
						- (int) (1 * UIScale.getUserScaleFactor()); // for bottom border
				icon.paintIcon( c, g, xi, yi );
			}

			// restore modified renderer component properties
			if( background != null ) {
				l.setBackground( oldBackground );
				l.setOpaque( oldOpaque );
			}
			if( foreground != null )
				l.setForeground( oldForeground );
			if( oldIcon != null )
				l.setIcon( oldIcon );
			if( oldHorizontalTextPosition >= 0 )
				l.setHorizontalTextPosition( oldHorizontalTextPosition );
		}

		/**
		 * Get column index for given coordinates.
		 */
		private int getColumn( int x, int width ) {
			TableColumnModel columnModel = header.getColumnModel();
			int columnCount = columnModel.getColumnCount();
			boolean ltr = header.getComponentOrientation().isLeftToRight();
			int cx = ltr ? 0 : getWidthInRightToLef();

			for( int i = 0; i < columnCount; i++ ) {
				int cw = columnModel.getColumn( i ).getWidth();
				if( x == cx - (ltr ? 0 : cw) && width == cw )
					return i;

				cx += ltr ? cw : -cw;
			}
			return -1;
		}

		// similar to JTableHeader.getWidthInRightToLeft()
		private int getWidthInRightToLef() {
			JTable table = header.getTable();
			return (table != null && table.getAutoResizeMode() != JTable.AUTO_RESIZE_OFF)
				? table.getWidth()
				: header.getWidth();
		}
	}

	//---- class FlatMouseInputHandler ----------------------------------------

	/** @since 1.6 */
	protected class FlatMouseInputHandler
		extends MouseInputHandler
	{
		Cursor oldCursor;

		@Override
		public void mouseMoved( MouseEvent e ) {
			// restore old cursor, which is necessary because super.mouseMoved() swaps cursors
			if( oldCursor != null ) {
				header.setCursor( oldCursor );
				oldCursor = null;
			}

			super.mouseMoved( e );

			// if resizing last column is not possible, then Swing still shows a resize cursor,
			// which can be confusing for the user --> change cursor to standard cursor
			JTable table;
			int column;
			if( header.isEnabled() &&
				(table = header.getTable()) != null &&
				table.getAutoResizeMode() != JTable.AUTO_RESIZE_OFF &&
				header.getCursor() == Cursor.getPredefinedCursor( Cursor.E_RESIZE_CURSOR ) &&
				(column = header.columnAtPoint( e.getPoint() )) >= 0 &&
				column == header.getColumnModel().getColumnCount() - 1 )
			{
				// mouse is in last column
				Rectangle r = header.getHeaderRect( column );
				r.grow( -3, 0 );
				if( !r.contains( e.getX(), e.getY() ) ) {
					// mouse is in left or right resize area of last column
					boolean isResizeLastColumn = (e.getX() >= r.x + (r.width / 2));
					if( !header.getComponentOrientation().isLeftToRight() )
						isResizeLastColumn = !isResizeLastColumn;

					if( isResizeLastColumn ) {
						// resize is not possible --> change cursor to standard cursor
						oldCursor = header.getCursor();
						header.setCursor( Cursor.getDefaultCursor() );
					}
				}
			}
		}
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy