org.uberfire.client.workbench.widgets.split.WorkbenchSplitLayoutPanel.java Source code

Java tutorial

Introduction

Here is the source code for org.uberfire.client.workbench.widgets.split.WorkbenchSplitLayoutPanel.java

Source

/*
 * Copyright 2015 Red Hat, Inc. and/or its affiliates.
 *
 * 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 org.uberfire.client.workbench.widgets.split;

import com.google.gwt.core.client.Scheduler;
import com.google.gwt.core.client.Scheduler.ScheduledCommand;
import com.google.gwt.dom.client.Document;
import com.google.gwt.dom.client.Element;
import com.google.gwt.dom.client.Style;
import com.google.gwt.dom.client.Style.Position;
import com.google.gwt.dom.client.Style.Unit;
import com.google.gwt.user.client.Command;
import com.google.gwt.user.client.Event;
import com.google.gwt.user.client.Window;
import com.google.gwt.user.client.ui.Composite;
import com.google.gwt.user.client.ui.DockLayoutPanel;
import com.google.gwt.user.client.ui.FlowPanel;
import com.google.gwt.user.client.ui.RequiresResize;
import com.google.gwt.user.client.ui.SplitLayoutPanel;
import com.google.gwt.user.client.ui.Widget;
import org.uberfire.client.resources.WorkbenchResources;
import org.uberfire.client.workbench.widgets.listbar.ResizeFlowPanel;
import org.uberfire.workbench.model.CompassPosition;

/**
 * A panel that adds user-positioned splitters between each of its child
 * widgets.
 * <p>
 * This is a fork of the standard SplitLayoutPanel with the following changes:-
 * <ol>
 * <li>http://code.google.com/p/google-web-toolkit/issues/detail?id=7135</li>
 * </ol>
 * </p>
 * <p>
 * This panel is used in the same way as {@link DockLayoutPanel}, except that
 * its children's sizes are always specified in {@link Unit#PX} units, and each
 * pair of child widgets has a splitter between them that the user can drag.
 * </p>
 * <p>
 * This widget will <em>only</em> work in standards mode, which requires that
 * the HTML page in which it is run have an explicit &lt;!DOCTYPE&gt;
 * declaration.
 * </p>
 * <h3>CSS Style Rules</h3>
 * <ul class='css'>
 * <li>.gwt-SplitLayoutPanel { the panel itself }</li>
 * <li>.gwt-SplitLayoutPanel .gwt-SplitLayoutPanel-HDragger { horizontal dragger
 * }</li>
 * <li>.gwt-SplitLayoutPanel .gwt-SplitLayoutPanel-VDragger { vertical dragger }
 * </li>
 * </ul>
 * <p>
 * <h3>Example</h3> {@example com.google.gwt.examples.SplitLayoutPanelExample}
 * </p>
 */
public class WorkbenchSplitLayoutPanel extends DockLayoutPanel {

    private static final int DEFAULT_SPLITTER_SIZE = 1;
    private static final int DEFAULT_SPLITTER_HOVER_SIZE = 10;
    /**
     * The element that masks the screen so we can catch mouse events over
     * iframes.
     */
    private static Element glassElem = null;
    private final int splitterSize;
    private int minCenterSize = 0;

    /**
     * Construct a new {@link SplitLayoutPanel} with the default splitter size
     * of 8px.
     */
    public WorkbenchSplitLayoutPanel() {
        this(DEFAULT_SPLITTER_SIZE);
    }

    /**
     * Construct a new {@link SplitLayoutPanel} with the specified splitter size
     * in pixels.
     * @param splitterSize the size of the splitter in pixels
     */
    public WorkbenchSplitLayoutPanel(int splitterSize) {
        super(Unit.PX);
        this.splitterSize = splitterSize;
        setStyleName(WorkbenchResources.INSTANCE.CSS().splitLayoutPanel());

        if (glassElem == null) {
            glassElem = Document.get().createDivElement();
            glassElem.getStyle().setPosition(Position.ABSOLUTE);
            glassElem.getStyle().setTop(0, Unit.PX);
            glassElem.getStyle().setLeft(0, Unit.PX);
            glassElem.getStyle().setMargin(0, Unit.PX);
            glassElem.getStyle().setPadding(0, Unit.PX);
            glassElem.getStyle().setBorderWidth(0, Unit.PX);

            // We need to set the background color or mouse events will go right
            // through the glassElem. If the SplitPanel contains an iframe, the
            // iframe will capture the event and the slider will stop moving.
            glassElem.getStyle().setProperty("background", "white");
            glassElem.getStyle().setOpacity(0.0);
        }
    }

    /**
     * Return the size of the splitter in pixels.
     * @return the splitter size
     */
    public int getSplitterSize() {
        return splitterSize;
    }

    @Override
    public void insert(Widget child, Direction direction, double size, Widget before) {
        super.insert(child, direction, size, before);
        if (direction != Direction.CENTER) {
            insertSplitter(child, before);
        }
    }

    @Override
    public boolean remove(Widget child) {
        assert !(child instanceof Splitter) : "Splitters may not be directly removed";

        int idx = getWidgetIndex(child);
        if (super.remove(child)) {
            // Remove the associated splitter, if any.
            // Now that the widget is removed, idx is the index of the splitter.
            if (idx < getWidgetCount()) {
                // Call super.remove(), or we'll end up recursing.
                super.remove(getWidget(idx));
            }
            return true;
        }
        return false;
    }

    /**
     * Sets the minimum allowable size for the given widget.
     * <p>
     * Its associated splitter cannot be dragged to a position that would make
     * it smaller than this size. This method has no effect for the
     * {@link DockLayoutPanel.Direction#CENTER} widget.
     * </p>
     * @param child the child whose minimum size will be set
     * @param minSize the minimum size for this widget
     */
    public void setWidgetMinSize(Widget child, int minSize) {
        assertIsAChild(child);
        Splitter splitter = getAssociatedSplitter(child);
        // The splitter is null for the center element.
        if (splitter != null) {
            splitter.setMinSize(minSize);
        } else {
            minCenterSize = minSize;
        }
    }

    private void assertIsAChild(Widget widget) {
        assert (widget == null)
                || (widget.getParent() == this) : "The specified widget is not a child of this panel";
    }

    private Splitter getAssociatedSplitter(Widget child) {
        // If a widget has a next sibling, it must be a splitter, because the only
        // widget that *isn't* followed by a splitter must be the CENTER, which has
        // no associated splitter.
        int idx = getWidgetIndex(child);
        if (idx > -1 && idx < getWidgetCount() - 1) {
            Widget splitter = getWidget(idx + 1);
            assert splitter instanceof Splitter : "Expected child widget to be splitter";
            return (Splitter) splitter;
        }
        return null;
    }

    private void insertSplitter(Widget widget, Widget before) {
        assert getChildren().size() > 0 : "Can't add a splitter before any children";

        LayoutData layout = (LayoutData) widget.getLayoutData();
        Splitter splitter = null;
        switch (getResolvedDirection(layout.direction)) {
        case WEST:
            splitter = new HSplitter(widget, false);
            break;
        case EAST:
            splitter = new HSplitter(widget, true);
            break;
        case NORTH:
            splitter = new VSplitter(widget, false);
            break;
        case SOUTH:
            splitter = new VSplitter(widget, true);
            break;
        default:
            assert false : "Unexpected direction";
        }

        super.insert(splitter, layout.direction, splitterSize, before);
    }

    /**
     * Adds the given widget as a child of this splitter.
     * @param child the widget to add
     * @param position the position to dock the widget at (must be an actual compass position NORTH, SOUTH, EAST, or WEST)
     * @param size the width or height to give the added child.
     */
    public void add(Widget child, CompassPosition position, double size) {
        switch (position) {
        case NORTH:
            addNorth(child, size);
            break;
        case SOUTH:
            addSouth(child, size);
            break;
        case EAST:
            addEast(child, size);
            break;
        case WEST:
            addWest(child, size);
            break;
        default:
            throw new IllegalArgumentException("Bad child position: " + position);
        }
    }

    /**
     * Adds the given widget as a child of this splitter.
     * @param child the widget to add
     * @param position the position to dock the widget at (must be an actual compass position NORTH, SOUTH, EAST, or WEST)
     * @param size the width or height to give the added child
     */
    public void add(Widget child, CompassPosition position, int size) {
        double doubleSize = (double) size;
        add(child, position, doubleSize);
    }

    class HSplitter extends Splitter {

        public HSplitter(Widget target, boolean reverse) {
            super(target, reverse);
            setStyleName(WorkbenchResources.INSTANCE.CSS().splitLayoutPanelHDragger());
        }

        @Override
        protected void setUpHoverStyle(final Style style, int size) {
            style.setWidth(size, Unit.PX);
            style.setMarginLeft(-(size / 2), Unit.PX);
        }

        @Override
        public void onResize() {
            hover.getElement().getStyle().setHeight(target.getOffsetHeight(), Unit.PX);
        }

        @Override
        protected Style.Cursor getHoverCursorStyle() {
            return Style.Cursor.COL_RESIZE;
        }

        @Override
        protected int getAbsolutePosition() {
            return getAbsoluteLeft();
        }

        @Override
        protected double getCenterSize() {
            return getCenterWidth();
        }

        @Override
        protected int getEventPosition(Event event) {
            return event.getClientX();
        }

        @Override
        protected int getTargetPosition() {
            return target.getAbsoluteLeft();
        }

        @Override
        protected int getTargetSize() {
            return target.getOffsetWidth();
        }
    }

    abstract class Splitter extends Composite implements RequiresResize {

        protected final Widget target;
        protected final Widget hover;
        protected final Element mouseTracker;
        private final boolean reverse;
        private int offset;
        private boolean mouseDown;
        private ScheduledCommand layoutCommand;
        private int minSize;

        private double centerSize, syncedCenterSize;

        public Splitter(Widget target, boolean reverse) {
            this.target = target;
            this.reverse = reverse;

            final ResizeFlowPanel widget = new ResizeFlowPanel();

            sinkEvents(Event.ONMOUSEDOWN | Event.ONMOUSEUP | Event.ONMOUSEMOVE | Event.ONDBLCLICK);

            this.hover = new FlowPanel();
            final Style style = hover.getElement().getStyle();
            style.setOpacity(0);
            style.setZIndex(2);
            style.setPosition(Position.FIXED);
            setUpHoverStyle(style, DEFAULT_SPLITTER_HOVER_SIZE);
            widget.add(hover);
            initWidget(widget);

            mouseTracker = Document.get().createDivElement();
            mouseTracker.getStyle().setCursor(getHoverCursorStyle());
            mouseTracker.getStyle().setZIndex(Integer.MAX_VALUE);
            mouseTracker.getStyle().setPosition(Position.ABSOLUTE);
            mouseTracker.getStyle().setHeight(50, Unit.PX);
            mouseTracker.getStyle().setWidth(50, Unit.PX);
        }

        @Override
        public void onBrowserEvent(final Event event) {
            switch (event.getTypeInt()) {
            case Event.ONMOUSEDOWN:
                mouseDown = true;
                /*
                 * Resize glassElem to take up the entire scrollable window
                 * area, which is the greater of the scroll size and the
                 * client size.
                 */
                int width = Math.max(Window.getClientWidth(), Document.get().getScrollWidth());
                int height = Math.max(Window.getClientHeight(), Document.get().getScrollHeight());
                glassElem.getStyle().setHeight(height, Unit.PX);
                glassElem.getStyle().setWidth(width, Unit.PX);
                Document.get().getBody().appendChild(glassElem);

                offset = getEventPosition(event) - getAbsolutePosition();
                Event.setCapture(getElement());
                event.preventDefault();

                Document.get().getBody().appendChild(mouseTracker);
                break;

            case Event.ONMOUSEUP:
                mouseDown = false;

                glassElem.removeFromParent();

                Event.releaseCapture(getElement());
                event.preventDefault();

                mouseTracker.removeFromParent();
                break;

            case Event.ONMOUSEMOVE:
                if (mouseDown) {
                    int size;
                    if (reverse) {
                        size = getTargetPosition() + getTargetSize() - getEventPosition(event) - offset;
                    } else {
                        size = getEventPosition(event) - getTargetPosition() - offset;
                    }
                    setAssociatedWidgetSize(size);
                    event.preventDefault();

                    mouseTracker.getStyle().setLeft(event.getClientX() - mouseTracker.getOffsetWidth() / 2,
                            Unit.PX);
                    mouseTracker.getStyle().setTop(event.getClientY() - mouseTracker.getOffsetHeight() / 2,
                            Unit.PX);
                }
                break;
            }
        }

        public void setMinSize(int minSize) {
            this.minSize = minSize;
            LayoutData layout = (LayoutData) target.getLayoutData();

            // Try resetting the associated widget's size, which will enforce the new
            // minSize value.
            setAssociatedWidgetSize((int) layout.size);
        }

        protected abstract void setUpHoverStyle(Style style, int size);

        protected abstract Style.Cursor getHoverCursorStyle();

        protected abstract int getAbsolutePosition();

        protected abstract double getCenterSize();

        protected abstract int getEventPosition(Event event);

        protected abstract int getTargetPosition();

        protected abstract int getTargetSize();

        private double getMaxSize() {
            // To avoid seeing stale center size values due to deferred layout
            // updates, maintain our own copy up to date and resync when the
            // DockLayoutPanel value changes.
            double newCenterSize = getCenterSize();
            if (syncedCenterSize != newCenterSize) {
                syncedCenterSize = newCenterSize;
                centerSize = newCenterSize;
            }

            return Math.max(((LayoutData) target.getLayoutData()).size + centerSize, 0);
        }

        private void setAssociatedWidgetSize(double size) {
            double maxSize = getMaxSize();
            if (size > maxSize) {
                size = maxSize;
            }

            if (size < minSize) {
                size = minSize;
            }

            LayoutData layout = (LayoutData) target.getLayoutData();
            if (size == layout.size) {
                return;
            }

            double newCenterSize = centerSize + (layout.size - size);
            if (newCenterSize < minCenterSize) {
                return;
            }

            // Adjust our view until the deferred layout gets scheduled.
            centerSize += layout.size - size;
            centerSize = newCenterSize;
            layout.size = size;

            // Defer actually updating the layout, so that if we receive many
            // mouse events before layout/paint occurs, we'll only update once.
            if (layoutCommand == null) {
                layoutCommand = new Command() {
                    @Override
                    public void execute() {
                        layoutCommand = null;
                        forceLayout();
                    }
                };
                Scheduler.get().scheduleDeferred(layoutCommand);
            }
        }
    }

    class VSplitter extends Splitter {

        public VSplitter(Widget target, boolean reverse) {
            super(target, reverse);
            setStyleName(WorkbenchResources.INSTANCE.CSS().splitLayoutPanelVDragger());
        }

        @Override
        protected void setUpHoverStyle(final Style style, int size) {
            style.setHeight(size, Unit.PX);
            style.setMarginTop(-(size / 2), Unit.PX);
        }

        @Override
        public void onResize() {
            hover.getElement().getStyle().setWidth(target.getOffsetWidth(), Unit.PX);
        }

        @Override
        protected Style.Cursor getHoverCursorStyle() {
            return Style.Cursor.ROW_RESIZE;
        }

        @Override
        protected int getAbsolutePosition() {
            return getAbsoluteTop();
        }

        @Override
        protected double getCenterSize() {
            return getCenterHeight();
        }

        @Override
        protected int getEventPosition(Event event) {
            return event.getClientY();
        }

        @Override
        protected int getTargetPosition() {
            return target.getAbsoluteTop();
        }

        @Override
        protected int getTargetSize() {
            return target.getOffsetHeight();
        }
    }
}