com.vaadin.flow.component.upload.Upload Maven / Gradle / Ivy
/*
* Copyright 2000-2017 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.flow.component.upload;
import java.io.OutputStream;
import java.util.ArrayDeque;
import java.util.Arrays;
import java.util.Collections;
import java.util.Deque;
import java.util.List;
import java.util.Objects;
import com.vaadin.flow.component.Component;
import com.vaadin.flow.component.ComponentEventListener;
import com.vaadin.flow.component.HasSize;
import com.vaadin.flow.component.UI;
import com.vaadin.flow.dom.Element;
import com.vaadin.flow.function.SerializableConsumer;
import com.vaadin.flow.internal.JsonSerializer;
import com.vaadin.flow.server.NoInputStreamException;
import com.vaadin.flow.server.NoOutputStreamException;
import com.vaadin.flow.server.StreamReceiver;
import com.vaadin.flow.server.StreamVariable;
import com.vaadin.flow.shared.Registration;
import elemental.json.JsonNull;
import elemental.json.JsonObject;
/**
* Server-side component for the {@code vaadin-upload} element.
*
* @author Vaadin Ltd.
*/
public class Upload extends GeneratedVaadinUpload implements HasSize {
private StreamVariable streamVariable;
private boolean interrupted = false;
private int activeUploads = 0;
private static final String I18N_PROPERTY = "i18n";
private UploadI18N i18n;
/**
* The output of the upload is redirected to this receiver.
*/
private Receiver receiver;
/**
* Create a new instance of Upload.
*
* The receiver must be set before performing an upload.
*/
public Upload() {
// Get a server round trip for upload error and success.
addUploadErrorListener(event -> {
});
addUploadSuccessListener(event -> {
});
// If client aborts upload mark upload as interrupted on server also
addUploadAbortListener(event -> interruptUpload());
getElement().setAttribute("target", new StreamReceiver(
getElement().getNode(), "upload", getStreamVariable()));
}
/**
* Create a new instance of Upload with the given receiver.
*
* @param receiver
* receiver that handles the upload
*/
public Upload(Receiver receiver) {
this();
setReceiver(receiver);
}
private StreamVariable getStreamVariable() {
if (streamVariable == null) {
streamVariable = new DefaultStreamVariable(this);
}
return streamVariable;
}
/**
* Limit of files to upload, by default it is unlimited. If the value is set
* to one, the native file browser will prevent selecting multiple files.
*
* @param maxFiles
* the maximum number of files allowed for the user to select
*/
public void setMaxFiles(int maxFiles) {
super.setMaxFiles(maxFiles);
}
/**
* Get the maximum number of files allowed for the user to select to
* upload.
*
* @return the maximum number of files
*/
public int getMaxFiles() {
return (int) getMaxFilesDouble();
}
/**
* Specify the maximum file size in bytes allowed to upload. Notice that
* it is a client-side constraint, which will be checked before sending the
* request.
*
* @param maxFileSize
* the maximum file size in bytes
*/
public void setMaxFileSize(int maxFileSize) {
super.setMaxFileSize(maxFileSize);
}
/**
* Get the maximum allowed file size in the client-side, in bytes.
*
* @return the maximum file size in bytes
*/
public int getMaxFileSize() {
return (int) getMaxFileSizeDouble();
}
/**
* When false
, it prevents uploads from triggering immediately
* upon adding file(s). The default is true
.
*
* @param autoUpload
* true
to allow uploads to start immediately after
* selecting files, false
otherwise.
*/
public void setAutoUpload(boolean autoUpload) {
setNoAuto(!autoUpload);
}
/**
* Get the auto upload status.
*
* @return true
if the upload of files should start immediately
* after they are selected, false
otherwise.
*/
public boolean isAutoUpload() {
return isNoAutoBoolean();
}
/**
* Define whether the element supports dropping files on it for uploading.
* By default it's enabled in desktop and disabled in touch devices because
* mobile devices do not support drag events in general. Setting it
* true
means that drop is enabled even in touch-devices, and
* false
disables drop in all devices.
*
* @param dropAllowed
* true
to allow file dropping, false
* otherwise
*/
public void setDropAllowed(boolean dropAllowed) {
setNodrop(!dropAllowed);
}
/**
* Get whether file dropping is allowed or not. By default it's enabled in
* desktop and disabled in touch devices because mobile devices do not
* support drag events in general.
*
* @return true
if file dropping is allowed, false
* otherwise.
*/
public boolean isDropAllowed() {
return !isNodropBoolean();
}
/**
* Specify the types of files that the server accepts. Syntax: a MIME type
* pattern (wildcards are allowed) or file extensions. Notice that MIME
* types are widely supported, while file extensions are only implemented in
* certain browsers, so it should be avoided.
*
* Example: "video/*","image/tiff"
or
* ".pdf","audio/mp3"
*
* @param acceptedFileTypes
* the allowed file types to be uploaded, or null
to
* clear any restrictions
*/
public void setAcceptedFileTypes(String... acceptedFileTypes) {
String accepted = "";
if (acceptedFileTypes != null) {
accepted = String.join(",", acceptedFileTypes);
}
setAccept(accepted);
}
/**
* Get the list of accepted file types for upload.
*
* @return a list of allowed file types, never null
.
*/
public List getAcceptedFileTypes() {
String accepted = getAcceptString();
if (accepted == null) {
return Collections.emptyList();
}
return Arrays.asList(accepted.split(","));
}
/**
* Set the component as the actionable button inside the upload component,
* that starts the upload of the selected files.
*
* @param uploadButton
* the component to be clicked by the user to start the upload,
* or null
to clear it
*/
public void setUploadButton(Component uploadButton) {
removeElementsAtSlot("add-button");
if (uploadButton != null) {
addToAddButton(uploadButton);
}
}
/**
* Get the component set as the upload button for the upload, if any.
*
* @return the actionable button, or null
if none was set
*/
public Component getUploadButton() {
return getComponentAtSlot("add-button");
}
/**
* Set the component to show as a message to the user to drop files in the
* upload component. Despite of the name, the label can be any component.
*
* @param dropLabel
* the label to show for the users when it's possible drop files,
* or null
to clear it
*/
public void setDropLabel(Component dropLabel) {
removeElementsAtSlot("drop-label");
if (dropLabel != null) {
addToDropLabel(dropLabel);
}
}
/**
* Get the component set as the drop label, if any.
*
* @return the drop label component, or null
if none was set
*/
public Component getDropLabel() {
return getComponentAtSlot("drop-label");
}
/**
* Set the component to show as the drop label icon. The icon is visible
* when the user can drop files to this upload component. Despite of the
* name, the drop label icon can be any component.
*
* @param dropLabelIcon
* the label icon to show for the users when it's possible to
* drop files, or null
to cleat it
*/
public void setDropLabelIcon(Component dropLabelIcon) {
removeElementsAtSlot("drop-label-icon");
if (dropLabelIcon != null) {
addToDropLabelIcon(dropLabelIcon);
}
}
/**
* Get the component set as the drop label icon, if any.
*
* @return the drop label icon component, or null
if none was
* set
*/
public Component getDropLabelIcon() {
return getComponentAtSlot("drop-label-icon");
}
private void removeElementsAtSlot(String slot) {
getElement().getChildren()
.filter(child -> slot.equals(child.getAttribute("slot")))
.forEach(Element::removeFromParent);
}
private Component getComponentAtSlot(String slot) {
return getElement().getChildren()
.filter(child -> slot.equals(child.getAttribute("slot")))
.filter(child -> child.getComponent().isPresent())
.map(child -> child.getComponent().get()).findFirst()
.orElse(null);
}
/**
* Go into upload state. This is to prevent uploading more files than
* accepted on same component.
*/
private void startUpload() {
if (getMaxFiles() != 0 && getMaxFiles() <= activeUploads) {
throw new IllegalStateException(
"Maximum supported amount of uploads already started");
}
activeUploads++;
}
/**
* Interrupt the upload currently being received.
*
* The interruption will be done by the receiving thread so this method
* will return immediately and the actual interrupt will happen a bit later.
*
* Note! this will interrupt all uploads in multi-upload mode.
*/
public void interruptUpload() {
if (isUploading()) {
interrupted = true;
}
}
private void endUpload() {
activeUploads--;
interrupted = false;
}
/**
* Is upload in progress.
*
* @return true if receiving upload
*/
public boolean isUploading() {
return activeUploads > 0;
}
private void fireStarted(String filename, String MIMEType,
long contentLength) {
fireEvent(new StartedEvent(this, filename, MIMEType, contentLength));
}
private void fireUploadInterrupted(String filename, String MIMEType,
long length) {
fireEvent(new FailedEvent(this, filename, MIMEType, length));
}
private void fireNoInputStream(String filename, String MIMEType,
long length) {
fireEvent(new NoInputStreamEvent(this, filename, MIMEType, length));
}
private void fireNoOutputStream(String filename, String MIMEType,
long length) {
fireEvent(new NoOutputStreamEvent(this, filename, MIMEType, length));
}
private void fireUploadInterrupted(String filename, String MIMEType,
long length, Exception e) {
fireEvent(new FailedEvent(this, filename, MIMEType, length, e));
}
private void fireUploadSuccess(String filename, String MIMEType,
long length) {
fireEvent(new SucceededEvent(this, filename, MIMEType, length));
}
private void fireUploadFinish(String filename, String MIMEType,
long length) {
fireEvent(new FinishedEvent(this, filename, MIMEType, length));
}
/**
* Emit the progress event.
*
* @param totalBytes
* bytes received so far
* @param contentLength
* actual size of the file being uploaded, if known
*/
protected void fireUpdateProgress(long totalBytes, long contentLength) {
fireEvent(new ProgressUpdateEvent(this, totalBytes, contentLength));
}
/**
* Add a progress listener that is informed on upload progress.
*
* @param listener
* progress listener to add
* @return registration for removal of listener
*/
public Registration addProgressListener(
ComponentEventListener listener) {
return addListener(ProgressUpdateEvent.class, listener);
}
/**
* Add a succeeded listener that is informed on upload failure.
*
* @param listener
* failed listener to add
* @return registration for removal of listener
*/
public Registration addFailedListener(
ComponentEventListener listener) {
return addListener(FailedEvent.class, listener);
}
/**
* Add a succeeded listener that is informed on upload finished.
*
* @param listener
* finished listener to add
* @return registration for removal of listener
*/
public Registration addFinishedListener(
ComponentEventListener listener) {
return addListener(FinishedEvent.class, listener);
}
/**
* Add a succeeded listener that is informed on upload start.
*
* @param listener
* start listener to add
* @return registration for removal of listener
*/
public Registration addStartedListener(
ComponentEventListener listener) {
return addListener(StartedEvent.class, listener);
}
/**
* Add a succeeded listener that is informed on upload succeeded.
*
* @param listener
* succeeded listener to add
* @return registration for removal of listener
*/
public Registration addSucceededListener(
ComponentEventListener listener) {
return addListener(SucceededEvent.class, listener);
}
/**
* Return the current receiver.
*
* @return the StreamVariable.
*/
public Receiver getReceiver() {
return receiver;
}
/**
* Set the receiver implementation that should be used for this upload
* component.
*
* Note! If the receiver doesn't implement {@link MultiFileReceiver} then
* the upload will be automatically set to only accept one file.
*
* @param receiver
* receiver to use for file reception
*/
public void setReceiver(Receiver receiver) {
this.receiver = receiver;
if (!(receiver instanceof MultiFileReceiver)) {
setMaxFiles(1);
} else {
getElement().removeAttribute("maxFiles");
}
}
/**
* Set the internationalization properties for this component.
*
* @param i18n
* the internationalized properties, not null
*/
public void setI18n(UploadI18N i18n) {
Objects.requireNonNull(i18n,
"The I18N properties object should not be null");
this.i18n = i18n;
runBeforeClientResponse(ui -> {
if (i18n == this.i18n) {
JsonObject i18nObject = (JsonObject) JsonSerializer
.toJson(this.i18n);
for (String key : i18nObject.keys()) {
ui.getPage().executeJavaScript(
"$0.set('i18n." + key + "', $1)", getElement(),
i18nObject.get(key));
}
}
});
}
void runBeforeClientResponse(SerializableConsumer command) {
getElement().getNode().runWhenAttached(ui -> ui
.beforeClientResponse(this, context -> command.accept(ui)));
}
/**
* Get the internationalization object previously set for this component.
*
* Note: updating the object content that is gotten from this method will
* not update the language on the component if not set back using
* {@link Upload#setI18n(UploadI18N)}
*
* @return the object with the i18n properties. If the i18n properties
* weren't set, the object will return null
.
*/
public UploadI18N getI18n() {
return i18n;
}
private String getStringObject(String propertyName, String subName) {
String result = null;
JsonObject json = (JsonObject) getElement()
.getPropertyRaw(propertyName);
if (json != null && json.hasKey(subName)
&& !(json.get(subName) instanceof JsonNull)) {
result = json.getString(subName);
}
return result;
}
private String getStringObject(String propertyName, String object,
String subName) {
String result = null;
JsonObject json = (JsonObject) getElement()
.getPropertyRaw(propertyName);
if (json != null && json.hasKey(object)
&& !(json.get(object) instanceof JsonNull)) {
json = json.getObject(object);
if (json != null && json.hasKey(subName)
&& !(json.get(subName) instanceof JsonNull)) {
result = json.getString(subName);
}
}
return result;
}
private static class DefaultStreamVariable implements StreamVariable {
private Deque lastStartedEvent = new ArrayDeque<>();
private final Upload upload;
public DefaultStreamVariable(Upload upload) {
this.upload = upload;
}
@Override
public boolean listenProgress() {
return upload.getEventBus().hasListener(ProgressUpdateEvent.class);
}
@Override
public void onProgress(StreamVariable.StreamingProgressEvent event) {
upload.fireUpdateProgress(event.getBytesReceived(),
event.getContentLength());
}
@Override
public boolean isInterrupted() {
return upload.interrupted;
}
@Override
public OutputStream getOutputStream() {
if (upload.getReceiver() == null) {
throw new IllegalStateException(
"Upload cannot be performed without a receiver set");
}
StreamVariable.StreamingStartEvent event = lastStartedEvent.pop();
OutputStream receiveUpload = upload.getReceiver()
.receiveUpload(event.getFileName(), event.getMimeType());
return receiveUpload;
}
@Override
public void streamingStarted(StreamVariable.StreamingStartEvent event) {
upload.startUpload();
try {
upload.fireStarted(event.getFileName(), event.getMimeType(),
event.getContentLength());
} finally {
lastStartedEvent.addLast(event);
}
}
@Override
public void streamingFinished(StreamVariable.StreamingEndEvent event) {
try {
upload.fireUploadSuccess(event.getFileName(),
event.getMimeType(), event.getContentLength());
} finally {
upload.endUpload();
upload.fireUploadFinish(event.getFileName(), event.getMimeType(), event.getContentLength());
}
}
@Override
public void streamingFailed(StreamVariable.StreamingErrorEvent event) {
try {
Exception exception = event.getException();
if (exception instanceof NoInputStreamException) {
upload.fireNoInputStream(event.getFileName(),
event.getMimeType(), 0);
} else if (exception instanceof NoOutputStreamException) {
upload.fireNoOutputStream(event.getFileName(),
event.getMimeType(), 0);
} else {
upload.fireUploadInterrupted(event.getFileName(),
event.getMimeType(), event.getBytesReceived(),
exception);
}
} finally {
upload.endUpload();
upload.fireUploadFinish(event.getFileName(), event.getMimeType(), event.getContentLength());
}
}
}
}