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

com.vaadin.flow.component.upload.Upload Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2000-2024 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 java.util.stream.IntStream;

import com.vaadin.flow.component.AttachEvent;
import com.vaadin.flow.component.Component;
import com.vaadin.flow.component.ComponentEventListener;
import com.vaadin.flow.component.HasSize;
import com.vaadin.flow.component.HasStyle;
import com.vaadin.flow.component.Tag;
import com.vaadin.flow.component.UI;
import com.vaadin.flow.component.button.Button;
import com.vaadin.flow.component.dependency.JsModule;
import com.vaadin.flow.component.dependency.NpmPackage;
import com.vaadin.flow.component.html.Span;
import com.vaadin.flow.component.shared.SlotUtils;
import com.vaadin.flow.dom.DomEventListener;
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.Json;
import elemental.json.JsonArray;
import elemental.json.JsonNull;
import elemental.json.JsonObject;
import elemental.json.JsonType;

/**
 * Upload is a component for uploading one or more files. It shows the upload
 * progression and status of each file. Files can be uploaded using the Upload
 * button or via drag and drop.
 *
 * @author Vaadin Ltd.
 */
@Tag("vaadin-upload")
@NpmPackage(value = "@vaadin/polymer-legacy-adapter", version = "24.5.3")
@JsModule("@vaadin/polymer-legacy-adapter/style-modules.js")
@NpmPackage(value = "@vaadin/upload", version = "24.5.3")
@JsModule("@vaadin/upload/src/vaadin-upload.js")
public class Upload extends Component implements HasSize, HasStyle {

    /**
     * Server-side component for the default {@code } icon.
     */
    @Tag("vaadin-upload-icon")
    static class UploadIcon extends Component {

        public UploadIcon() {
        }
    }

    private StreamVariable streamVariable;
    private boolean interrupted = false;

    private int activeUploads = 0;
    private boolean uploading;

    private UploadI18N i18n;

    private Component uploadButton;
    private Component defaultUploadButton;

    private Component dropLabel;
    private Component defaultDropLabel;

    private Component dropLabelIcon;
    private Component defaultDropLabelIcon;

    /**
     * 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() { final String eventDetailError = "event.detail.error"; final String eventDetailFileName = "event.detail.file.name"; getElement().addEventListener("file-reject", event -> { String detailError = event.getEventData() .getString(eventDetailError); String detailFileName = event.getEventData() .getString(eventDetailFileName); fireEvent(new FileRejectedEvent(this, detailError, detailFileName)); }).addEventData(eventDetailError).addEventData(eventDetailFileName); getElement().addEventListener("file-remove", event -> { String detailFileName = event.getEventData() .getString(eventDetailFileName); fireEvent(new FileRemovedEvent(this, detailFileName)); }).addEventData(eventDetailFileName); // If client aborts upload mark upload as interrupted on server also getElement().addEventListener("upload-abort", event -> interruptUpload()); runBeforeClientResponse(ui -> getElement().setAttribute("target", new StreamReceiver(getElement().getNode(), "upload", getStreamVariable()))); final String elementFiles = "element.files"; DomEventListener allFinishedListener = e -> { JsonArray files = e.getEventData().getArray(elementFiles); boolean isUploading = IntStream.range(0, files.length()) .anyMatch(index -> { final String KEY = "uploading"; JsonObject object = files.getObject(index); return object.hasKey(KEY) && object.getBoolean(KEY); }); if (this.uploading && !isUploading) { this.fireAllFinish(); } this.uploading = isUploading; }; getElement().addEventListener("upload-start", e -> this.uploading = true); getElement().addEventListener("upload-success", allFinishedListener) .addEventData(elementFiles); getElement().addEventListener("upload-error", allFinishedListener) .addEventData(elementFiles); getElement().addEventListener("upload-abort", allFinishedListener) .addEventData(elementFiles); defaultUploadButton = new Button(); // Ensure the flag is set before the element is added to the slot defaultUploadButton.getElement().setProperty("_isDefault", true); setUploadButton(defaultUploadButton); defaultDropLabel = new Span(); // Ensure the flag is set before the element is added to the slot defaultDropLabel.getElement().setProperty("_isDefault", true); setDropLabel(defaultDropLabel); defaultDropLabelIcon = new UploadIcon(); setDropLabelIcon(defaultDropLabelIcon); } /** * Create a new instance of Upload with the given receiver. * * @param receiver * receiver that handles the upload */ public Upload(Receiver receiver) { this(); setReceiver(receiver); } /** * Add listener that is informed on all uploads finished. * * @param listener * all finished listener to add * @return a {@link Registration} for removing the event listener */ public Registration addAllFinishedListener( ComponentEventListener listener) { return addListener(AllFinishedEvent.class, listener); } 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) { getElement().setProperty("maxFiles", maxFiles); getElement().executeJs("this.maxFiles = $0", 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) getElement().getProperty("maxFiles", 0.0); } private void removeMaxFiles() { getElement().removeProperty("maxFiles"); getElement().executeJs("this.maxFiles = Infinity"); } /** * 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) { getElement().setProperty("maxFileSize", 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) getElement().getProperty("maxFileSize", 0.0); } /** * 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) { getElement().setProperty("noAuto", !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 !getElement().getProperty("noAuto", false); } /** * 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) { getElement().setProperty("nodrop", !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 !getElement().getProperty("nodrop", false); } /** * Specify the types of files that the Upload web-component 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" *

* File format restrictions are checked only on the client side (browser). * They indicate the hints for users as to what file types to upload. Using * this method won’t restrict the uploaded file’s format on the server side. * If required, it’s the responsibility of the application developer to * implement application-specific restrictions on the server side in one or * more of the Upload component’s event listeners (e.g., in * {@link #addSucceededListener}). * * @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); } getElement().setProperty("accept", accepted); } /** * Get the list of accepted file types for upload. * * @return a list of allowed file types, never null. */ public List getAcceptedFileTypes() { String accepted = getElement().getProperty("accept"); if (accepted == null) { return Collections.emptyList(); } return Arrays.asList(accepted.split(",")); } /** * Set the component as the actionable button inside the upload component, * that opens the dialog for choosing the files to be upload. * * @param button * the component to be clicked by the user to open the dialog, or * null to reset to the default button */ public void setUploadButton(Component button) { if (button != null) { uploadButton = button; } else { uploadButton = defaultUploadButton; } SlotUtils.setSlot(this, "add-button", uploadButton); } /** * Get the component set as the upload button for the upload. * * @return the actionable button, never null. */ public Component getUploadButton() { return uploadButton; } /** * 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 label * the label to show for the users when it's possible drop files, * or null to reset to the default label */ public void setDropLabel(Component label) { if (label != null) { dropLabel = label; } else { dropLabel = defaultDropLabel; } SlotUtils.setSlot(this, "drop-label", dropLabel); } /** * Get the component set as the drop label. * * @return the drop label component, never null. */ public Component getDropLabel() { return dropLabel; } /** * 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 icon * the label icon to show for the users when it's possible to * drop files, or null to reset to the default icon */ public void setDropLabelIcon(Component icon) { if (icon != null) { dropLabelIcon = icon; } else { dropLabelIcon = defaultDropLabelIcon; } SlotUtils.setSlot(this, "drop-label-icon", dropLabelIcon); } /** * Get the component set as the drop label icon. * * @return the drop label icon component, never null. */ public Component getDropLabelIcon() { return dropLabelIcon; } /** * 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)); } private void fireAllFinish() { fireEvent(new AllFinishedEvent(this)); } /** * Emit the progress event. * * @param totalBytes * bytes received so far * @param contentLength * actual size of the file being uploaded, if known * * @deprecated since 24.4. Use * {@link #fireUpdateProgress(long, long, String)} */ @Deprecated(since = "24.4") protected void fireUpdateProgress(long totalBytes, long contentLength) { fireEvent( new ProgressUpdateEvent(this, totalBytes, contentLength, null)); } /** * Emit the progress event. * * @param totalBytes * bytes received so far * @param contentLength * actual size of the file being uploaded, if known * @param fileName * name of the file being uploaded */ protected void fireUpdateProgress(long totalBytes, long contentLength, String fileName) { fireEvent(new ProgressUpdateEvent(this, totalBytes, contentLength, fileName)); } /** * 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); } /** * Adds a listener for {@code file-reject} events fired when a file cannot * be added due to some constrains: * {@code setMaxFileSize, setMaxFiles, setAcceptedFileTypes} * * @param listener * the listener * @return a {@link Registration} for removing the event listener */ public Registration addFileRejectedListener( ComponentEventListener listener) { return addListener(FileRejectedEvent.class, listener); } /** * Adds a listener for events fired when a file is removed. * * @param listener * the listener * @return a {@link Registration} for removing the event listener */ public Registration addFileRemovedListener( ComponentEventListener listener) { return addListener(FileRemovedEvent.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) { Receiver oldReceiver = this.receiver; this.receiver = receiver; if (isMultiFileReceiver(receiver)) { if (oldReceiver != null && !isMultiFileReceiver(oldReceiver)) { removeMaxFiles(); } } else { setMaxFiles(1); } } private boolean isMultiFileReceiver(Receiver receiver) { return receiver instanceof MultiFileReceiver; } /** * Set the internationalization properties for this component. * * @param i18n * the i18n object, not {@code null} */ public void setI18n(UploadI18N i18n) { this.i18n = Objects.requireNonNull(i18n, "The i18n properties object should not be null"); runBeforeClientResponse(ui -> { if (i18n == this.i18n) { setI18nWithJS(); } }); } private void setI18nWithJS() { JsonObject i18nJson = (JsonObject) JsonSerializer.toJson(this.i18n); // Remove null values so that we don't overwrite existing WC // translations with empty ones deeplyRemoveNullValuesFromJsonObject(i18nJson); // Assign new I18N object to WC, by deeply merging the existing // WC I18N, and the values from the new UploadI18N instance, // into an empty object getElement().executeJs( "const dropFiles = Object.assign({}, this.i18n.dropFiles, $0.dropFiles);" + "const addFiles = Object.assign({}, this.i18n.addFiles, $0.addFiles);" + "const error = Object.assign({}, this.i18n.error, $0.error);" + "const uploadingStatus = Object.assign({}, this.i18n.uploading.status, $0.uploading && $0.uploading.status);" + "const uploadingRemainingTime = Object.assign({}, this.i18n.uploading.remainingTime, $0.uploading && $0.uploading.remainingTime);" + "const uploadingError = Object.assign({}, this.i18n.uploading.error, $0.uploading && $0.uploading.error);" + "const uploading = {status: uploadingStatus," + " remainingTime: uploadingRemainingTime," + " error: uploadingError};" + "const units = $0.units || this.i18n.units;" + "this.i18n = Object.assign({}, this.i18n, $0, {" + " addFiles: addFiles, dropFiles: dropFiles," + " uploading: uploading, units: units});", i18nJson); } @Override protected void onAttach(AttachEvent attachEvent) { super.onAttach(attachEvent); // Element state is not persisted across attach/detach if (this.i18n != null) { setI18nWithJS(); } } private void deeplyRemoveNullValuesFromJsonObject(JsonObject jsonObject) { for (String key : jsonObject.keys()) { if (jsonObject.get(key).getType() == JsonType.OBJECT) { deeplyRemoveNullValuesFromJsonObject(jsonObject.get(key)); } else if (jsonObject.get(key).getType() == JsonType.NULL) { jsonObject.remove(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 instance that is returned from this method will not * update the component if not set again using {@link #setI18n(UploadI18N)} * * @return the i18n object or {@code null} if no i18n object has been set */ 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; } /** * Clear the list of files being processed, or already uploaded. */ public void clearFileList() { getElement().setPropertyJson("files", Json.createArray()); } 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(), event.getFileName()); } @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. " + "Please firstly set the receiver implementation with upload.setReceiver"); } 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()); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy