You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

DropTargetExtensionConnector.java 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. /*
  2. * Copyright 2000-2016 Vaadin Ltd.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License"); you may not
  5. * use this file except in compliance with the License. You may obtain a copy of
  6. * the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
  12. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
  13. * License for the specific language governing permissions and limitations under
  14. * the License.
  15. */
  16. package com.vaadin.client.extensions;
  17. import java.util.ArrayList;
  18. import java.util.HashMap;
  19. import java.util.List;
  20. import java.util.Map;
  21. import com.google.gwt.core.client.JsArrayString;
  22. import com.google.gwt.dom.client.DataTransfer;
  23. import com.google.gwt.dom.client.Element;
  24. import com.google.gwt.dom.client.NativeEvent;
  25. import com.google.gwt.user.client.ui.Widget;
  26. import com.vaadin.client.BrowserInfo;
  27. import com.vaadin.client.ComponentConnector;
  28. import com.vaadin.client.ServerConnector;
  29. import com.vaadin.client.ui.AbstractComponentConnector;
  30. import com.vaadin.shared.ui.Connect;
  31. import com.vaadin.shared.ui.dnd.DropEffect;
  32. import com.vaadin.shared.ui.dnd.DropTargetRpc;
  33. import com.vaadin.shared.ui.dnd.DropTargetState;
  34. import com.vaadin.ui.dnd.DropTargetExtension;
  35. import elemental.events.Event;
  36. import elemental.events.EventListener;
  37. import elemental.events.EventTarget;
  38. /**
  39. * Extension to add drop target functionality to a widget for using HTML5 drag
  40. * and drop. Client side counterpart of {@link DropTargetExtension}.
  41. *
  42. * @author Vaadin Ltd
  43. * @since 8.1
  44. */
  45. @Connect(DropTargetExtension.class)
  46. public class DropTargetExtensionConnector extends AbstractExtensionConnector {
  47. /**
  48. * Style name suffix for dragging data over the center of the drop target.
  49. */
  50. protected static final String STYLE_SUFFIX_DRAG_CENTER = "-drag-center";
  51. /**
  52. * Style name suffix for dragging data over the top part of the drop target.
  53. */
  54. protected static final String STYLE_SUFFIX_DRAG_TOP = "-drag-top";
  55. /**
  56. * Style name suffix for dragging data over the bottom part of the drop
  57. * target.
  58. */
  59. protected static final String STYLE_SUFFIX_DRAG_BOTTOM = "-drag-bottom";
  60. /**
  61. * Style name suffix for indicating that the element is drop target.
  62. */
  63. protected static final String STYLE_SUFFIX_DROPTARGET = "-droptarget";
  64. // Create event listeners
  65. private final EventListener dragEnterListener = this::onDragEnter;
  66. private final EventListener dragOverListener = this::onDragOver;
  67. private final EventListener dragLeaveListener = this::onDragLeave;
  68. private final EventListener dropListener = this::onDrop;
  69. /**
  70. * Widget of the drop target component.
  71. */
  72. private Widget dropTargetWidget;
  73. /**
  74. * Class name to apply when an element is dragged over the center of the
  75. * target.
  76. */
  77. private String styleDragCenter;
  78. @Override
  79. protected void extend(ServerConnector target) {
  80. dropTargetWidget = ((ComponentConnector) target).getWidget();
  81. // HTML5 DnD is by default not enabled for mobile devices
  82. if (BrowserInfo.get().isTouchDevice() && !getConnection()
  83. .getUIConnector().isMobileHTML5DndEnabled()) {
  84. return;
  85. }
  86. addDropListeners(getDropTargetElement());
  87. ((AbstractComponentConnector) target).onDropTargetAttached();
  88. // Add drop target indicator to the drop target element
  89. addDropTargetStyle();
  90. }
  91. /**
  92. * Adds dragenter, dragover, dragleave and drop event listeners to the given
  93. * DOM element.
  94. *
  95. * @param element
  96. * DOM element to attach event listeners to.
  97. */
  98. private void addDropListeners(Element element) {
  99. EventTarget target = element.cast();
  100. target.addEventListener(Event.DRAGENTER, dragEnterListener);
  101. target.addEventListener(Event.DRAGOVER, dragOverListener);
  102. target.addEventListener(Event.DRAGLEAVE, dragLeaveListener);
  103. target.addEventListener(Event.DROP, dropListener);
  104. }
  105. /**
  106. * Removes dragenter, dragover, dragleave and drop event listeners from the
  107. * given DOM element.
  108. *
  109. * @param element
  110. * DOM element to remove event listeners from.
  111. */
  112. private void removeDropListeners(Element element) {
  113. EventTarget target = element.cast();
  114. target.removeEventListener(Event.DRAGENTER, dragEnterListener);
  115. target.removeEventListener(Event.DRAGOVER, dragOverListener);
  116. target.removeEventListener(Event.DRAGLEAVE, dragLeaveListener);
  117. target.removeEventListener(Event.DROP, dropListener);
  118. }
  119. @Override
  120. public void onUnregister() {
  121. super.onUnregister();
  122. removeDropListeners(getDropTargetElement());
  123. ((AbstractComponentConnector) getParent()).onDropTargetDetached();
  124. // Remove drop target indicator
  125. removeDropTargetStyle();
  126. }
  127. /**
  128. * Finds the drop target element within the widget. By default, returns the
  129. * topmost element.
  130. *
  131. * @return the drop target element in the parent widget.
  132. */
  133. protected Element getDropTargetElement() {
  134. return dropTargetWidget.getElement();
  135. }
  136. /**
  137. * Event handler for the {@code dragenter} event.
  138. * <p>
  139. * Override this method in case custom handling for the dragstart event is
  140. * required. If the drop is allowed, the event should prevent default.
  141. *
  142. * @param event
  143. * browser event to be handled
  144. */
  145. protected void onDragEnter(Event event) {
  146. NativeEvent nativeEvent = (NativeEvent) event;
  147. // Generate style name for drop target
  148. styleDragCenter = dropTargetWidget.getStylePrimaryName()
  149. + STYLE_SUFFIX_DRAG_CENTER;
  150. if (isDropAllowed(nativeEvent)) {
  151. addDragOverStyle(nativeEvent);
  152. setDropEffect(nativeEvent);
  153. // According to spec, need to call this for allowing dropping, the
  154. // default action would be to reject as target
  155. event.preventDefault();
  156. } else {
  157. // Remove drop effect
  158. nativeEvent.getDataTransfer()
  159. .setDropEffect(DataTransfer.DropEffect.NONE);
  160. }
  161. }
  162. /**
  163. * Set the drop effect for the dragenter / dragover event, if one has been
  164. * set from server side.
  165. * <p>
  166. * From Moz Foundation: "You can modify the dropEffect property during the
  167. * dragenter or dragover events, if for example, a particular drop target
  168. * only supports certain operations. You can modify the dropEffect property
  169. * to override the user effect, and enforce a specific drop operation to
  170. * occur. Note that this effect must be one listed within the effectAllowed
  171. * property. Otherwise, it will be set to an alternate value that is
  172. * allowed."
  173. *
  174. * @param event
  175. * the dragenter or dragover event.
  176. */
  177. private void setDropEffect(NativeEvent event) {
  178. if (getState().dropEffect != null) {
  179. DataTransfer.DropEffect dropEffect = DataTransfer.DropEffect
  180. // the valueOf() needs to have equal string and name()
  181. // doesn't return in all upper case
  182. .valueOf(getState().dropEffect.name().toUpperCase());
  183. event.getDataTransfer().setDropEffect(dropEffect);
  184. }
  185. }
  186. /**
  187. * Event handler for the {@code dragover} event.
  188. * <p>
  189. * Override this method in case custom handling for the dragover event is
  190. * required. If the drop is allowed, the event should prevent default.
  191. *
  192. * @param event
  193. * browser event to be handled
  194. */
  195. protected void onDragOver(Event event) {
  196. NativeEvent nativeEvent = (NativeEvent) event;
  197. if (isDropAllowed(nativeEvent)) {
  198. setDropEffect(nativeEvent);
  199. // Add drag over indicator in case the element doesn't have one
  200. addDragOverStyle(nativeEvent);
  201. // Prevent default to allow drop
  202. nativeEvent.preventDefault();
  203. nativeEvent.stopPropagation();
  204. } else {
  205. // Remove drop effect
  206. nativeEvent.getDataTransfer()
  207. .setDropEffect(DataTransfer.DropEffect.NONE);
  208. // Remove drag over indicator
  209. removeDragOverStyle(nativeEvent);
  210. }
  211. }
  212. /**
  213. * Event handler for the {@code dragleave} event.
  214. * <p>
  215. * Override this method in case custom handling for the dragleave event is
  216. * required.
  217. *
  218. * @param event
  219. * browser event to be handled
  220. */
  221. protected void onDragLeave(Event event) {
  222. removeDragOverStyle((NativeEvent) event);
  223. }
  224. /**
  225. * Event handler for the {@code drop} event.
  226. * <p>
  227. * Override this method in case custom handling for the drop event is
  228. * required. If the drop is allowed, the event should prevent default.
  229. *
  230. * @param event
  231. * browser event to be handled
  232. */
  233. protected void onDrop(Event event) {
  234. NativeEvent nativeEvent = (NativeEvent) event;
  235. if (isDropAllowed(nativeEvent)) {
  236. nativeEvent.preventDefault();
  237. nativeEvent.stopPropagation();
  238. JsArrayString typesJsArray = getTypes(
  239. nativeEvent.getDataTransfer());
  240. List<String> types = new ArrayList<>();
  241. Map<String, String> data = new HashMap<>();
  242. for (int i = 0; i < typesJsArray.length(); i++) {
  243. String type = typesJsArray.get(i);
  244. types.add(type);
  245. data.put(type, nativeEvent.getDataTransfer().getData(type));
  246. }
  247. sendDropEventToServer(types, data, DragSourceExtensionConnector
  248. .getDropEffect(nativeEvent.getDataTransfer()), nativeEvent);
  249. }
  250. removeDragOverStyle(nativeEvent);
  251. }
  252. private boolean isDropAllowed(NativeEvent event) {
  253. // there never should be a drop when effect has been set to none
  254. if (getState().dropEffect != null
  255. && getState().dropEffect == DropEffect.NONE) {
  256. return false;
  257. }
  258. // TODO #9246: Should add verification for checking effectAllowed and
  259. // dropEffect from event and comparing that to target's dropEffect.
  260. // Currently Safari, Edge and IE don't follow the spec by allowing drop
  261. // if those don't match
  262. if (getState().dropCriteria != null) {
  263. return executeScript(event, getState().dropCriteria);
  264. }
  265. // Allow when criteria not set
  266. return true;
  267. }
  268. /**
  269. * Initiates a server RPC for the drop event.
  270. *
  271. * @param types
  272. * List of data types from {@code DataTransfer.types} object.
  273. * @param data
  274. * Map containing all types and corresponding data from the {@code
  275. * DataTransfer} object.
  276. * @param dropEffect
  277. * The desired drop effect.
  278. */
  279. protected void sendDropEventToServer(List<String> types,
  280. Map<String, String> data, String dropEffect,
  281. NativeEvent dropEvent) {
  282. getRpcProxy(DropTargetRpc.class).drop(types, data, dropEffect);
  283. }
  284. /**
  285. * Add class name for the drop target element indicating that data can be
  286. * dropped onto it. The class name has the following format:
  287. * <pre>
  288. * [primaryStyleName]-droptarget
  289. * </pre>
  290. * The added class name is update
  291. * automatically by the framework when the primary style name changes.
  292. */
  293. protected void addDropTargetStyle() {
  294. getDropTargetElement().addClassName(
  295. getStylePrimaryName(getDropTargetElement())
  296. + STYLE_SUFFIX_DROPTARGET);
  297. }
  298. /**
  299. * Remove class name from the drop target element indication that data can
  300. * be dropped onto it.
  301. */
  302. protected void removeDropTargetStyle() {
  303. getDropTargetElement().removeClassName(
  304. getStylePrimaryName(getDropTargetElement())
  305. + STYLE_SUFFIX_DROPTARGET);
  306. }
  307. /**
  308. * Add class that indicates that the component is a target while data is
  309. * being dragged over it.
  310. * <p>
  311. * This is triggered on {@link #onDragEnter(Event) dragenter} and
  312. * {@link #onDragOver(Event) dragover} events pending if the drop is
  313. * possible. The drop is possible if the drop effect for the target and
  314. * source do match and the drop criteria script evaluates to true or is not
  315. * set.
  316. *
  317. * @param event
  318. * the dragenter or dragover event that triggered the indication.
  319. */
  320. protected void addDragOverStyle(NativeEvent event) {
  321. getDropTargetElement().addClassName(styleDragCenter);
  322. }
  323. /**
  324. * Remove the drag over indicator class name from the target element.
  325. * <p>
  326. * This is triggered on {@link #onDrop(Event) drop},
  327. * {@link #onDragLeave(Event) dragleave} and {@link #onDragOver(Event)
  328. * dragover} events pending on whether the drop has happened or if it is not
  329. * possible. The drop is not possible if the drop effect for the source and
  330. * target don't match or if there is a drop criteria script that evaluates
  331. * to false.
  332. *
  333. * @param event
  334. * the event that triggered the removal of the indicator
  335. */
  336. protected void removeDragOverStyle(NativeEvent event) {
  337. getDropTargetElement().removeClassName(styleDragCenter);
  338. }
  339. private native JsArrayString getTypes(DataTransfer dataTransfer)
  340. /*-{
  341. return dataTransfer.types;
  342. }-*/;
  343. private native boolean executeScript(NativeEvent event, String script)
  344. /*-{
  345. return new Function('event', script)(event);
  346. }-*/;
  347. private native boolean getStylePrimaryName(Element element)
  348. /*-{
  349. return @com.google.gwt.user.client.ui.UIObject::getStylePrimaryName(Lcom/google/gwt/dom/client/Element;)(element);
  350. }-*/;
  351. @Override
  352. public DropTargetState getState() {
  353. return (DropTargetState) super.getState();
  354. }
  355. }