<html lang="en">
<head>
<meta charset="UTF-8" />
<title>HTML5 File</title>
</head>
<body onload="retrieveValues();">
<p>This example sends files as soon as they have been selected. You cannot select more than once the same file. HTML5 form validation works, you can not sublit the form if there are invalid fields.</p>
<p>Notice that the JavaScript source code uses localStorage to save/restore form content as we type, and selected files as we select them. Reloading the page will restore the entered values. We saw this technique in the HTML5 part 1 course.</p>
<form id="formUpload" name="formUpload" method="post" action="upload.php" enctype="multipart/form-data">
<label>Given name: <input type="text" id="givenname" name="givenname" required/></label><br />
<label>Family name: <input type="text" id="familyname" name="familyname" required/></label><br />
Please select one or more pictures/files :<br/>
<input type="hidden" id="namesAllFiles" name="namesAllFiles" />
<input type="file" multiple name="formFiles" id="formFiles" onchange="uploadFileWithoutParameters()"/>
<input type="submit" id="submitForm" value="Send" onclick="sendForm(event)" disabled>
</form>
<div id="message"></div>
</body>
</html>
// object that holds form values
var formValues = {};
// function that iterates through the form and dynamically
// assigns properties/values to form values object,
// and store it to sessionStorage
storeValues = function () {
var formElements = document.formUpload.elements;
for (i = 0; i < formElements.length; i++) {
var formElement = formElements[i];
if (formElement.localName == "input" && formElement.type == "text") {
var name = formElements[i].name;
var value = formElements[i].value;
formValues[name] = value;
}
}
// Store the object as a JSON String
sessionStorage.setItem('formUploadSave', JSON.stringify(formValues));
};
// on loading, retrieve any form values previously stored in the session.
// called by body onload
// function that retrieves form values from sessionStorage, iterates through the form and restores input fields values; also assign onchange event listener.
retrieveValues = function () {
var formElements = document.formUpload.elements;
// Retrieve the object from storage
var retrievedValues = JSON.parse(sessionStorage.getItem("formUploadSave"));
// If found, populate the form
for (i = 0; i < formElements.length; i++) {
var formElement = formElements[i];
if (formElement.localName == "input" && formElement.type == "text") {
// Assign onchange event listener
formElement.addEventListener("change", storeValues, false);
var formElementName = formElement.name;
if (retrievedValues) {// != null
formElement.value = retrievedValues[formElementName];
}
}
}
// On refresh we restore only last uploaded files so we have to
// - reconstruct namesAllFiles concatenation
// - reconstruct files list with progress bar
// - enable send button if there's restored file(s)
document.getElementById("namesAllFiles").value = '';
uploadFileWithoutParameters();
if(document.getElementById('formFiles').files.length) {
document.getElementById("submitForm").disabled = false;
}
};
function uploadFileWithoutParameters() {
// to upload files in more than one drag'n'drop
// we have to count already uploaded files to set an unused
// id for the progress bar
if(document.getElementById('namesAllFiles').value === "") {
nbUploadedFiles = 0;
} else {
nbUploadedFiles = document.getElementById('namesAllFiles').value.split('::').length;
}
var inputFiles = document.getElementById('formFiles');
var files = inputFiles.files;
for(i=0; i<files.length; i++) {
var li = document.createElement('li');
li.textContent = files[i].name;
// create a progress bar for each file
var progressBar = document.createElement('progress');
progressBar.id = "progressBar"+(nbUploadedFiles+i);
progressBar.value = 0;
progressBar.max = 100;
li.appendChild(progressBar);
document.getElementById("message").appendChild(li);
uploadOneFile(files[i], (nbUploadedFiles+i), files.length);
// if there is more than on uploaded file or more than one file to upload
// we need to concatenate
if (nbUploadedFiles >= 1 || i >= 1) {
document.getElementById("namesAllFiles").value += "::" + files[i].name;
}
else {
document.getElementById("namesAllFiles").value += files[i].name;
}
}
}
var nbFilesUpload = 0;
function sendForm(event) {
// here we reinit the files form to no file to restore
document.getElementById("formFiles").value = null;
}
function uploadOneFile(file, indice, nbFiles) {
var xhr = new XMLHttpRequest();
xhr.open("POST", formUpload.getAttribute("action"), true);
// looks like PHP / Apache should automatically translate
// header X_FILENAME in HTTTP-X-FILENAME which is the correct header
// in some case they don't do that so let's write it a clean way
xhr.setRequestHeader("X-FILENAME", file.name);
var progressBar = document.getElementById("progressBar"+indice);
xhr.upload.onprogress = function(e) {
progressBar.value = e.loaded;
progressBar.max = e.total;
if(progressBar.value == progressBar.max) {
nbFilesUpload++;
if(nbFilesUpload == nbFiles) {
document.getElementById("submitForm").disabled = false;
}
}
};
// Send the Ajax request
xhr.send(file);
}
Output
You can jump to the latest bin by adding /latest
to your URL
Keyboard Shortcuts
Shortcut | Action |
---|---|
ctrl + [num] | Toggle nth panel |
ctrl + 0 | Close focused panel |
ctrl + enter | Re-render output. If console visible: run JS in console |
Ctrl + l | Clear the console |
ctrl + / | Toggle comment on selected lines |
ctrl + ] | Indents selected lines |
ctrl + [ | Unindents selected lines |
tab | Code complete & Emmet expand |
ctrl + shift + L | Beautify code in active panel |
ctrl + s | Save & lock current Bin from further changes |
ctrl + shift + s | Open the share options |
ctrl + y | Archive Bin |
Complete list of JS Bin shortcuts |
JS Bin URLs
URL | Action |
---|---|
/ | Show the full rendered output. This content will update in real time as it's updated from the /edit url. |
/edit | Edit the current bin |
/watch | Follow a Code Casting session |
/embed | Create an embeddable version of the bin |
/latest | Load the very latest bin (/latest goes in place of the revision) |
/[username]/last | View the last edited bin for this user |
/[username]/last/edit | Edit the last edited bin for this user |
/[username]/last/watch | Follow the Code Casting session for the latest bin for this user |
/quiet | Remove analytics and edit button from rendered output |
.js | Load only the JavaScript for a bin |
.css | Load only the CSS for a bin |
Except for username prefixed urls, the url may start with http://jsbin.com/abc and the url fragments can be added to the url to view it differently. |