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

com.emitrom.touch4j.client.core.ComplexContainer Maven / Gradle / Ivy

/************************************************************************
  ComplexContainer.java is part of Touch4j 4.2.2.1  Copyright 2013 Emitrom LLC

  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.emitrom.touch4j.client.core;

import java.util.Iterator;
import java.util.List;

import com.emitrom.touch4j.client.ui.Container;
import com.google.gwt.core.client.JavaScriptObject;
import com.google.gwt.core.client.JsArray;
import com.google.gwt.user.client.DOM;
import com.google.gwt.user.client.Element;
import com.google.gwt.user.client.ui.AttachDetachException;
import com.google.gwt.user.client.ui.ComplexPanel;
import com.google.gwt.user.client.ui.HasWidgets;
import com.google.gwt.user.client.ui.IsWidget;
import com.google.gwt.user.client.ui.Widget;
import com.google.gwt.user.client.ui.WidgetCollection;

/**
 * A class used to handle the logical and physical basic operations on
 * containers as well as to provide a basic iterator implementation. The class
 * attempts to mimic the behavior from GWT @See {@link ComplexPanel}. In fact,
 * many of the methods here are borrowed from it. It would have been better to
 * simply have @See {@link Container} extend it, however due to the current
 * hierarchy's restrictions we've decided to go this route.
 */
public abstract class ComplexContainer extends Component implements HasWidgets, HasItems {

    private WidgetCollection children = new WidgetCollection(this);
    private AttachDetachException.Command orphanCommand;

    public ComplexContainer() {
    }

    public ComplexContainer(JavaScriptObject config) {
        super(config);
    }

    public Widget getWidget(int index) {
        return getChildren().get(index);
    }

    public int getWidgetCount() {
        return getChildren().size();
    }

    public int getWidgetIndex(Widget child) {
        return getChildren().indexOf(child);
    }

    public int getWidgetIndex(IsWidget child) {
        return getWidgetIndex(asWidgetOrNull(child));
    }

    @Override
    public Iterator iterator() {
        return getChildren().iterator();
    }

    /**
     * Gets the list of children contained in this panel.
     * 
     * @return a collection of child widgets
     */
    public WidgetCollection getChildren() {
        return children;
    }

    /**
     * Add a widget to the Container.
     * 
     * @param widget
     *            the widget to add
     */
    public void add(IsComponent widget) {
        add(widget.asComponent());
    }

    /**
     * Add a widget to the Container.
     * 
     * @param widget
     *            the widget to add
     */
    @Override
    public void add(Widget widget) {

        // Physical Dettach
        widget.removeFromParent();

        if (widget instanceof Component) {

            // Logical attach.
            getChildren().add(widget);

            // Physical Attach
            add((Component) widget);

            // Adopt widget
            // adopt(widget);

        } else {

            // We must be in the presence of GWT Widgets.
            String id = DOMUtil.getID(widget);
            if (id == null) {
                id = Ext.generateId();
                DOMUtil.setID(widget, id);
            }
            JavaScriptObject compJS = getComponentJS(id);
            WidgetComponent component = null;
            if (compJS != null) {
                component = new WidgetComponent(compJS);
                component.setVisible(true);
            } else {
                component = new WidgetComponent(widget);
            }

            // Logical attach.
            getChildren().add(widget);

            // Physical Attach
            add(component);

            // Adopt widget
            // adopt(widget);

        }

    }

    public void add(IsWidget widget) {
        add(widget.asWidget());
    }

    public boolean remove(int index) {
        return remove(getWidget(index));
    }

    public void remove(IsWidget widget) {
        remove(widget.asWidget());
    }

    @Override
    public boolean remove(Widget w) {

        // Validate.
        if (w.getParent() != this) {
            return false;
        }
        // Orphan.
        try {
            orphan(w);
        } finally {
            // Physical detach.
            remove((Component) w, true);
            // Logical detach.
            getChildren().remove(w);
        }

        return true;

    }

    @Override
    public void removeAll() {
        Iterator iterator = getChildren().iterator();
        while (iterator.hasNext()) {
            remove(iterator.next());
        }
    }

    @Override
    public void clear() {
        removeAll();
    }

    void doLogicalClear() {
        if (orphanCommand == null) {
            orphanCommand = new AttachDetachException.Command() {
                public void execute(Widget w) {
                    orphan(w);
                }
            };
        }
        try {
            AttachDetachException.tryCommand(this, orphanCommand);
        } finally {
            children = new WidgetCollection(this);
        }
    }

    /**
     * Examines this container's items property and gets a direct child
     * component of this container.
     * 
     * String : representing the itemId or id of the child component
     */
    @Override
    public native Component getComponent(String id) /*-{
		var container = [email protected]::getOrCreateJsObj()();
		return container.getComponent(id);
    }-*/;

    /**
     * Examines this container's items property and gets a direct child
     * component of this container.
     * 
     * index : representing the position of the child component within the items
     * property
     */
    @Override
    public native Component getComponent(int index) /*-{
		var container = [email protected]::getOrCreateJsObj()();
		return container.getComponent(index);
    }-*/;

    @Override
    public native Component[] getComponents()/*-{
		var container = [email protected]::getOrCreateJsObj()();
		var items = container.items;
		if (items === undefined || items == null) {
			items = null;
		} else {
			//FormPanel doesn't use MixedCollection
			items = container.items.items || container.items;
		}
		return @com.emitrom.touch4j.client.core.TouchJsoHelper::convertToJavaComponentArray(Lcom/google/gwt/core/client/JavaScriptObject;)(items);
    }-*/;

    @Override
    public void setActiveItem(int activeItem) {
        setActiveItemRendered(activeItem);
    }

    public void setActiveItem(Container container) {
        setActiveItemRendered(container.getOrCreateJsObj());
    }

    public native  T getActiveItem() /*-{
		var container = [email protected]::getOrCreateJsObj()();
		return container.getActiveItem();
    }-*/;

    public void setItems(List components) {
        JsArray values = JsArray.createArray().cast();
        for (Component c : components) {
            values.push(c.getJsObj());
        }
        setItems(values);
    }

    public void setItems(Component... components) {
        JsArray values = JsArray.createArray().cast();
        for (Component c : components) {
            values.push(c.getJsObj());
        }
        setItems(values);
    }

    @Override
    public native void setItems(JsArray items) /*-{
		var container = [email protected]::getOrCreateJsObj()();
		container.setItems(items);
    }-*/;

    /**
     * Returns the value of items.
     */
    @Override
    public native  T getItems() /*-{
		var container = [email protected]::getOrCreateJsObj()();
		return container.getItems();
    }-*/;

    /**
     * Finalize the attachment of a Widget to this Panel. This method is the
     * last step in adding or inserting a Widget into a Panel, and should
     * be called after physical attachment in the DOM is complete. This Panel
     * becomes the parent of the child Widget, and the child will now fire its
     * {@link Widget#onAttach()} event if this Panel is currently attached.
     * 
     * @param child
     *            the widget to be adopted
     * @see #add(Widget)
     */
    protected final void adopt(Widget child) {
        assert (child.getParent() == null);
        adopt(child, this);
    }

    /**
     * 

* This method must be called as part of the remove method of any Panel. It * ensures that the Widget's parent is cleared. This method should be called * after verifying that the child Widget is an existing child of the Panel, * but before physically removing the child Widget from the DOM. The child * will now fire its {@link Widget#onDetach()} event if this Panel is * currently attached. *

*

* Calls to {@link #orphan(Widget)} should be wrapped in a try/finally block * to ensure that the widget is physically detached even if orphan throws an * exception. *

* * @param child * the widget to be disowned * @see #add(Widget) */ protected final void orphan(Widget child) { assert (child.getParent() == this); setParent(child, null); } /** * Adjusts beforeIndex to account for the possibility that the given widget * is already a child of this panel. * * @param child * the widget that might be an existing child * @param beforeIndex * the index at which it will be added to this panel * @return the modified index */ protected int adjustIndex(Widget child, int beforeIndex) { checkIndexBoundsForInsertion(beforeIndex); // Check to see if this widget is already a direct child. if (child.getParent() == this) { // If the Widget's previous position was left of the desired new // position // shift the desired position left to reflect the removal int idx = getWidgetIndex(child); if (idx < beforeIndex) { beforeIndex--; } } return beforeIndex; } /** * Checks that index is in the range [0, getWidgetCount()), * which is the valid range on accessible indexes. * * @param index * the index being accessed */ protected void checkIndexBoundsForAccess(int index) { if (index < 0 || index >= getWidgetCount()) { throw new IndexOutOfBoundsException(); } } /** * Checks that index is in the range [0, getWidgetCount()], * which is the valid range for indexes on an insertion. * * @param index * the index where insertion will occur */ protected void checkIndexBoundsForInsertion(int index) { if (index < 0 || index > getWidgetCount()) { throw new IndexOutOfBoundsException(); } } /** * Insert a new child Widget into this Panel at a specified index, attaching * its Element to the specified container Element. The child Element will * either be attached to the container at the same index, or simply appended * to the container, depending on the value of domInsert. * * @param child * the child Widget to be added * @param container * the Element within which child will be contained * @param beforeIndex * the index before which child will be inserted * @param domInsert * if true, insert child into * container at beforeIndex; otherwise * append child to the end of container * . */ protected void insert(Widget child, Element container, int beforeIndex, boolean domInsert) { // Validate index; adjust if the widget is already a child of this // panel. beforeIndex = adjustIndex(child, beforeIndex); // Detach new child. child.removeFromParent(); // Logical attach. getChildren().insert(child, beforeIndex); // Physical attach. if (domInsert) { DOM.insertChild(container, child.getElement(), beforeIndex); } else { DOM.appendChild(container, child.getElement()); } // Adopt. adopt(child); } private native void addPostCreate(JavaScriptObject componentJS) /*-{ var container = [email protected]::getOrCreateJsObj()(); container.add(componentJS); }-*/; private native void addPreCreate(JavaScriptObject componentJS) /*-{ var config = [email protected]::config; if (!config.items) { config.items = @com.emitrom.touch4j.client.core.JsoHelper::createJavaScriptArray()(); } config.items.push(componentJS); }-*/; private native void remove(String id) /*-{ var container = [email protected]::getOrCreateJsObj()(); container.remove(id); }-*/; private native void remove(String id, boolean autoDestroy) /*-{ var container = [email protected]::getOrCreateJsObj()(); container.remove(id, autoDestroy); }-*/; private native void remove(Component component, boolean autoDestroy) /*-{ var container = [email protected]::getOrCreateJsObj()(); var componentJS = [email protected]::getOrCreateJsObj()(); container.remove(componentJS, autoDestroy); }-*/; private native void _removeAll() /*-{ var container = [email protected]::getOrCreateJsObj()(); if (container.items) { var cs = container.items.items; var len = cs.length; for ( var i = 0; i < len; i++) { cs[0].hide(); container.remove(cs[0], false); } } }-*/; private native void _removeAll(boolean autoDestroy) /*-{ var container = [email protected]::getOrCreateJsObj()(); if (container.items) { var cs = container.items.items; var len = cs.length for ( var i = 0; i < len; i++) { if (!autoDestroy) cs[0].hide(); container.remove(cs[0], autoDestroy); } } }-*/; private native void adopt(Widget widget, ComplexContainer parent) /*-{ [email protected]::setParent(Lcom/google/gwt/user/client/ui/Widget;)(parent); }-*/; private native void setParent(Widget child, Widget parent) /*-{ [email protected]::setParent(Lcom/google/gwt/user/client/ui/Widget;)(parent); }-*/; private native void setActiveItemRendered(int item) /*-{ var container = [email protected]::getOrCreateJsObj()(); container.setActiveItem(item); }-*/; private native void setActiveItemRendered(JavaScriptObject item) /*-{ var container = [email protected]::getOrCreateJsObj()(); container.setActiveItem(item); }-*/; /** * Add a component. */ private void add(Component component) { JavaScriptObject componentJS = component.isCreated() ? component.getOrCreateJsObj() : component.getConfig(); if (isCreated()) { addPostCreate(componentJS); } else { addPreCreate(componentJS); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy