diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/com/vaadin/event/dd/DropHandler.java | 2 | ||||
-rw-r--r-- | src/com/vaadin/terminal/gwt/client/ui/VDragAndDropWrapper.java | 44 | ||||
-rw-r--r-- | src/com/vaadin/terminal/gwt/client/ui/VDragDropPane.java | 274 | ||||
-rw-r--r-- | src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java | 11 | ||||
-rw-r--r-- | src/com/vaadin/ui/DragAndDropWrapper.java | 39 | ||||
-rw-r--r-- | src/com/vaadin/ui/DragDropPane.java | 205 | ||||
-rw-r--r-- | src/com/vaadin/ui/Upload.java | 2 |
7 files changed, 50 insertions, 527 deletions
diff --git a/src/com/vaadin/event/dd/DropHandler.java b/src/com/vaadin/event/dd/DropHandler.java index cf6d850b7b..555bd7136b 100644 --- a/src/com/vaadin/event/dd/DropHandler.java +++ b/src/com/vaadin/event/dd/DropHandler.java @@ -16,7 +16,7 @@ import com.vaadin.event.dd.acceptCriteria.AcceptCriterion; */ public interface DropHandler extends Serializable { - public void drop(DragAndDropEvent dropEvent); + public void drop(DragAndDropEvent event); /** * Returns the {@link AcceptCriterion} used to evaluate whether the diff --git a/src/com/vaadin/terminal/gwt/client/ui/VDragAndDropWrapper.java b/src/com/vaadin/terminal/gwt/client/ui/VDragAndDropWrapper.java index 606b382c26..990dee9f3c 100644 --- a/src/com/vaadin/terminal/gwt/client/ui/VDragAndDropWrapper.java +++ b/src/com/vaadin/terminal/gwt/client/ui/VDragAndDropWrapper.java @@ -226,7 +226,6 @@ public class VDragAndDropWrapper extends VCustomComponent implements } /** - * * Currently supports only FF36 as no other browser supprots natively File * api. * @@ -238,42 +237,17 @@ public class VDragAndDropWrapper extends VCustomComponent implements public void execute() { /* * File contents is sent deferred to allow quick reaction on GUI - * although file upload may last long. TODO make this use apache - * file upload instead of our variable post like in upload. - * Currently stalls the GUI during upload. Also need to use - * dataurl to support all possible bytes in file content + * although file upload may last long. */ file.readAsBinary(new Callback() { public void handleFile(final JavaScriptObject object) { - DeferredCommand.addCommand(new Command() { - - public void execute() { - - ExtendedXHR extendedXHR = (ExtendedXHR) ExtendedXHR - .create(); - extendedXHR.open("POST", client.getAppUri()); - extendedXHR - .setRequestHeader( - "PaintableId", - client - .getPid(VDragAndDropWrapper.this)); - extendedXHR.setRequestHeader("FileId", "" - + fileId); - - // extendedXHR.setRequestHeader("Connection", - // "close"); - - multipartSend( - extendedXHR, - object, - "XHRFILE" - + client - .getPid(VDragAndDropWrapper.this) - + "." + fileId); - - } - }); + ExtendedXHR extendedXHR = (ExtendedXHR) ExtendedXHR + .create(); + extendedXHR.open("POST", client.getAppUri()); + String name = "XHRFILE" + getPid() + "." + fileId; + multipartSend(extendedXHR, object, name); + } }); @@ -282,6 +256,10 @@ public class VDragAndDropWrapper extends VCustomComponent implements } + private String getPid() { + return client.getPid(this); + } + private native void multipartSend(JavaScriptObject xhr, JavaScriptObject data, String name) /*-{ diff --git a/src/com/vaadin/terminal/gwt/client/ui/VDragDropPane.java b/src/com/vaadin/terminal/gwt/client/ui/VDragDropPane.java deleted file mode 100644 index a93d73e8da..0000000000 --- a/src/com/vaadin/terminal/gwt/client/ui/VDragDropPane.java +++ /dev/null @@ -1,274 +0,0 @@ -/* -@ITMillApache2LicenseForJavaFiles@ - */ -package com.vaadin.terminal.gwt.client.ui; - -import java.util.Map; - -import com.google.gwt.core.client.JsArrayString; -import com.google.gwt.dom.client.EventTarget; -import com.google.gwt.event.dom.client.MouseDownEvent; -import com.google.gwt.event.dom.client.MouseDownHandler; -import com.google.gwt.user.client.Command; -import com.google.gwt.user.client.DeferredCommand; -import com.google.gwt.user.client.Element; -import com.google.gwt.user.client.ui.Widget; -import com.vaadin.terminal.gwt.client.ApplicationConnection; -import com.vaadin.terminal.gwt.client.Container; -import com.vaadin.terminal.gwt.client.MouseEventDetails; -import com.vaadin.terminal.gwt.client.Paintable; -import com.vaadin.terminal.gwt.client.UIDL; -import com.vaadin.terminal.gwt.client.ui.dd.VAbstractDropHandler; -import com.vaadin.terminal.gwt.client.ui.dd.VDragAndDropManager; -import com.vaadin.terminal.gwt.client.ui.dd.VDragEvent; -import com.vaadin.terminal.gwt.client.ui.dd.VHasDropHandler; -import com.vaadin.terminal.gwt.client.ui.dd.VHtml5DragEvent; -import com.vaadin.terminal.gwt.client.ui.dd.VTransferable; - -public class VDragDropPane extends VAbsoluteLayout implements Container, - VHasDropHandler { - - private String paintableId; - - /** - * DragEvent is stored here in case of HTML5 drag event. - */ - private VDragEvent vaadinDragEvent; - - public VDragDropPane() { - super(); - addDomHandler(new MouseDownHandler() { - public void onMouseDown(MouseDownEvent event) { - EventTarget eventTarget = event.getNativeEvent() - .getEventTarget(); - Paintable paintable = client.getPaintable((Element) eventTarget - .cast()); - if (paintable != null) { - VTransferable transferable = new VTransferable(); - transferable.setDragSource(VDragDropPane.this); - transferable.setData("component", paintable); - VDragEvent drag = VDragAndDropManager.get().startDrag( - transferable, event.getNativeEvent(), true); - drag.createDragImage(((Widget) paintable).getElement(), - true); - drag.getDropDetails().put( - "mouseDown", - new MouseEventDetails(event.getNativeEvent()) - .serialize()); - event.preventDefault(); // prevent text selection - } - } - }, MouseDownEvent.getType()); - - hookHtml5Events(getElement()); - - getStyleElement().getStyle().setBackgroundColor("yellow"); - - } - - /** - * Prototype code, memory leak risk. - * - * @param el - */ - private native void hookHtml5Events(Element el) - /*-{ - - var me = this; - - if(el.addEventListener) { - el.addEventListener("dragenter", function(ev) { - return me.@com.vaadin.terminal.gwt.client.ui.VDragDropPane::html5DragEnter(Lcom/vaadin/terminal/gwt/client/ui/dd/VHtml5DragEvent;)(ev); - }, false); - - el.addEventListener("dragleave", function(ev) { - return me.@com.vaadin.terminal.gwt.client.ui.VDragDropPane::html5DragLeave(Lcom/vaadin/terminal/gwt/client/ui/dd/VHtml5DragEvent;)(ev); - }, false); - - el.addEventListener("dragover", function(ev) { - return me.@com.vaadin.terminal.gwt.client.ui.VDragDropPane::html5DragOver(Lcom/vaadin/terminal/gwt/client/ui/dd/VHtml5DragEvent;)(ev); - }, false); - - el.addEventListener("drop", function(ev) { - return me.@com.vaadin.terminal.gwt.client.ui.VDragDropPane::html5DragDrop(Lcom/vaadin/terminal/gwt/client/ui/dd/VHtml5DragEvent;)(ev); - }, false); - - } else { - el.attachEvent("ondragenter", function(ev) { - return me.@com.vaadin.terminal.gwt.client.ui.VDragDropPane::html5DragEnter(Lcom/vaadin/terminal/gwt/client/ui/dd/VHtml5DragEvent;)(ev); - }); - - el.attachEvent("ondragleave", function(ev) { - return me.@com.vaadin.terminal.gwt.client.ui.VDragDropPane::html5DragLeave(Lcom/vaadin/terminal/gwt/client/ui/dd/VHtml5DragEvent;)(ev); - }); - - el.attachEvent("ondragover", function(ev) { - return me.@com.vaadin.terminal.gwt.client.ui.VDragDropPane::html5DragOver(Lcom/vaadin/terminal/gwt/client/ui/dd/VHtml5DragEvent;)(ev); - }); - - el.attachEvent("ondrop", function(ev) { - return me.@com.vaadin.terminal.gwt.client.ui.VDragDropPane::html5DragDrop(Lcom/vaadin/terminal/gwt/client/ui/dd/VHtml5DragEvent;)(ev); - }); - } - - }-*/; - - public boolean html5DragEnter(VHtml5DragEvent event) { - ApplicationConnection.getConsole().log("HTML 5 Drag Enter"); - VTransferable transferable = new VTransferable(); - - // TODO refine api somehow so that we will now not use the event preview - // method provided by manager - vaadinDragEvent = VDragAndDropManager.get().startDrag(transferable, - event, false); - event.preventDefault(); - event.stopPropagation(); - return false; - } - - public boolean html5DragLeave(VHtml5DragEvent event) { - ApplicationConnection.getConsole().log("HTML 5 Drag Leave posponed..."); - DeferredCommand.addCommand(new Command() { - public void execute() { - // Yes, dragleave happens before drop. Makes no sense to me. - // IMO shouldn't fire leave at all if drop happens (I guess this - // is what IE does). - // In Vaadin we fire it only if drop did not happen. - if (vaadinDragEvent != null) { - ApplicationConnection.getConsole().log( - "...HTML 5 Drag Leave"); - getDropHandler().dragLeave(vaadinDragEvent); - } - } - }); - event.preventDefault(); - event.stopPropagation(); - return false; - } - - public boolean html5DragOver(VHtml5DragEvent event) { - ApplicationConnection.getConsole().log("HTML 5 Drag Over"); - getDropHandler().dragOver(vaadinDragEvent); - // needed to be set for Safari, otherwise drop will not happen - String s = event.getEffectAllowed(); - if ("all".equals(s) || s.contains("opy")) { - event.setDragEffect("copy"); - } else { - event.setDragEffect(s); - ApplicationConnection.getConsole().log("Drag effect set to " + s); - } - event.preventDefault(); - event.stopPropagation(); - return false; - } - - public boolean html5DragDrop(VHtml5DragEvent event) { - ApplicationConnection.getConsole().log("HTML 5 Drag Drop"); - VTransferable transferable = vaadinDragEvent.getTransferable(); - - JsArrayString types = event.getTypes(); - for (int i = 0; i < types.length(); i++) { - String type = types.get(i); - ApplicationConnection.getConsole().log("Type: " + type); - if ("text/plain".equals(type)) { - String data = event.getDataAsText(type); - ApplicationConnection.getConsole().log(type + " : " + data); - transferable.setData("text/plain", data); - } - } - - String fileAsString = event.getFileAsString(0); - if (fileAsString != null) { - ApplicationConnection.getConsole().log(fileAsString); - transferable.setData("fileContents", fileAsString); - } - - VDragAndDropManager.get().endDrag(); - vaadinDragEvent = null; - event.preventDefault(); - event.stopPropagation(); - - return false; - } - - @Override - public void updateFromUIDL(UIDL uidl, ApplicationConnection client) { - super.updateFromUIDL(uidl, client); - if (!uidl.hasAttribute("cached")) { - int childCount = uidl.getChildCount(); - UIDL childUIDL = uidl.getChildUIDL(childCount - 1); - getDropHandler().updateAcceptRules(childUIDL); - } - } - - private VAbstractDropHandler dropHandler; - - public VAbstractDropHandler getDropHandler() { - if (dropHandler == null) { - dropHandler = new VAbstractDropHandler() { - - @Override - public void dragEnter(VDragEvent drag) { - ApplicationConnection.getConsole().log("DDPane DragEnter"); - super.dragEnter(drag); - } - - @Override - public Paintable getPaintable() { - return VDragDropPane.this; - } - - @Override - public void dragLeave(VDragEvent drag) { - ApplicationConnection.getConsole().log("DragLeave"); - getStyleElement().getStyle().setBackgroundColor("yellow"); - } - - @Override - public boolean drop(VDragEvent drag) { - ApplicationConnection.getConsole().log( - "Drop" + drag.sinceStart()); - - if (getStyleElement().getStyle().getBackgroundColor() - .equals("yellow")) { - // not accepted - ApplicationConnection.getConsole().log( - "Drop was not accepted"); - return false; - } - - Map<String, Object> transferable = drag.getDropDetails(); - - // this is absolute layout based, and we may want to set - // component - // relatively to where the drag ended. - // need to add current location of the drop area - - int absoluteLeft = getAbsoluteLeft(); - int absoluteTop = getAbsoluteTop(); - - transferable.put("absoluteLeft", absoluteLeft); - transferable.put("absoluteTop", absoluteTop); - - getStyleElement().getStyle().setBackgroundColor("yellow"); - return super.drop(drag); - } - - @Override - protected void dragAccepted(VDragEvent drag) { - getStyleElement().getStyle().setBackgroundColor("green"); - } - - public ApplicationConnection getApplicationConnection() { - return client; - } - }; - } - return dropHandler; - } - - public Paintable getPaintable() { - return this; - } - -} diff --git a/src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java b/src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java index ada3869269..f326846c68 100644 --- a/src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java +++ b/src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java @@ -413,10 +413,15 @@ public abstract class AbstractCommunicationManager implements .split("\\."); DragAndDropWrapper ddw = (DragAndDropWrapper) idPaintableMap .get(split[0]); - - ddw.receiveFile(upstream, split[1]); - String debugId = ddw.getDebugId(); + try { + ddw.receiveFile(upstream, split[1]); + } catch (UploadException e) { + synchronized (application) { + handleChangeVariablesError(application, ddw, e, + new HashMap<String, Object>()); + } + } } else { diff --git a/src/com/vaadin/ui/DragAndDropWrapper.java b/src/com/vaadin/ui/DragAndDropWrapper.java index a848d7af40..65bcbaaa6a 100644 --- a/src/com/vaadin/ui/DragAndDropWrapper.java +++ b/src/com/vaadin/ui/DragAndDropWrapper.java @@ -26,6 +26,7 @@ import com.vaadin.terminal.gwt.client.ui.dd.VerticalDropLocation; import com.vaadin.terminal.gwt.server.AbstractApplicationServlet; import com.vaadin.ui.DragAndDropWrapper.WrapperTransferable.Html5File; import com.vaadin.ui.Upload.Receiver; +import com.vaadin.ui.Upload.UploadException; @ClientWidget(VDragAndDropWrapper.class) public class DragAndDropWrapper extends CustomComponent implements DropTarget, @@ -75,10 +76,14 @@ public class DragAndDropWrapper extends CustomComponent implements DropTarget, return files; } + /** + * {@link DragAndDropWrapper} can receive also files from client + * computer if appropriate HTML 5 features are supported on client side. + * This class wraps information about dragged file on server side. + */ public class Html5File { public String name; - private String id; private int size; private Receiver receiver; private String type; @@ -96,15 +101,20 @@ public class DragAndDropWrapper extends CustomComponent implements DropTarget, } /** - * HTML5 drags are read from client disk with a callback. This and - * possibly long transfer time forces us to receive dragged file - * contents with a callback. + * Sets the {@link Receiver} that into which the file contents will + * be written. Usage of Reveiver is similar to {@link Upload} + * component. + * + * <p> + * <em>Note!</em> receiving file contents is experimental feature + * depending on HTML 5 API's. It is supported only by Firefox 3.6 at + * this time. * * @param receiver * the callback that returns stream where the * implementation writes the file contents as it arrives. */ - public void receive(Receiver receiver) { + public void setReceiver(Receiver receiver) { this.receiver = receiver; } @@ -178,6 +188,12 @@ public class DragAndDropWrapper extends CustomComponent implements DropTarget, private DragStartMode dragStartMode = DragStartMode.NONE; + /** + * Wraps given component in a {@link DragAndDropWrapper}. + * + * @param root + * the component to be wrapped + */ public DragAndDropWrapper(Component root) { super(root); } @@ -228,12 +244,15 @@ public class DragAndDropWrapper extends CustomComponent implements DropTarget, * This method should only be used by Vaadin terminal implementation. This * is not end user api. * - * TODO should fire progress events + end/succes events like upload + * TODO should fire progress events + end/succes events like upload. Not + * critical until we have a wider browser support for HTML5 File API * * @param upstream * @param fileId + * @throws UploadException */ - public void receiveFile(UploadStream upstream, String fileId) { + public void receiveFile(UploadStream upstream, String fileId) + throws UploadException { Html5File file = receivers.get(fileId); if (file != null && file.receiver != null) { OutputStream receiveUpload = file.receiver.receiveUpload(file @@ -247,10 +266,10 @@ public class DragAndDropWrapper extends CustomComponent implements DropTarget, receiveUpload.write(buf, 0, bytesRead); } } catch (IOException e) { - // TODO Auto-generated catch block - e.printStackTrace(); + throw new UploadException(e); } - + // clean up the reference when file is downloaded + receivers.remove(fileId); } } diff --git a/src/com/vaadin/ui/DragDropPane.java b/src/com/vaadin/ui/DragDropPane.java deleted file mode 100644 index b3dc9bd30f..0000000000 --- a/src/com/vaadin/ui/DragDropPane.java +++ /dev/null @@ -1,205 +0,0 @@ -/* -@ITMillApache2LicenseForJavaFiles@ - */ -package com.vaadin.ui; - -import java.util.Map; - -import com.vaadin.event.DataBoundTransferable; -import com.vaadin.event.Transferable; -import com.vaadin.event.dd.DragAndDropEvent; -import com.vaadin.event.dd.DropHandler; -import com.vaadin.event.dd.DropTarget; -import com.vaadin.event.dd.DropTargetDetails; -import com.vaadin.event.dd.DropTargetDetailsImpl; -import com.vaadin.event.dd.acceptCriteria.AcceptAll; -import com.vaadin.event.dd.acceptCriteria.AcceptCriterion; -import com.vaadin.terminal.PaintException; -import com.vaadin.terminal.PaintTarget; -import com.vaadin.terminal.gwt.client.MouseEventDetails; - -/** - * Test class that implements various component related Drag and Drop features. - * - * TODO consider implementing this kind of general purpose layout class: - * - * - extend simple component (CssLayout or CustomComponent, instead of absolute - * layout) - * - * - implement both drag source and drop handler with server side api - * - * - implement html5 drop target etc - * - * - include a lots of details for drop event (coordinates & sizes of drop - * position and widget/Paintable hierarchy up to drop handler) - * - * This way we could have one rather complex dd component that could be used (by - * wrapping layouts) in most common situations with server side api. Core - * layouts wouldn't need changes (and have regression risk/ performance - * penalty). - * - * @deprecated use {@link DragAndDropWrapper} instead - * - */ -@Deprecated -@SuppressWarnings("serial") -@ClientWidget(com.vaadin.terminal.gwt.client.ui.VDragDropPane.class) -public class DragDropPane extends AbsoluteLayout implements DropTarget { - - private DropHandler dropHandler; - - public DragDropPane(DropHandler dropHandler) { - setWidth("400px"); - setHeight("300px"); - - if (dropHandler == null) { - this.dropHandler = new ImportPrettyMuchAnything(); - } else { - this.dropHandler = dropHandler; - } - - } - - public DragDropPane() { - this(null); - } - - @Override - public void paintContent(PaintTarget target) throws PaintException { - super.paintContent(target); - dropHandler.getAcceptCriterion().paint(target); - } - - public DropHandler getDropHandler() { - return dropHandler; - } - - public static class ImportPrettyMuchAnything implements DropHandler { - public void drop(DragAndDropEvent event) { - DragDropPane pane = (DragDropPane) event.getDropTargetDetails() - .getTarget(); - - DragEventDetails ed = (DragEventDetails) event - .getDropTargetDetails(); - Transferable ctr = event.getTransferable(); - if (ctr.getSourceComponent() != null) { - // use "component" (from DragDropPane) if available, else take - // the source component - Component component = (Component) ctr.getData("component"); - if (component == null) { - component = ctr.getSourceComponent(); - } - - if (component.getParent() != pane) { - if (ctr instanceof DataBoundTransferable) { - // Item has been dragged, construct a Label from - // Item id - Label l = new Label(); - l.setSizeUndefined(); - l.setValue("ItemId : " - + ((DataBoundTransferable) ctr).getItemId()); - pane.addComponent(l); - component = l; - - } else { - // we have a component that is been dragged, add - // it - // to - // this - pane.addComponent(component); - } - - Integer left = ed.getAbsoluteLeft(); - Integer top = ed.getAbsoluteTop(); - - MouseEventDetails eventDetails = ed.getMouseEvent(); - - int clientX = eventDetails.getClientX(); - int clientY = eventDetails.getClientY(); - - try { - pane.getPosition(component).setTopValue(clientY - top); - pane.getPosition(component) - .setLeftValue(clientX - left); - } catch (Exception e) { - // TODO: handle exception - } - } else { - // drag ended inside the this Pane - - MouseEventDetails start = ed.getMouseDownEvent(); - MouseEventDetails eventDetails = ed.getMouseEvent(); - - int deltaX = eventDetails.getClientX() - start.getClientX(); - int deltaY = eventDetails.getClientY() - start.getClientY(); - - ComponentPosition p = pane.getPosition(component); - p.setTopValue(p.getTopValue() + deltaY); - p.setLeftValue(p.getLeftValue() + deltaX); - - } - - } else { - // drag coming outside of Vaadin - String object = (String) ctr.getData("text/plain"); - - String content = (String) ctr.getData("fileContents"); - - Label l = new Label(); - l.setCaption("Generated from HTML5 drag:"); - if (object != null) { - l.setValue(object); - } else { - l.setValue("HTML5 dd"); - } - - l.setDescription(content); - l.setSizeUndefined(); - - pane.addComponent(l); - - } - return; - } - - public AcceptCriterion getAcceptCriterion() { - return AcceptAll.get(); - } - } - - class DragEventDetails extends DropTargetDetailsImpl { - - public DragEventDetails(Map<String, Object> rawVariables) { - super(rawVariables); - } - - public Integer getAbsoluteTop() { - return (Integer) getData("absoluteTop"); - } - - public Integer getAbsoluteLeft() { - return (Integer) getData("absoluteLeft"); - } - - public MouseEventDetails getMouseDownEvent() { - return MouseEventDetails.deSerialize((String) getData("mouseDown")); - } - - public MouseEventDetails getMouseEvent() { - return MouseEventDetails - .deSerialize((String) getData("mouseEvent")); - } - - } - - public DropTargetDetails translateDropTargetDetails( - Map<String, Object> clientVariables) { - return new DragEventDetails(clientVariables); - } - - public void setDropHandler(DropHandler dropHandler2) { - dropHandler = dropHandler2; - requestRepaint(); - } - -} diff --git a/src/com/vaadin/ui/Upload.java b/src/com/vaadin/ui/Upload.java index 7a3fac9167..fdad5a75cb 100644 --- a/src/com/vaadin/ui/Upload.java +++ b/src/com/vaadin/ui/Upload.java @@ -311,7 +311,7 @@ public class Upload extends AbstractComponent implements Component.Focusable { } - public class UploadException extends Exception { + public static class UploadException extends Exception { public UploadException(Exception e) { super("Upload failed", e); } |