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

com.alee.extended.inspector.InterfaceInspector 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.extended.behavior.VisibilityBehavior;
import com.alee.extended.panel.GroupPanel;
import com.alee.extended.panel.GroupingType;
import com.alee.extended.tree.WebTreeFilterField;
import com.alee.extended.window.PopOverDirection;
import com.alee.extended.window.WebPopOver;
import com.alee.laf.WebLookAndFeel;
import com.alee.laf.button.WebToggleButton;
import com.alee.laf.panel.WebPanel;
import com.alee.laf.scroll.WebScrollPane;
import com.alee.laf.separator.WebSeparator;
import com.alee.laf.window.WebDialog;
import com.alee.laf.window.WebFrame;
import com.alee.managers.hotkey.Hotkey;
import com.alee.managers.icon.Icons;
import com.alee.managers.style.StyleId;
import com.alee.utils.ProprietaryUtils;

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

/**
 * Complex component that provides Swing components structure overview and allows inspecting specific components.
 * It also has a few methods to display as a separate window on top of any UI element.
 *
 * @author Mikle Garin
 * @see InterfaceTree
 * @see ComponentInspectBehavior
 * @see ComponentHighlighter
 * @see #showFrame(Component)
 * @see #showDialog(Component, Component)
 * @see #showPopOver(Component, Component)
 * @see #showPopOver(Component, Component, PopOverDirection)
 */
public class InterfaceInspector extends WebPanel
{
    /**
     * {@link InterfaceTree}.
     */
    @NotNull
    protected final InterfaceTree tree;

    /**
     * {@link Component} inspect behavior.
     */
    @Nullable
    protected transient ComponentInspectBehavior inspectBehavior;

    /**
     * Constructs new empty {@link InterfaceInspector}.
     */
    public InterfaceInspector ()
    {
        this ( StyleId.inspector, null );
    }

    /**
     * Constructs new {@link InterfaceInspector} for the specified {@link Component} and its childrens tree.
     *
     * @param inspected {@link Component} to inspect
     */
    public InterfaceInspector ( @Nullable final Component inspected )
    {
        this ( StyleId.inspector, inspected );
    }

    /**
     * Constructs new empty {@link InterfaceInspector}.
     *
     * @param id {@link StyleId}
     */
    public InterfaceInspector ( @NotNull final StyleId id )
    {
        this ( id, null );
    }

    /**
     * Constructs new {@link InterfaceInspector} for the specified {@link Component} and its childrens tree.
     *
     * @param id        {@link StyleId}
     * @param inspected {@link Component} to inspect
     */
    public InterfaceInspector ( @NotNull final StyleId id, @Nullable final Component inspected )
    {
        super ( id );

        // Tree scroll
        final WebScrollPane scrollPane = new WebScrollPane ( StyleId.inspectorScroll.at ( InterfaceInspector.this ) );
        scrollPane.setPreferredWidth ( 300 );

        // Interface tree
        tree = new InterfaceTree ( StyleId.inspectorTree.at ( scrollPane ), inspected );
        scrollPane.getViewport ().setView ( tree );

        // Filtering field
        final WebTreeFilterField filter = new WebTreeFilterField ( StyleId.inspectorFilter.at ( InterfaceInspector.this ), tree );

        // Component inspect behavior
        final WebToggleButton inspectToggle = new WebToggleButton ( StyleId.inspectorInspect.at ( filter ), Icons.target );
        inspectToggle.setRolloverIcon ( Icons.targetHover );
        inspectToggle.setSelectedIcon ( Icons.targetSelected );
        inspectToggle.setCursor ( Cursor.getDefaultCursor () );
        inspectToggle.addHotkey ( Hotkey.CTRL_I );
        inspectToggle.addActionListener ( new ActionListener ()
        {
            @Override
            public void actionPerformed ( @NotNull final ActionEvent e )
            {
                if ( inspectToggle.isSelected () )
                {
                    if ( inspectBehavior == null )
                    {
                        inspectBehavior = new ComponentInspectBehavior ();
                    }
                    if ( !inspectBehavior.isInstalled () )
                    {
                        inspectBehavior.install ( inspected, new InspectionListener ()
                        {
                            @Override
                            public void inspected ( @NotNull final Component component )
                            {
                                tree.navigate ( component );
                                inspectToggle.setSelected ( false );
                                tree.requestFocusInWindow ();
                            }

                            @Override
                            public void cancelled ()
                            {
                                inspectToggle.setSelected ( false );
                            }
                        } );
                    }
                }
                else
                {
                    if ( inspectBehavior != null && inspectBehavior.isInstalled () )
                    {
                        inspectBehavior.uninstall ();
                    }
                }
            }
        } );
        filter.setTrailingComponent ( inspectToggle );

        // UI composition
        final WebSeparator separator = new WebSeparator ( StyleId.inspectorSeparator.at ( InterfaceInspector.this ) );
        add ( new GroupPanel ( GroupingType.fillLast, 0, false, filter, separator, scrollPane ) );

        // Expanding tree root by default
        tree.expandRoot ();

        // Visibility behavior
        new VisibilityBehavior ( this, true )
        {
            @Override
            protected void hidden ( @NotNull final InterfaceInspector inspector )
            {
                if ( inspectBehavior != null && inspectBehavior.isInstalled () )
                {
                    inspectBehavior.uninstall ();
                }
            }
        }.install ();
    }

    /**
     * Sets inspected {@link Component}.
     *
     * @param inspected {@link Component} to inspect
     */
    public void setInspected ( @Nullable final Component inspected )
    {
        tree.setRootComponent ( inspected );
    }

    /**
     * Clears highlighted {@link Component}s.
     */
    public void clearHighlights ()
    {
        tree.clearSelection ();
    }

    /**
     * Returns separate {@link WebFrame} with inspector for all visible {@link Component}s.
     * That {@link WebFrame} will be displayed straight away on the screen.
     *
     * @return separate {@link WebFrame} with inspector for all visible {@link Component}s
     */
    @NotNull
    public static WebFrame showFrame ()
    {
        return showFrame ( null );
    }

    /**
     * Returns separate {@link WebFrame} with inspector for the specified {@link Component}.
     * That {@link WebFrame} will be displayed straight away on the screen.
     *
     * @param inspected {@link Component} to inspect
     * @return separate {@link WebFrame} with inspector for the specified {@link Component}
     */
    @NotNull
    public static WebFrame showFrame ( @Nullable final Component inspected )
    {
        final WebFrame frame = new WebFrame ();
        frame.setIconImages ( WebLookAndFeel.getImages () );
        frame.add ( new InterfaceInspector ( inspected ) );
        ProprietaryUtils.setUtilityWindowType ( frame );
        frame.setModalExclusionType ( Dialog.ModalExclusionType.APPLICATION_EXCLUDE );
        frame.pack ();
        frame.setLocationRelativeTo ( inspected );
        frame.setVisible ( true );
        return frame;
    }

    /**
     * Returns separate {@link WebDialog} with inspector for all visible {@link Component}s.
     * That {@link WebDialog} will be displayed straight away on the screen.
     *
     * @param parent parent {@link Component} for {@link WebDialog}
     * @return separate {@link WebDialog} with inspector for all visible {@link Component}s
     */
    @NotNull
    public static WebDialog showDialog ( @Nullable final Component parent )
    {
        return showDialog ( parent, null );
    }

    /**
     * Returns separate {@link WebDialog} with inspector for the specified {@link Component}.
     * That {@link WebDialog} will be displayed straight away on the screen.
     *
     * @param parent    parent {@link Component} for {@link WebDialog}
     * @param inspected {@link Component} to inspect
     * @return separate {@link WebDialog} with inspector for the specified {@link Component}
     */
    @NotNull
    public static WebDialog showDialog ( @Nullable final Component parent, @Nullable final Component inspected )
    {
        final WebDialog dialog = new WebDialog ( parent );
        dialog.setIconImages ( WebLookAndFeel.getImages () );
        dialog.add ( new InterfaceInspector ( inspected ) );
        ProprietaryUtils.setUtilityWindowType ( dialog );
        dialog.setModalExclusionType ( Dialog.ModalExclusionType.APPLICATION_EXCLUDE );
        dialog.setModal ( false );
        dialog.pack ();
        dialog.setLocationRelativeTo ( inspected );
        // dialog.setAttachedTo ( ? );
        dialog.setVisible ( true );
        return dialog;
    }

    /**
     * Returns separate {@link WebPopOver} with inspector for all visible {@link Component}s.
     * That {@link WebPopOver} will be displayed straight away near the parent {@link Component}.
     *
     * @param parent parent {@link Component} for {@link WebPopOver}
     * @return separate {@link WebPopOver} with inspector for all visible {@link Component}s
     */
    @NotNull
    public static WebPopOver showPopOver ( @Nullable final Component parent )
    {
        return showPopOver ( parent, null, PopOverDirection.right );
    }

    /**
     * Returns separate {@link WebPopOver} with inspector for all visible {@link Component}s.
     * That {@link WebPopOver} will be displayed straight away near the parent {@link Component}.
     *
     * @param parent    parent {@link Component} for {@link WebPopOver}
     * @param direction {@link PopOverDirection}
     * @return separate {@link WebPopOver} with inspector for all visible {@link Component}s
     */
    @NotNull
    public static WebPopOver showPopOver ( @Nullable final Component parent, @NotNull final PopOverDirection direction )
    {
        return showPopOver ( parent, null, direction );
    }

    /**
     * Returns separate {@link WebPopOver} with inspector for the specified {@link Component}.
     * That {@link WebPopOver} will be displayed straight away near the parent {@link Component}.
     *
     * @param parent    parent {@link Component} for {@link WebPopOver}
     * @param inspected {@link Component} to inspect
     * @return separate {@link WebPopOver} with inspector for the specified {@link Component}
     */
    @NotNull
    public static WebPopOver showPopOver ( @Nullable final Component parent, @Nullable final Component inspected )
    {
        return showPopOver ( parent, inspected, PopOverDirection.right );
    }

    /**
     * Returns separate {@link WebPopOver} with inspector for the specified {@link Component}.
     * That {@link WebPopOver} will be displayed straight away near the parent {@link Component}.
     *
     * @param parent    parent {@link Component} for {@link WebPopOver}
     * @param inspected {@link Component} to inspect
     * @param direction {@link PopOverDirection}
     * @return separate {@link WebPopOver} with inspector for the specified {@link Component}
     */
    @NotNull
    public static WebPopOver showPopOver ( @Nullable final Component parent, @Nullable final Component inspected,
                                           @NotNull final PopOverDirection direction )
    {
        final WebPopOver popOver = new WebPopOver ( parent );
        popOver.setIconImages ( WebLookAndFeel.getImages () );
        popOver.add ( new InterfaceInspector ( StyleId.inspectorPopover, inspected ) );
        popOver.setModalExclusionType ( Dialog.ModalExclusionType.APPLICATION_EXCLUDE );
        popOver.show ( parent, direction );
        return popOver;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy