The first solution that you may implement and probably the only idea that will occur to everybody, when you need to upload a Base64 image into the server, is to upload this file as a string and then convert it to a file in the server side. This implementation works like a charm, however there's another implementation that you may want to know if you don't want to make that your server does all the hard job for the user.
In this article, you are going to learn how to convert a Base64 string into a Blob to upload it as a file to our server. This approach will be equivalent to the action that an user does when he drags and drop a file into a file input.
Implementation
To get started, we need to convert a base64 string into a "file" using Javascript, to do that, we are going to convert a Base64 string to a Blob and that will be interpreted as a File in our server.
We are going to use the following method to convert a base64 string into a blob:
/**
* Convert a base64 string in a Blob according to the data and contentType.
*
* @param b64Data {String} Pure base64 string without contentType
* @param contentType {String} the content type of the file i.e (image/jpeg - image/png - text/plain)
* @param sliceSize {Int} SliceSize to process the byteCharacters
* @see http://stackoverflow.com/questions/16245767/creating-a-blob-from-a-base64-string-in-javascript
* @return Blob
*/
function b64toBlob(b64Data, contentType, sliceSize) {
contentType = contentType || '';
sliceSize = sliceSize || 512;
var byteCharacters = atob(b64Data);
var byteArrays = [];
for (var offset = 0; offset < byteCharacters.length; offset += sliceSize) {
var slice = byteCharacters.slice(offset, offset + sliceSize);
var byteNumbers = new Array(slice.length);
for (var i = 0; i < slice.length; i++) {
byteNumbers[i] = slice.charCodeAt(i);
}
var byteArray = new Uint8Array(byteNumbers);
byteArrays.push(byteArray);
}
var blob = new Blob(byteArrays, {type: contentType});
return blob;
}
Now, as you simply can't attach a blob to a visible file input, we need to make it with an asynchronous form using a Post request. We are going to use the following form:
<form id="myAwesomeForm" method="post" action="/php-code-that-handles-fileupload.php">
<input type="text" id="filename" name="filename" /> <!-- Filename -->
<input type="submit" id="submitButton" name="submitButton" /> <!-- Submit -->
</form>
The submit of the previous form, would only send the filename text parameter but not any file. Use Javascript to retrieve the form and pass it as first parameter to a new instance of FormData
, that will include the text input to our asynchronous form, besides, process your Base64 string to fit the format required by the b64toblob function (first parameter only the base64 string without content type and as second parameter only the datatype):
// Get the form element withot jQuery
var form = document.getElementById("myAwesomeForm");
var ImageURL = "";
// Split the base64 string in data and contentType
var block = ImageURL.split(";");
// Get the content type of the image
var contentType = block[0].split(":")[1];// In this case "image/gif"
// get the real base64 content of the file
var realData = block[1].split(",")[1];// In this case "R0lGODlhPQBEAPeoAJosM...."
// Convert it to a blob to upload
var blob = b64toBlob(realData, contentType);
// Create a FormData and append the file with "image" as parameter name
var formDataToUpload = new FormData(form);
formDataToUpload.append("image", blob);
Now that the FormData
has a file you only need to submit it asynchronously using jQuery:
/**
* The following code should send 2 post parameters:
* filename: providen by the text input
* image: a file, dinamically added from a base64 string using javascript
*
* Is up to you how to receive the file in the Server side.
*/
$.ajax({
url:"/php-code-that-handles-fileupload.php",
data: formDataToUpload,// Add as Data the Previously create formData
type:"POST",
contentType:false,
processData:false,
cache:false,
dataType:"json", // Change this according to your response from the server.
error:function(err){
console.error(err);
},
success:function(data){
console.log(data);
},
complete:function(){
console.log("Request finished.");
}
});
Is up to you how to handle the previous code, you can prevent the submit event of the form and then execute the code as shown in the following example:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Base64 string to a file in form</title>
</head>
<body>
<form id="myAwesomeForm" method="post" action="endpoint.php">
<input type="text" id="filename" name="filename" /> <!-- Filename -->
<input type="submit" id="submitButton" name="submitButton" /> <!-- Submit -->
</form>
<script>
function b64toBlob(b64Data, contentType, sliceSize) {
contentType = contentType || '';
sliceSize = sliceSize || 512;
var byteCharacters = atob(b64Data);
var byteArrays = [];
for (var offset = 0; offset < byteCharacters.length; offset += sliceSize) {
var slice = byteCharacters.slice(offset, offset + sliceSize);
var byteNumbers = new Array(slice.length);
for (var i = 0; i < slice.length; i++) {
byteNumbers[i] = slice.charCodeAt(i);
}
var byteArray = new Uint8Array(byteNumbers);
byteArrays.push(byteArray);
}
var blob = new Blob(byteArrays, {type: contentType});
return blob;
}
$("#myAwesomeForm").submit(function(e){
e.preventDefault();
appendFileAndSubmit();
});
function appendFileAndSubmit(){
// Get the form
var form = document.getElementById("myAwesomeForm");
var ImageURL = "";
// Split the base64 string in data and contentType
var block = ImageURL.split(";");
// Get the content type
var contentType = block[0].split(":")[1];// In this case "image/gif"
// get the real base64 content of the file
var realData = block[1].split(",")[1];// In this case "iVBORw0KGg...."
// Convert to blob
var blob = b64toBlob(realData, contentType);
// Create a FormData and append the file
var fd = new FormData(form);
fd.append("image", blob);
// Submit Form and upload file
$.ajax({
url:"endpoint.php",
data: fd,// the formData function is available in almost all new browsers.
type:"POST",
contentType:false,
processData:false,
cache:false,
dataType:"json", // Change this according to your response from the server.
error:function(err){
console.error(err);
},
success:function(data){
console.log(data);
},
complete:function(){
console.log("Request finished.");
}
});
}
</script>
</body>
</html>
Retrieve the file in the server side
Is up to you how to receive the file in the server side, as we don't know if you're using PHP, a framework, django or other languages, however, we provide 2 examples with PHP that could be useful to understand what we've just do.
Symfony 2
<?php
// Important to use these statemenets, the json response is optional for our response.
use Symfony\Component\HttpFoundation\Request;
public function retrieveAction(Request $request){
// retrieve the file with the name given in the form.
// do var_dump($request->files->all()); if you need to know if the file is being uploaded.
$file = $request->files->get('image');
$nameInTextInput = $request->request->get("filename");
}
Plain PHP
// Do something with the gif image
$file = $_FILES['image']['tmp_name'];
Happy coding !