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

com.alee.extended.dock.WebDockableFrame 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.dock;

import com.alee.api.Identifiable;
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.WebContainer;
import com.alee.managers.style.StyleId;
import com.alee.managers.style.StyleManager;
import com.alee.painter.decoration.Stateful;
import com.alee.utils.CollectionUtils;

import javax.swing.*;
import java.awt.*;
import java.util.List;

/**
 * Frame element for {@link WebDockablePane}.
 * It can have its own content which can be easily moved within dockable pane or even outside of its bounds.
 * Frame usually take side space within dockable pane and its center area is taken by some custom content.
 *
 * This component should never be used with a non-Web UIs as it might cause an unexpected behavior.
 * You could still use that component even if WebLaF is not your application LaF as this component will use Web-UI in any case.
 *
 * @author Mikle Garin
 * @see How to use WebDockablePane
 * @see DockableFrameDescriptor
 * @see WDockableFrameUI
 * @see WebDockableFrameUI
 * @see IDockableFramePainter
 * @see DockablePanePainter
 * @see WebContainer
 */
public class WebDockableFrame extends WebContainer implements Identifiable, Stateful
{
    /**
     * Component properties.
     */
    public static final String FRAME_ID_PROPERTY = "frameId";
    public static final String STATE_PROPERTY = "state";
    public static final String MAXIMIZED_PROPERTY = "maximized";
    public static final String RESTORE_STATE_PROPERTY = "restoreState";
    public static final String DRAGGABLE_PROPERTY = "draggable";
    public static final String FLOATABLE_PROPERTY = "floatable";
    public static final String MAXIMIZABLE_PROPERTY = "maximizable";
    public static final String CLOSABLE_PROPERTY = "closable";
    public static final String ICON_PROPERTY = "icon";
    public static final String TITLE_PROPERTY = "title";
    public static final String POSITION_PROPERTY = "position";
    public static final String DOCKABLE_PANE_PROPERTY = "dockablePane";

    /**
     * Frame ID unique within its dockable pane.
     * It is used to connect frame with its position data inside dockable pane model.
     */
    @NotNull
    protected String frameId;

    /**
     * Current frame state.
     */
    @NotNull
    protected DockableFrameState state;

    /**
     * Whether or not frame is maximized.
     */
    protected boolean maximized;

    /**
     * State to restore frame into from {@link DockableFrameState#minimized}.
     */
    @NotNull
    protected DockableFrameState restoreState;

    /**
     * Position of this frame on dockable pane.
     * This value will only be used if dockable pane didn't save any position for this frame yet.
     */
    @NotNull
    protected CompassDirection position;

    /**
     * Frame icon.
     */
    @Nullable
    protected Icon icon;

    /**
     * Frame title.
     */
    @Nullable
    protected String title;

    /**
     * Whether or not frame can be dragged.
     */
    protected boolean draggable;

    /**
     * Whether or not frame can be separated into floating window from the UI.
     * You can still float the frame from the code even if this setting is set to {@code false}.
     */
    protected boolean floatable;

    /**
     * Whether or not frame can be maximized.
     * It can only be maximized when it is in {@link DockableFrameState#docked} or
     * {@link DockableFrameState#floating} states.
     */
    protected boolean maximizable;

    /**
     * Whether or not frame can be closed from the UI.
     * You can still close the frame from the code even if this setting is set to {@code false}.
     */
    protected boolean closable;

    /**
     * Dockable pane this frame is added into.
     */
    @Nullable
    protected WebDockablePane dockablePane;

    /**
     * Constructs new {@link WebDockableFrame}.
     *
     * @param frameId unique frame ID
     * @param title   frame title
     */
    public WebDockableFrame ( @NotNull final String frameId, @Nullable final String title )
    {
        this ( StyleId.auto, frameId, null, title );
    }

    /**
     * Constructs new {@link WebDockableFrame}.
     *
     * @param frameId unique frame ID
     * @param icon    frame icon
     */
    public WebDockableFrame ( @NotNull final String frameId, @Nullable final Icon icon )
    {
        this ( StyleId.auto, frameId, icon, null );
    }

    /**
     * Constructs new {@link WebDockableFrame}.
     *
     * @param frameId unique frame ID
     * @param icon    frame icon
     * @param title   frame title
     */
    public WebDockableFrame ( @NotNull final String frameId, @Nullable final Icon icon, @Nullable final String title )
    {
        this ( StyleId.auto, frameId, icon, title );
    }

    /**
     * Constructs new {@link WebDockableFrame}.
     *
     * @param id      {@link StyleId}
     * @param frameId unique frame ID
     * @param title   frame title
     */
    public WebDockableFrame ( @NotNull final StyleId id, @NotNull final String frameId, @Nullable final String title )
    {
        this ( id, frameId, null, title );
    }

    /**
     * Constructs new {@link WebDockableFrame}.
     *
     * @param id      {@link StyleId}
     * @param frameId unique frame ID
     * @param icon    frame icon
     */
    public WebDockableFrame ( @NotNull final StyleId id, @NotNull final String frameId, @Nullable final Icon icon )
    {
        this ( id, frameId, icon, null );
    }

    /**
     * Constructs new {@link WebDockableFrame}.
     *
     * @param id      {@link StyleId}
     * @param frameId unique frame ID
     * @param icon    frame icon
     * @param title   frame title
     */
    public WebDockableFrame ( @NotNull final StyleId id, @NotNull final String frameId, @Nullable final Icon icon,
                              @Nullable final String title )
    {
        this.frameId = frameId;
        this.icon = icon;
        this.title = title;
        this.state = DockableFrameState.docked;
        this.maximized = false;
        this.restoreState = DockableFrameState.docked;
        this.position = CompassDirection.west;
        this.draggable = true;
        this.closable = true;
        this.floatable = true;
        this.maximizable = true;
        setFocusCycleRoot ( true );
        updateUI ();
        setStyleId ( id );
    }

    @NotNull
    @Override
    public StyleId getDefaultStyleId ()
    {
        return StyleId.dockableframe;
    }

    @NotNull
    @Override
    public String getId ()
    {
        return frameId;
    }

    /**
     * Sets frame ID unique within its dockable pane.
     *
     * @param id frame ID unique within its dockable pane
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setId ( @NotNull final String id )
    {
        if ( Objects.notEquals ( this.frameId, id ) )
        {
            final String old = this.frameId;
            this.frameId = id;
            firePropertyChange ( FRAME_ID_PROPERTY, old, id );
        }
        return this;
    }

    @Nullable
    @Override
    public List getStates ()
    {
        return CollectionUtils.asList ( getState ().name () );
    }

    /**
     * Returns current frame state.
     *
     * @return current frame state
     */
    @NotNull
    public DockableFrameState getState ()
    {
        return state;
    }

    /**
     * Sets frame state.
     *
     * @param state frame state
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setState ( @NotNull final DockableFrameState state )
    {
        if ( this.state != state )
        {
            final DockableFrameState old = this.state;
            this.state = state;
            firePropertyChange ( STATE_PROPERTY, old, state );
        }
        return this;
    }

    /**
     * Returns whether or not frame is maximized.
     *
     * @return true if frame is maximized, false otherwise
     */
    public boolean isMaximized ()
    {
        return maximized;
    }

    /**
     * Sets whether or not frame is maximized.
     *
     * @param maximized whether or not frame is maximized
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setMaximized ( final boolean maximized )
    {
        if ( Objects.notEquals ( this.maximized, maximized ) )
        {
            // todo Restore frame if maximized on "false"?
            final boolean old = this.maximized;
            this.maximized = maximized;
            firePropertyChange ( MAXIMIZED_PROPERTY, old, maximized );
        }
        return this;
    }

    /**
     * Returns state to restore frame into from {@link DockableFrameState#minimized}.
     *
     * @return state to restore frame into from {@link DockableFrameState#minimized}
     */
    @NotNull
    public DockableFrameState getRestoreState ()
    {
        return restoreState;
    }

    /**
     * Sets state to restore frame into from {@link DockableFrameState#minimized}.
     *
     * @param state state to restore frame into from {@link DockableFrameState#minimized}
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setRestoreState ( @NotNull final DockableFrameState state )
    {
        if ( this.restoreState != state )
        {
            final DockableFrameState old = this.restoreState;
            this.restoreState = state;
            firePropertyChange ( RESTORE_STATE_PROPERTY, old, state );
        }
        return this;
    }

    /**
     * Returns position of this frame on dockable pane.
     *
     * @return position of this frame on dockable pane
     */
    @NotNull
    public CompassDirection getPosition ()
    {
        return position;
    }

    /**
     * Sets position of this frame on dockable pane.
     *
     * @param position position of this frame on dockable pane
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setPosition ( @NotNull final CompassDirection position )
    {
        if ( this.position != position )
        {
            final CompassDirection old = this.position;
            this.position = position;
            firePropertyChange ( POSITION_PROPERTY, old, position );
        }
        return this;
    }

    /**
     * Returns frame icon.
     *
     * @return frame icon
     */
    @Nullable
    public Icon getIcon ()
    {
        return icon;
    }

    /**
     * Sets frame icon.
     *
     * @param icon frame icon
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setIcon ( @Nullable final Icon icon )
    {
        if ( this.icon != icon )
        {
            final Icon old = this.icon;
            this.icon = icon;
            firePropertyChange ( ICON_PROPERTY, old, icon );
        }
        return this;
    }

    /**
     * Returns frame title.
     *
     * @return frame title
     */
    @Nullable
    public String getTitle ()
    {
        return title;
    }

    /**
     * Sets frame title.
     *
     * @param title frame title
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setTitle ( @Nullable final String title )
    {
        if ( Objects.notEquals ( this.title, title ) )
        {
            final String old = this.title;
            this.title = title;
            firePropertyChange ( TITLE_PROPERTY, old, title );
        }
        return this;
    }

    /**
     * Returns whether or not frame can be dragged.
     *
     * @return true if frame can be dragged, false otherwise
     */
    public boolean isDraggable ()
    {
        return draggable;
    }

    /**
     * Sets whether or not frame can be dragged.
     *
     * @param draggable whether or not frame can be dragged
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setDraggable ( final boolean draggable )
    {
        if ( this.draggable != draggable )
        {
            final boolean old = this.draggable;
            this.draggable = draggable;
            firePropertyChange ( DRAGGABLE_PROPERTY, old, draggable );
        }
        return this;
    }

    /**
     * Returns whether or not frame can be separated into floating window from the UI.
     *
     * @return true if frame can be separated into floating window from the UI, false otherwise
     */
    public boolean isFloatable ()
    {
        return floatable;
    }

    /**
     * Sets whether or not frame can be separated into floating window from the UI.
     *
     * @param floatable whether or not frame can be separated into floating window from the UI
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setFloatable ( final boolean floatable )
    {
        if ( this.floatable != floatable )
        {
            final boolean old = this.floatable;
            this.floatable = floatable;
            firePropertyChange ( FLOATABLE_PROPERTY, old, floatable );
        }
        return this;
    }

    /**
     * Returns whether or not frame can be maximized.
     *
     * @return true if frame can be maximized, false otherwise
     */
    public boolean isMaximizable ()
    {
        return maximizable;
    }

    /**
     * Sets whether or not frame can be maximized.
     *
     * @param maximizable whether or not frame can be maximized
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setMaximizable ( final boolean maximizable )
    {
        if ( this.maximizable != maximizable )
        {
            final boolean old = this.maximizable;
            this.maximizable = maximizable;
            firePropertyChange ( MAXIMIZABLE_PROPERTY, old, maximizable );
        }
        return this;
    }

    /**
     * Returns whether or not frame can be closed from the UI.
     *
     * @return true if frame can be closed from the UI, false otherwise
     */
    public boolean isClosable ()
    {
        return closable;
    }

    /**
     * Sets whether or not frame can be closed from the UI.
     *
     * @param closable whether or not frame can be closed from the UI
     * @return this frame
     */
    @NotNull
    public WebDockableFrame setClosable ( final boolean closable )
    {
        if ( this.closable != closable )
        {
            final boolean old = this.closable;
            this.closable = closable;
            firePropertyChange ( CLOSABLE_PROPERTY, old, closable );
        }
        return this;
    }

    /**
     * Returns dockable pane this frame is added into.
     *
     * @return dockable pane this frame is added into
     */
    @Nullable
    public WebDockablePane getDockablePane ()
    {
        return dockablePane;
    }

    /**
     * Sets dockable pane this frame is added into.
     *
     * @param dockablePane dockable pane this frame is added into
     * @return this frame
     */
    @NotNull
    protected WebDockableFrame setDockablePane ( @Nullable final WebDockablePane dockablePane )
    {
        if ( this.dockablePane != dockablePane )
        {
            final WebDockablePane old = this.dockablePane;
            this.dockablePane = dockablePane;
            firePropertyChange ( DOCKABLE_PANE_PROPERTY, old, dockablePane );
        }
        return this;
    }

    /**
     * Returns {@link SidebarButton} for this {@link WebDockableFrame}.
     *
     * @return {@link SidebarButton} for this {@link WebDockableFrame}
     */
    @NotNull
    public SidebarButton getSidebarButton ()
    {
        return getUI ().getSidebarButton ();
    }

    /**
     * Returns whether or not sidebar button is currently visible.
     *
     * @return true if sidebar button is currently visible, false otherwise
     */
    public boolean isSidebarButtonVisible ()
    {
        boolean visible = false;
        final WebDockablePane dockablePane = getDockablePane ();
        if ( dockablePane != null )
        {
            switch ( dockablePane.getSidebarButtonVisibility () )
            {
                case never:
                    visible = false;
                    break;

                case minimized:
                    visible = isMinimized () || isPreview ();
                    break;

                case anyMinimized:
                    if ( !isClosed () )
                    {
                        for ( final WebDockableFrame frame : dockablePane.getFrames ( getPosition () ) )
                        {
                            if ( frame.isMinimized () || frame.isPreview () )
                            {
                                visible = true;
                                break;
                            }
                        }
                    }
                    break;

                case always:
                    visible = isOpened ();
                    break;
            }
        }
        return visible;
    }

    /**
     * Returns whether or not frame is closed.
     *
     * @return true if frame is closed, false otherwise
     */
    public boolean isClosed ()
    {
        return state == DockableFrameState.closed;
    }

    /**
     * Returns whether or not frame is opened.
     *
     * @return true if frame is opened, false otherwise
     */
    public boolean isOpened ()
    {
        return state != DockableFrameState.closed;
    }

    /**
     * Returns whether or not frame is minimized.
     *
     * @return true if frame is minimized, false otherwise
     */
    public boolean isMinimized ()
    {
        return state == DockableFrameState.minimized;
    }

    /**
     * Returns whether or not frame is in preview state.
     *
     * @return true if frame is in preview state, false otherwise
     */
    public boolean isPreview ()
    {
        return state == DockableFrameState.preview;
    }

    /**
     * Returns whether or not frame is docked.
     *
     * @return true if frame is docked, false otherwise
     */
    public boolean isDocked ()
    {
        return state == DockableFrameState.docked;
    }

    /**
     * Returns whether or not frame is floating.
     *
     * @return true if frame is floating, false otherwise
     */
    public boolean isFloating ()
    {
        return state == DockableFrameState.floating;
    }

    /**
     * Hides this dockable frame, only its sidebar button will be visible.
     * If {@link #maximized} was {@code true} it will be switched to {@code false}.
     *
     * @return this frame
     */
    @NotNull
    public WebDockableFrame minimize ()
    {
        return setState ( DockableFrameState.minimized );
    }

    /**
     * Docks this dockable frame to its last position on the pane.
     *
     * @return this frame
     */
    @NotNull
    public WebDockableFrame dock ()
    {
        return setState ( DockableFrameState.docked );
    }

    /**
     * Displays frame preview on top of other pane elements, sidebar button will still be kept visible.
     *
     * @return this frame
     */
    @NotNull
    public WebDockableFrame preview ()
    {
        return setState ( DockableFrameState.preview );
    }

    /**
     * Displays frame on a separate non-modal dialog window attached to dockable frame window.
     *
     * @return this frame
     */
    @NotNull
    public WebDockableFrame detach ()
    {
        return setState ( DockableFrameState.floating );
    }

    /**
     * Maximizes dockable frame.
     * Frame will be made
     */
    public void maximize ()
    {
        if ( !isMaximized () )
        {
            // Restoring frame state if needed
            if ( !isDocked () && !isFloating () )
            {
                restore ();
            }

            // Maximize frame
            setMaximized ( true );
        }
        else
        {
            // Restore frame size
            setMaximized ( false );
        }
    }

    /**
     * Restores either docked or floating state of this dockable frame.
     *
     * @return this frame
     */
    @NotNull
    public WebDockableFrame restore ()
    {
        return setState ( getRestoreState () );
    }

    /**
     * Closes this dockable frame.
     *
     * @return this frame
     */
    @NotNull
    public WebDockableFrame close ()
    {
        return setState ( DockableFrameState.closed );
    }

    /**
     * Adds new {@link DockableFrameListener}.
     *
     * @param listener {@link DockableFrameListener} to add
     */
    public void addFrameListener ( @NotNull final DockableFrameListener listener )
    {
        listenerList.add ( DockableFrameListener.class, listener );
    }

    /**
     * Removes specified {@link DockableFrameListener}.
     *
     * @param listener {@link DockableFrameListener} to remove
     */
    public void removeFrameListener ( @NotNull final DockableFrameListener listener )
    {
        listenerList.remove ( DockableFrameListener.class, listener );
    }

    /**
     * Informs listeners about frame being added.
     */
    public void fireFrameAdded ()
    {
        final WebDockablePane dockablePane = getDockablePane ();
        if ( dockablePane != null )
        {
            for ( final DockableFrameListener listener : listenerList.getListeners ( DockableFrameListener.class ) )
            {
                listener.frameAdded ( this, dockablePane );
            }
        }
    }

    /**
     * Informs listeners about frame state change.
     *
     * @param oldState previous frame state
     * @param newState current frame state
     */
    public void fireFrameStateChanged ( @NotNull final DockableFrameState oldState, @NotNull final DockableFrameState newState )
    {
        for ( final DockableFrameListener listener : listenerList.getListeners ( DockableFrameListener.class ) )
        {
            listener.frameStateChanged ( this, oldState, newState );
        }
    }

    /**
     * Informs listeners about frame being moved.
     *
     * @param position current frame position relative to content
     */
    public void fireFrameMoved ( @NotNull final CompassDirection position )
    {
        for ( final DockableFrameListener listener : listenerList.getListeners ( DockableFrameListener.class ) )
        {
            listener.frameMoved ( this, position );
        }
    }

    /**
     * Informs listeners about frame being removed.
     */
    public void fireFrameRemoved ()
    {
        final WebDockablePane dockablePane = getDockablePane ();
        if ( dockablePane != null )
        {
            for ( final DockableFrameListener listener : listenerList.getListeners ( DockableFrameListener.class ) )
            {
                listener.frameRemoved ( this, dockablePane );
            }
        }
    }

    @Override
    public void applyComponentOrientation ( final ComponentOrientation orientation )
    {
        super.applyComponentOrientation ( orientation );
        getSidebarButton ().applyComponentOrientation ( orientation );
    }

    /**
     * Returns the look and feel (LaF) object that renders this component.
     *
     * @return the {@link WDockableFrameUI} object that renders this component
     */
    public WDockableFrameUI getUI ()
    {
        return ( WDockableFrameUI ) ui;
    }

    /**
     * Sets the LaF object that renders this component.
     *
     * @param ui {@link WDockableFrameUI}
     */
    public void setUI ( final WDockableFrameUI ui )
    {
        super.setUI ( ui );
    }

    @Override
    public void updateUI ()
    {
        StyleManager.getDescriptor ( this ).updateUI ( this );
    }

    @NotNull
    @Override
    public String getUIClassID ()
    {
        return StyleManager.getDescriptor ( this ).getUIClassId ();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy