/* * Copyright 2000-2021 Vaadin Ltd. * * 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.vaadin.ui.dnd; import java.util.Arrays; import java.util.LinkedHashMap; import java.util.List; import java.util.Locale; import java.util.Map; import java.util.Objects; import com.vaadin.server.AbstractExtension; import com.vaadin.shared.MouseEventDetails; import com.vaadin.shared.Registration; import com.vaadin.shared.ui.dnd.DropEffect; import com.vaadin.shared.ui.dnd.DropTargetRpc; import com.vaadin.shared.ui.dnd.DropTargetState; import com.vaadin.shared.ui.dnd.criteria.ComparisonOperator; import com.vaadin.shared.ui.dnd.criteria.Criterion; import com.vaadin.ui.AbstractComponent; import com.vaadin.ui.dnd.event.DropEvent; import com.vaadin.ui.dnd.event.DropListener; /** * Extension to make a component a drop target for HTML5 drag and drop * functionality. * * @param * Type of the component to be extended. * @author Vaadin Ltd * @since 8.1 */ public class DropTargetExtension extends AbstractExtension { /** * Extends {@code target} component and makes it a drop target. * * @param target * Component to be extended. */ public DropTargetExtension(T target) { super.extend(target); } @Override public void attach() { super.attach(); registerDropTargetRpc(); } /** * Registers the server side RPC methods invoked from client side on * drop event. *

* Override this method if you need to have a custom RPC interface for * transmitting the drop event with more data. If just need to do additional * things before firing the drop event, then you should override * {@link #onDrop(List, Map, DropEffect, MouseEventDetails)} instead. */ protected void registerDropTargetRpc() { registerRpc((DropTargetRpc) (types, data, dropEffect, mouseEventDetails) -> onDrop(types, data, DropEffect.valueOf(dropEffect.toUpperCase(Locale.ROOT)), mouseEventDetails)); } /** * Invoked when a drop has been received from client side. * Fires the {@link DropEvent}. * * @param types * List of data types from {@code DataTransfer.types} object. * @param data * Map containing all types and corresponding data from the * {@code * DataTransfer} object. * @param dropEffect * the drop effect * @param mouseEventDetails * mouse event details object containing information about the * drop event */ protected void onDrop(List types, Map data, DropEffect dropEffect, MouseEventDetails mouseEventDetails) { // Create a linked map that preserves the order of types Map dataPreserveOrder = new LinkedHashMap<>(); types.forEach(type -> dataPreserveOrder.put(type, data.get(type))); DropEvent event = new DropEvent<>(getParent(), dataPreserveOrder, dropEffect, getUI().getActiveDragSource(), mouseEventDetails); fireEvent(event); } /** * Sets the drop effect for the current drop target. This is set to the * dropEffect on {@code dragenter} and {@code dragover} events. *

* NOTE: If the drop effect that doesn't match the dropEffect / * effectAllowed of the drag source, it DOES NOT prevent drop on IE and * Safari! For FireFox and Chrome the drop is prevented if there they don't * match. *

* Default value is browser dependent and can depend on e.g. modifier keys. *

* From Moz Foundation: "You can modify the dropEffect property during the * dragenter or dragover events, if for example, a particular drop target * only supports certain operations. You can modify the dropEffect property * to override the user effect, and enforce a specific drop operation to * occur. Note that this effect must be one listed within the effectAllowed * property. Otherwise, it will be set to an alternate value that is * allowed." * * @param dropEffect * the drop effect to be set or {@code null} to not modify */ public void setDropEffect(DropEffect dropEffect) { if (!Objects.equals(getState(false).dropEffect, dropEffect)) { getState().dropEffect = dropEffect; } } /** * Returns the drop effect for the current drop target. * * @return The drop effect of this drop target or {@code null} if none set * @see #setDropEffect(DropEffect) */ public DropEffect getDropEffect() { return getState(false).dropEffect; } /** * Sets a criteria script in JavaScript to allow drop on this drop target. * The script is executed when something is dragged on top of the target, * and the drop is not allowed in case the script returns {@code false}. *

* Drop will be allowed if it passes both this criteria script and the * criteria set via any of {@code setDropCriterion()} or {@code * setDropCriteria()} methods. If no criteria is set, then the drop is * always accepted, if the set {@link #setDropEffect(DropEffect) dropEffect} * matches the drag source. *

* IMPORTANT: Construct the criteria script carefully and do not * include untrusted sources such as user input. Always keep in mind that * the script is executed on the client as is. *

* Example: * *

     * target.setDropCriterion(
     *         // If dragged source contains a URL, allow it to be dropped
     *         "if (event.dataTransfer.types.includes('text/uri-list')) {"
     *                 + "    return true;" + "}" +
     *
     *                 // Otherwise cancel the event
     *                 "return false;");
     * 
* * @param criteriaScript * JavaScript to be executed when drop event happens or * {@code null} to clear. */ public void setDropCriteriaScript(String criteriaScript) { if (!Objects.equals(getState(false).criteriaScript, criteriaScript)) { getState().criteriaScript = criteriaScript; } } /** * Gets the criteria script that determines whether a drop is allowed. If * the script returns {@code false}, then it is determined the drop is not * allowed. * * @return JavaScript that executes when drop event happens. * @see #setDropCriteriaScript(String) */ public String getDropCriteriaScript() { return getState(false).criteriaScript; } /** * Set a drop criterion to allow drop on this drop target. When data is * dragged on top of the drop target, the given value is compared to the * drag source's payload with the same key. The drag passes this criterion * if the value of the payload and the value given here are equal. *

* Note that calling this method will overwrite the previously set criteria. * To set multiple criteria, call the * {@link #setDropCriteria(Criterion.Match, Criterion...)} method. *

* To handle more complex criteria, define a custom script with * {@link #setDropCriteriaScript(String)}. Drop will be allowed if both this * criterion and the criteria script are passed. * * @param key * key of the payload to be compared * @param value * value to be compared to the payload's value * @see DragSourceExtension#setPayload(String, String) */ public void setDropCriterion(String key, String value) { setDropCriteria(Criterion.Match.ANY, new Criterion(key, value)); } /** * Set a drop criterion to allow drop on this drop target. When data is * dragged on top of the drop target, the given value is compared to the * drag source's payload with the same key. The drag passes this criterion * if the value of the payload compared to the given value using the given * operator holds. *

* Note that calling this method will overwrite the previously set criteria. * To set multiple criteria, call the * {@link #setDropCriteria(Criterion.Match, Criterion...)} method. *

* To handle more complex criteria, define a custom script with * {@link #setDropCriteriaScript(String)}. Drop will be allowed if both this * criterion and the criteria script are passed. * * @param key * key of the payload to be compared * @param operator * comparison operator to be used * @param value * value to be compared to the payload's value * @see DragSourceExtension#setPayload(String, int) */ public void setDropCriterion(String key, ComparisonOperator operator, int value) { setDropCriteria(Criterion.Match.ANY, new Criterion(key, operator, value)); } /** * Set a drop criterion to allow drop on this drop target. When data is * dragged on top of the drop target, the given value is compared to the * drag source's payload with the same key. The drag passes this criterion * if the value of the payload compared to the given value using the given * operator holds. *

* Note that calling this method will overwrite the previously set criteria. * To set multiple criteria, call the * {@link #setDropCriteria(Criterion.Match, Criterion...)} method. *

* To handle more complex criteria, define a custom script with * {@link #setDropCriteriaScript(String)}. Drop will be allowed if both this * criterion and the criteria script are passed. * * @param key * key of the payload to be compared * @param operator * comparison operator to be used * @param value * value to be compared to the payload's value * @see DragSourceExtension#setPayload(String, double) */ public void setDropCriterion(String key, ComparisonOperator operator, double value) { setDropCriteria(Criterion.Match.ANY, new Criterion(key, operator, value)); } /** * Sets multiple drop criteria to allow drop on this drop target. When data * is dragged on top of the drop target, the value of the given criteria is * compared to the drag source's payload with the same key. *

* The drag passes these criteria if, depending on {@code match}, any or all * of the criteria matches the payload, that is the value of the payload * compared to the value of the criterion using the criterion's operator * holds. *

* Note that calling this method will overwrite the previously set criteria. *

* To handle more complex criteria, define a custom script with * {@link #setDropCriteriaScript(String)}. Drop will be allowed if both this * criterion and the criteria script are passed. * * @param match * defines whether any or all of the given criteria should match * to allow drop on this drop target * @param criteria * criteria to be compared to the payload */ public void setDropCriteria(Criterion.Match match, Criterion... criteria) { getState().criteriaMatch = match; getState().criteria = Arrays.asList(criteria); } /** * Attaches drop listener for the current drop target. * {@link DropListener#drop(DropEvent)} is called when drop event happens on * the client side. * * @param listener * Listener to handle drop event. * @return Handle to be used to remove this listener. */ public Registration addDropListener(DropListener listener) { return addListener(DropEvent.class, listener, DropListener.DROP_METHOD); } @Override protected DropTargetState getState() { return (DropTargetState) super.getState(); } @Override protected DropTargetState getState(boolean markAsDirty) { return (DropTargetState) super.getState(markAsDirty); } /** * Returns the component this extension is attached to. * * @return Extended component. */ @Override @SuppressWarnings("unchecked") public T getParent() { return (T) super.getParent(); } }