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

org.wicketstuff.jwicket.ui.dragdrop.DroppableBehavior Maven / Gradle / Ivy

package org.wicketstuff.jwicket.ui.dragdrop;


import org.apache.wicket.Component;
import org.apache.wicket.ajax.AjaxRequestTarget;
import org.apache.wicket.markup.head.IHeaderResponse;
import org.apache.wicket.request.Request;
import org.apache.wicket.util.visit.IVisitor;
import org.wicketstuff.jwicket.*;
import org.wicketstuff.jwicket.ui.AbstractJqueryUiEmbeddedBehavior;


/**
 * You can add an instance of this class to a Wicket {@link Component} to make it
 * a droppable {@link Component} i.e. the target of a drag operation.
 * An instance of this class can be added to one and only one
 * {@link Component}. Another {@link Component} that should have exactly the
 * same behavior needs it's own instance.
 */
public class DroppableBehavior extends AbstractDragDropBehavior {

    private static final long serialVersionUID = 1L;

    private static final String DROPPED_COMPONENTID_IDENTIFIER = "wsjqDroppedComponent";

    public static final JQueryResourceReference uiDroppableJs
            = JQuery.isDebug()
            ? new JQueryResourceReference(DraggableBehavior.class, "jquery.ui.droppable.js")
            : new JQueryResourceReference(DraggableBehavior.class, "jquery.ui.droppable.min.js");

    private JsMap options = new JsMap();

    private DraggablesAcceptedByDroppable draggablesAcceptedByDroppable = null;

    // We need to render the drop accept checker function only once
    private boolean dropAcceptedCheckerRendered = false;


    public DroppableBehavior() {
        super(AbstractJqueryUiEmbeddedBehavior.jQueryUiWidgetJs,
                AbstractJqueryUiEmbeddedBehavior.jQueryUiMouseJs,
                DraggableBehavior.jQueryUiDraggableJs,
                uiDroppableJs,
                SpecialKeys.specialKeysJs);
    }

    private boolean onActivatedNotificationWanted = false;

    /**
     * If set to {@code true}, the callback-Method {@link #onActivate(AjaxRequestTarget, Component, SpecialKeys)}
     * is called when the drag operation ends.
     *
     * @param value {@code true} or {@code false}.
     * @return this object
     */
    public DroppableBehavior setWantOnActivatedNotification(final boolean value) {
        this.onActivatedNotificationWanted = value;
        return this;
    }


    private boolean onDeactivateNotificationWanted = false;

    /**
     * If set to {@code true}, the callback-Method {@link #onDeactivate(AjaxRequestTarget, Component, SpecialKeys)}
     * is called when the drag operation ends.
     *
     * @param value {@code true} or {@code false}.
     * @return this object
     */
    public DroppableBehavior setWantOnDeactivateNotification(final boolean value) {
        this.onDeactivateNotificationWanted = value;
        return this;
    }

    /**
     * You can restrict the {@link Component}s that can be dragged and dropped to
     * a {@link Component} marked with this behavior. See {@link DraggablesAcceptedByDroppable}
     * for more information. If the {@link DraggablesAcceptedByDroppable} is empty (i.e. it
     * contains no names) then the droppable does not accept any draggables.
     *
     * @param accepted the accepted
     * @return this object
     */
    public DroppableBehavior setDraggablesAcceptedByDroppable(final DraggablesAcceptedByDroppable accepted) {
        this.draggablesAcceptedByDroppable = accepted;
        if (accepted != null) {
            this.options.put("accept", new JsFunction(accepted.getJsAcceptCheckerFunctionName()));
        } else {
            this.options.remove("accept");
        }
        return this;
    }


    /**
     * Sets the 'activeClass' property for this draggable. Please consult the
     * jQuery documentation for a detailled description of this property.
     *
     * @param activeClass the CSS class' name
     * @return this object
     */
    public DroppableBehavior setActiveClass(final String activeClass) {
        if (activeClass == null) {
            this.options.remove("activeClass");
        } else {
            this.options.put("activeClass", activeClass);
        }
        return this;
    }

    public DroppableBehavior setActiveClass(final AjaxRequestTarget target, final String activeClass) {
        setActiveClass(activeClass);
        if (activeClass != null) {
            target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','activeClass','" + activeClass + "');");
        } else {
            target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','activeClass',false);");
        }
        return this;
    }


    /**
     * Sets the 'addClasses' property for this draggable. Please consult the
     * jQuery documentation for a detailled description of this property.
     *
     * @param value {@code true} or {@code false}.
     * @return this object
     */
    public DroppableBehavior setAddClasses(final boolean value) {
        if (value) {
            this.options.remove("addClasses");
        } else {
            this.options.put("addClasses", value);
        }
        return this;
    }

    public DroppableBehavior setAddClasses(final AjaxRequestTarget target, final boolean value) {
        setAddClasses(value);
        target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','addClasses'," + value + ");");
        return this;
    }


    /**
     * Sets the 'greedy' property for this draggable. Please consult the
     * jQuery documentation for a detailled description of this property.
     *
     * @param value {@code true} or {@code false}.
     * @return this object
     */
    public DroppableBehavior setGreedy(final boolean value) {
        if (value) {
            this.options.remove("greedy");
        } else {
            this.options.put("greedy", value);
        }
        return this;
    }

    public DroppableBehavior setGreedy(final AjaxRequestTarget target, final boolean value) {
        setGreedy(value);
        target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','greedy'," + value + ");");
        return this;
    }


    /**
     * Sets the 'hoverClass' property for this draggable. Please consult the
     * jQuery documentation for a detailled description of this property.
     *
     * @param hoverClass the CSS class' name
     * @return this object
     */
    public DroppableBehavior setHoverClass(final String hoverClass) {
        if (hoverClass == null) {
            this.options.remove("hoverClass");
        } else {
            this.options.put("hoverClass", hoverClass);
        }
        return this;
    }

    public DroppableBehavior setHoverClass(final AjaxRequestTarget target, final String hoverClass) {
        setHoverClass(hoverClass);
        if (hoverClass != null) {
            target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','hoverClass','" + hoverClass + "');");
        } else {
            target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','hoverClass',false);");
        }
        return this;
    }


    /**
     * Sets the 'scope' property for this draggable. Please consult the
     * jQuery documentation for a detailled description of this property.
     *
     * @param scope the scode
     * @return this object
     */
    public DroppableBehavior setScope(final String scope) {
        if (scope == null) {
            this.options.remove("scope");
        } else {
            this.options.put("scope", scope);
        }
        return this;
    }

    public DroppableBehavior setScope(final AjaxRequestTarget target, final String scope) {
        setScope(scope);
        if (scope != null) {
            target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','scope','" + scope + "');");
        } else {
            target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','scope','default');");
        }
        return this;
    }


    public static enum DropTolerance {

        FTI("fit"),
        INTERSECT("intersect"),
        POINTER("pointer"),
        TOUCH("touch");

        private final String value;

        private DropTolerance(final String value) {
            this.value = value;
        }

        public String getValue() {
            return this.value;
        }

        public String toString() {
            return this.value;
        }
    }


    /**
     * Sets the 'tolerance' property for this droppable. Please consult the
     * jquery documentation for a detailled description of this property.
     *
     * @param tolerance the tolerance
     * @return this object
     */
    public DroppableBehavior setTolerance(final DropTolerance tolerance) {
        if (tolerance == null) {
            this.options.remove("tolerance");
        } else {
            this.options.put("tolerance", tolerance.getValue());
        }
        return this;
    }

    public DroppableBehavior setScope(final AjaxRequestTarget target, final DropTolerance tolerance) {
        setTolerance(tolerance);
        if (tolerance != null) {
            target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','tolerance','" + tolerance.getValue() + "');");
        } else {
            target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('option','tolerance','" + DropTolerance.INTERSECT + "');");
        }
        return this;
    }


    /**
     * handles the event processing during dragging.
     */
    @Override
    protected void respond(final AjaxRequestTarget target) {
        Component component = getComponent();
        Request request;

        if (component != null && (request = component.getRequest()) != null) {
            EventType dragEventType = EventType.stringToType(request.getRequestParameters().getParameterValue(EventType.IDENTIFIER).toString());

            IVisitor visitor = getComponentFindingVisitor(request.getRequestParameters().getParameterValue(DROPPED_COMPONENTID_IDENTIFIER).toString());
            Component foundComponent = component.getPage().visitChildren(visitor);

            if (component instanceof IDroppable) {
                IDroppable draggableComponent = (IDroppable) component;
                if (dragEventType == EventType.DROP) {
                    draggableComponent.onDrop(target, foundComponent, new SpecialKeys(request));
                } else if (dragEventType == EventType.DROP_ACTIVATE) {
                    draggableComponent.onActivate(target, foundComponent, new SpecialKeys(request));
                } else if (dragEventType == EventType.DROP_DEACTIVATE) {
                    draggableComponent.onDeactivate(target, foundComponent, new SpecialKeys(request));
                }
            }


            if (dragEventType == EventType.DROP) {
                onDrop(target, foundComponent, new SpecialKeys(request));
            } else if (dragEventType == EventType.DROP_ACTIVATE) {
                onActivate(target, foundComponent, new SpecialKeys(request));
            } else if (dragEventType == EventType.DROP_DEACTIVATE) {
                onDeactivate(target, foundComponent, new SpecialKeys(request));
            }
        }
    }

    protected IVisitor getComponentFindingVisitor(String markupId) {
        return new ComponentFinder(markupId);
    }


    /**
     * This method is called when a draggable {@link Component} is dropped onto
     * a {@link Component} marked with this behavior.
     *
     * @param target           the AjaxRequestTarget of the drop operation.
     * @param draggedComponent The dragged component
     * @param specialKeys      the special keys that were pressed when the event occurs
     */
    protected void onDrop(AjaxRequestTarget target, final Component draggedComponent, final SpecialKeys specialKeys) {
    }


    /**
     * This method is called when a draggable {@link Component} is starting to
     * drag and the dragging {@link Component}'s name is accepted to be
     * dropped onto this.
     *
     * @param target           The {@link AjaxRequestTarget} associated with this
     *                         drop operation.
     * @param draggedComponent The dragged component
     * @param specialKeys      the special keys that were pressed when the event occurs
     */
    protected void onActivate(final AjaxRequestTarget target, final Component draggedComponent, final SpecialKeys specialKeys) {
    }


    /**
     * This method is called when a draggable {@link Component} has stopped
     * dragging and the dragging {@link Component}'s name was accepted to be
     * dropped onto this.
     *
     * @param target           The {@link AjaxRequestTarget} associated with this
     *                         drop operation.
     * @param draggedComponent The dragged component
     * @param specialKeys      the special keys that were pressed when the event occurs
     */
    protected void onDeactivate(final AjaxRequestTarget target, final Component draggedComponent, final SpecialKeys specialKeys) {
    }


    /**
     * Disable the dropping
     *
     * @param target An AjaxRequestTarget
     */
    public void disable(final AjaxRequestTarget target) {
        target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('disable');");
    }


    /**
     * Enable the dropping
     *
     * @param target An AjaxRequestTarget
     */
    public void enable(final AjaxRequestTarget target) {
        target.appendJavaScript("jQuery('#" + getComponent().getMarkupId() + "').droppable('enable');");
    }


    @Override
    public void renderHead(Component component, final IHeaderResponse response) {
        super.renderHead(component, response);
        if (this.draggablesAcceptedByDroppable != null && !this.dropAcceptedCheckerRendered) {
            this.draggablesAcceptedByDroppable.renderJsDropAcceptFunction(response);
            this.dropAcceptedCheckerRendered = true;
        }
    }


    @Override
    /**
     * For internal use only.
     */
    protected JsBuilder getJsBuilder() {
        if (this.onDeactivateNotificationWanted) {
            this.options.put("deactivate",
                    new JsFunction("function(ev,ui) { \n" +
                            "wicketAjaxGet('" +
                            this.getCallbackUrl() +
                            "&" + EventType.IDENTIFIER + "=" + EventType.DROP_DEACTIVATE +
                            "&" + DROPPED_COMPONENTID_IDENTIFIER + "='+jQuery(ui.draggable).attr('id')" +
                            "+'&keys='+jQuery.jWicketSpecialKeysGetPressed()" +
                            "); }"));
        }

        this.options.put("drop",
                new JsFunction("function(ev,ui) { \n" +
                        "wicketAjaxGet('" +
                        this.getCallbackUrl() +
                        "&" + EventType.IDENTIFIER + "=" + EventType.DROP +
                        "&" + DROPPED_COMPONENTID_IDENTIFIER + "='+jQuery(ui.draggable).attr('id')" +
                        "+'&keys='+jQuery.jWicketSpecialKeysGetPressed()" +
                        "); " +
                        "}"));

        if (this.onActivatedNotificationWanted) {
            this.options.put("activate",
                    new JsFunction("function(ev,ui) { \n" +
                            "wicketAjaxGet('" +
                            this.getCallbackUrl() +
                            "&" + EventType.IDENTIFIER + "=" + EventType.DROP_ACTIVATE +
                            "&" + DROPPED_COMPONENTID_IDENTIFIER + "='+jQuery(ui.draggable).attr('id')" +
                            "+'&keys='+jQuery.jWicketSpecialKeysGetPressed()" +
                            "); }"));
        }

        JsBuilder builder = new JsBuilder();


        builder.append("jQuery('#" + getComponent().getMarkupId() + "').droppable(");
        builder.append("{");
        builder.append(this.options.toString(this.rawOptions));
        builder.append("}");
        builder.append(");");


        return builder;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy