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

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;

    /**
     * Creates 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); } /** * Gets 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(); } /** * Specifies 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); } /** * Gets 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); } /** * Gets 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); } /** * Gets 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(); } /** * Specifies 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); } /** * Gets 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(",")); } /** * Sets 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); } } /** * Gets 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"); } /** * Sets 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); } } /** * Gets 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"); } /** * Sets 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); } } /** * Gets 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++; } /** * Interrupts 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)); } /** * Emits 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); } /** * Returns 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"); } } /** * Sets 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))); } /** * Gets 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(); } } @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(); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy