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

com.alee.extended.inspector.ComponentHighlighter 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.extended.inspector;

import com.alee.api.annotations.NotNull;
import com.alee.api.annotations.Nullable;
import com.alee.api.data.CompassDirection;
import com.alee.api.jdk.Objects;
import com.alee.extended.behavior.VisibilityBehavior;
import com.alee.laf.WebLookAndFeel;
import com.alee.managers.glasspane.GlassPaneManager;
import com.alee.managers.glasspane.WebGlassPane;
import com.alee.painter.PainterSupport;
import com.alee.painter.decoration.shape.Round;
import com.alee.painter.decoration.shape.ShapeType;
import com.alee.painter.decoration.shape.Sides;
import com.alee.utils.*;

import javax.swing.*;
import java.awt.*;
import java.awt.event.ComponentEvent;
import java.awt.event.ComponentListener;
import java.awt.event.HierarchyBoundsListener;
import java.awt.event.HierarchyEvent;
import java.awt.geom.Area;
import java.util.Map;

/**
 * Custom component that visually displays component margin, border, padding and content areas.
 * It is designed to be used only as a part of {@link InterfaceTree}.
 *
 * @author Mikle Garin
 */
public final class ComponentHighlighter extends JComponent implements ComponentListener, HierarchyBoundsListener
{
    /**
     * Basic color scheme.
     */
    public static final Color marginColor = new Color ( 255, 255, 0, 50 );
    public static final Color borderColor = new Color ( 255, 0, 0, 50 );
    public static final Color paddingColor = new Color ( 0, 200, 0, 50 );
    public static final Color contentColor = new Color ( 0, 0, 255, 50 );

    /**
     * Internal constants.
     */
    private static final Insets emptyInsets = new Insets ( 0, 0, 0, 0 );
    private static final int sizeTipHeight = 20;

    /**
     * Highlighted component.
     */
    @Nullable
    private transient Component component;

    /**
     * Highlighter glasspane.
     */
    @Nullable
    private transient WebGlassPane glassPane;

    /**
     * Highlighted component visibility listener.
     */
    @Nullable
    private transient VisibilityBehavior visibilityListener;

    /**
     * Constructs new component highlighter.
     */
    public ComponentHighlighter ()
    {
        super ();
        setOpaque ( false );
        setFont ( WebLookAndFeel.globalTextFont );
    }

    /**
     * Displays this highlighter over the specified component.
     *
     * @param component component to be highlighted
     */
    public void install ( @NotNull final Component component )
    {
        if ( this.component == null && this.glassPane == null && visibilityListener == null )
        {
            this.component = Objects.requireNonNull ( component, "Component cannot be null" );

            // Retrieving glass pane
            final WebGlassPane webGlassPane = GlassPaneManager.getGlassPane ( component );
            this.glassPane = Objects.requireNonNull ( webGlassPane, "Component window must have a glass pane" );

            // Updating highligther position
            updateBounds ();

            // Adding highligthed component listeners
            component.addComponentListener ( this );
            component.addHierarchyBoundsListener ( this );
            visibilityListener = new VisibilityBehavior ( component )
            {
                @Override
                protected void hidden ( @NotNull final Component component )
                {
                    ComponentHighlighter.this.uninstall ();
                }
            };
            visibilityListener.install ();

            // Displaying highlighter on glass pane
            this.glassPane.showComponent ( this );
        }
    }

    /**
     * Hides this component highlighter.
     */
    public void uninstall ()
    {
        if ( component != null && glassPane != null && visibilityListener != null )
        {
            // Hiding highlighter from glass pane
            glassPane.hideComponent ( this );
            glassPane = null;

            // Removing listeners and references
            visibilityListener.uninstall ();
            visibilityListener = null;
            component.removeHierarchyBoundsListener ( this );
            component.removeComponentListener ( this );
            component = null;
        }
    }

    /**
     * Returns highlighted component.
     *
     * @return highlighted component
     */
    @Nullable
    public Component getComponent ()
    {
        return component;
    }

    @Override
    public void ancestorMoved ( @NotNull final HierarchyEvent e )
    {
        updateBounds ();
    }

    @Override
    public void ancestorResized ( @NotNull final HierarchyEvent e )
    {
        updateBounds ();
    }

    @Override
    public void componentResized ( @NotNull final ComponentEvent e )
    {
        updateBounds ();
    }

    @Override
    public void componentMoved ( @NotNull final ComponentEvent e )
    {
        updateBounds ();
    }

    @Override
    public void componentShown ( @NotNull final ComponentEvent e )
    {
        // Ignored event
    }

    @Override
    public void componentHidden ( @NotNull final ComponentEvent e )
    {
        uninstall ();
    }

    /**
     * Updates highlighter position.
     */
    private void updateBounds ()
    {
        if ( component != null && glassPane != null )
        {
            final Dimension glassPaneSize = glassPane.getSize ();
            final Rectangle componentBounds = CoreSwingUtils.getRelativeBounds ( component, glassPane );
            final int tipWidth = getTipWidth ();
            final CompassDirection tipDirection = getTipDirection ( tipWidth, glassPaneSize, componentBounds );
            final Rectangle tipBounds = getTipBounds ( true, tipWidth, tipDirection, componentBounds );
            final Rectangle bodyBounds = getBodyBounds ( true, tipWidth, tipDirection, componentBounds );
            setBounds ( GeometryUtils.getNonNullContainingRect ( tipBounds, bodyBounds ) );
        }
    }

    /**
     * Returns tip width.
     *
     * @return tip width
     */
    private int getTipWidth ()
    {
        return getFontMetrics ( getFont () ).stringWidth ( getSizeTip () ) + 8;
    }

    /**
     * Returns tip {@link CompassDirection}.
     *
     * @param tipWidth        tip width
     * @param glassPaneSize   glass pane size
     * @param componentBounds inspected component bounds
     * @return tip {@link CompassDirection}
     */
    @NotNull
    private CompassDirection getTipDirection ( final int tipWidth, @NotNull final Dimension glassPaneSize,
                                               @NotNull final Rectangle componentBounds )
    {
        final CompassDirection direction;
        final boolean fromLeft = componentBounds.x + tipWidth < glassPaneSize.width ||
                componentBounds.x + componentBounds.width - tipWidth < 0;
        if ( componentBounds.y > sizeTipHeight )
        {
            direction = fromLeft ? CompassDirection.northWest : CompassDirection.northEast;
        }
        else
        {
            direction = fromLeft ? CompassDirection.southWest : CompassDirection.southEast;
        }
        return direction;
    }

    /**
     * Returns tip bounds.
     *
     * @param relative        whether bounds should be relative to position on glass pane or not
     * @param tipWidth        tip width
     * @param tipDirection    tip {@link CompassDirection}
     * @param componentBounds inspected component bounds
     * @return tip bounds
     */
    @NotNull
    private Rectangle getTipBounds ( final boolean relative, final int tipWidth, @NotNull final CompassDirection tipDirection,
                                     @NotNull final Rectangle componentBounds )
    {
        final int x;
        if ( tipDirection == CompassDirection.northWest || tipDirection == CompassDirection.southWest )
        {
            x = relative ? componentBounds.x : 0;
        }
        else
        {
            x = relative ? componentBounds.x + componentBounds.width - tipWidth : 0;
        }
        final int y;
        if ( tipDirection == CompassDirection.northEast || tipDirection == CompassDirection.northWest )
        {
            y = relative ? componentBounds.y - sizeTipHeight : 0;
        }
        else
        {
            y = relative ? componentBounds.y + componentBounds.height : componentBounds.height;
        }
        final int w = relative ? tipWidth : tipWidth - 1;
        final int h = relative ? sizeTipHeight : sizeTipHeight - 1;
        return new Rectangle ( x, y, w, h );
    }

    /**
     * Returns body bounds.
     *
     * @param relative        whether bounds should be relative to position on glass pane or not
     * @param tipWidth        tip width
     * @param tipDirection    tip {@link CompassDirection}
     * @param componentBounds inspected component bounds
     * @return body bounds
     */
    @NotNull
    private Rectangle getBodyBounds ( final boolean relative, final int tipWidth, @NotNull final CompassDirection tipDirection,
                                      @NotNull final Rectangle componentBounds )
    {
        final int x;
        if ( tipDirection == CompassDirection.northWest || tipDirection == CompassDirection.southWest )
        {
            x = relative ? componentBounds.x : 0;
        }
        else
        {
            x = relative ? componentBounds.x : tipWidth - componentBounds.width;
        }
        final int y;
        if ( tipDirection == CompassDirection.northEast || tipDirection == CompassDirection.northWest )
        {
            y = relative ? componentBounds.y : sizeTipHeight;
        }
        else
        {
            y = relative ? componentBounds.y : 0;
        }
        final int w = componentBounds.width;
        final int h = componentBounds.height;
        return new Rectangle ( x, y, w, h );
    }

    /**
     * Returns highlighted component size tip text.
     *
     * @return highlighted component size tip text
     */
    @NotNull
    private String getSizeTip ()
    {
        return component != null ? component.getWidth () + " x " + component.getHeight () + " px" : "? x ?";
    }

    @Override
    protected void paintComponent ( @NotNull final Graphics g )
    {
        if ( component != null && glassPane != null )
        {
            paintHighlight ( ( Graphics2D ) g, component, glassPane );
        }
    }

    /**
     * Paints highlight.
     *
     * @param g2d       {@link Graphics2D}
     * @param component {@link Component} to highlight
     * @param glassPane {@link WebGlassPane}
     */
    private void paintHighlight ( @NotNull final Graphics2D g2d, @NotNull final Component component,
                                  @NotNull final WebGlassPane glassPane )
    {
        final Dimension glassPaneSize = glassPane.getSize ();
        final Rectangle componentBounds = CoreSwingUtils.getRelativeBounds ( component, glassPane );
        final int tipWidth = getTipWidth ();
        final CompassDirection tipPosition = getTipDirection ( tipWidth, glassPaneSize, componentBounds );
        final Rectangle tipBounds = getTipBounds ( false, tipWidth, tipPosition, componentBounds );
        final Rectangle bodyBounds = getBodyBounds ( false, tipWidth, tipPosition, componentBounds );

        // Painting size tip
        paintSizeTip ( g2d, component, tipBounds, tipPosition );

        // Painting areas
        paintAreas ( g2d, component, bodyBounds );
    }

    /**
     * Paints component size tip.
     *
     * @param g2d         {@link Graphics2D}
     * @param component   {@link Component} to highlight
     * @param tipBounds   tip bounds
     * @param tipPosition tip position
     */
    private void paintSizeTip ( @NotNull final Graphics2D g2d, @NotNull final Component component, @NotNull final Rectangle tipBounds,
                                @NotNull final CompassDirection tipPosition )
    {
        final FontMetrics fm = g2d.getFontMetrics ( g2d.getFont () );
        final int shearY = LafUtils.getTextCenterShiftY ( fm );

        // Creating shape
        final boolean top = tipPosition == CompassDirection.northWest || tipPosition == CompassDirection.northEast;
        final boolean left = tipPosition == CompassDirection.northWest || tipPosition == CompassDirection.southWest;
        final boolean longTip = component.getWidth () < tipBounds.width;

        final Object aa = GraphicsUtils.setupAntialias ( g2d );

        final Round round = new Round (
                top || !left && longTip ? 5 : 0,
                top || left && longTip ? 5 : 0,
                !top || left && longTip ? 5 : 0,
                !top || !left && longTip ? 5 : 0
        );
        final Sides sides = new Sides ( true );

        g2d.setPaint ( Color.WHITE );
        g2d.fill ( ShapeUtils.createFillShape ( 0, tipBounds, round, sides, ShapeType.background ) );

        g2d.setPaint ( Color.DARK_GRAY );
        g2d.draw ( ShapeUtils.createBorderShape ( 0, tipBounds, round, sides ) );

        GraphicsUtils.restoreAntialias ( g2d, aa );

        // Text
        final Map taa = SwingUtils.setupTextAntialias ( g2d );
        g2d.setPaint ( Color.BLACK );
        g2d.drawString ( getSizeTip (), tipBounds.x + 4, tipBounds.y + tipBounds.height / 2 + 1 + shearY );
        SwingUtils.restoreTextAntialias ( g2d, taa );
    }

    /**
     * Paints component areas.
     *
     * @param g2d        {@link Graphics2D}
     * @param component  {@link Component} to highlight
     * @param bodyBounds body bounds
     */
    private void paintAreas ( @NotNull final Graphics2D g2d, @NotNull final Component component, @NotNull final Rectangle bodyBounds )
    {
        if ( component instanceof JComponent && LafUtils.hasWebLafUI ( ( JComponent ) component ) )
        {
            final JComponent jComponent = ( JComponent ) component;

            // Component margin
            final Insets margin = PainterSupport.getMargin ( jComponent, true );
            final Insets m = margin != null ? margin : emptyInsets;

            // Component padding
            final Insets padding = PainterSupport.getPadding ( jComponent, true );
            final Insets p = padding != null ? padding : emptyInsets;

            // Component painter border
            final Insets insets = jComponent.getInsets ();
            final Insets b;
            if ( insets != null && ( insets.top > 0 || insets.left > 0 || insets.bottom > 0 || insets.right > 0 ) )
            {
                b = new Insets ( insets.top - m.top - p.top, insets.left - m.left - p.left,
                        insets.bottom - m.bottom - p.bottom, insets.right - m.right - p.right );
            }
            else
            {
                b = emptyInsets;
            }

            // Computing area sizes
            final Rectangle sr = new Rectangle ( bodyBounds );
            final Rectangle mr = new Rectangle ( sr.x + m.left, sr.y + m.top,
                    sr.width - m.left - m.right, sr.height - m.top - m.bottom );
            final Rectangle br = new Rectangle ( mr.x + b.left, mr.y + b.top,
                    mr.width - b.left - b.right, mr.height - b.top - b.bottom );
            final Rectangle pr = new Rectangle ( br.x + p.left, br.y + p.top,
                    br.width - p.left - p.right, br.height - p.top - p.bottom );

            // Painting component margin area
            paintComplexArea ( g2d, m, sr, mr, marginColor );

            // Painting component painter border area
            paintComplexArea ( g2d, b, mr, br, borderColor );

            // Painting component padding area
            paintComplexArea ( g2d, p, br, pr, paddingColor );

            // Painting component bounds area
            paintContentArea ( g2d, pr );
        }
        else
        {
            // Painting component bounds area
            paintContentArea ( g2d, bodyBounds );
        }
    }

    /**
     * Paints complex component area.
     *
     * @param g2d    graphics context
     * @param insets area insets
     * @param outer  outer area bounds
     * @param inner  inner area bounds
     * @param color  area color
     */
    private void paintComplexArea ( @NotNull final Graphics2D g2d, @NotNull final Insets insets, @NotNull final Rectangle outer,
                                    @NotNull final Rectangle inner, @NotNull final Color color )
    {
        if ( !insets.equals ( emptyInsets ) )
        {
            g2d.setPaint ( color );
            final Area ma = new Area ( outer );
            ma.exclusiveOr ( new Area ( inner ) );
            g2d.fill ( ma );
        }
    }

    /**
     * Paints component content area.
     *
     * @param g2d    graphics context
     * @param bounds content area bounds
     */
    private void paintContentArea ( @NotNull final Graphics2D g2d, @NotNull final Rectangle bounds )
    {
        g2d.setPaint ( contentColor );
        g2d.fill ( bounds );
    }

    /**
     * Ensure we are as hidden as possible.
     * This will also ensure that other {@link ComponentHighlighter}s cannot see us.
     */
    @Override
    public boolean contains ( final int x, final int y )
    {
        return false;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy