Как преобразовать изображение base64 в файл изображения и загрузить его в асинхронной форме, используя jQuery

Первое решение, которое вы можете реализовать, и, вероятно, единственная идея, которая придет всем в голову, когда вам нужно загрузить изображение Base64 на сервер, это загрузить этот файл в виде строки, а затем преобразовать его в файл на стороне сервера. Эта реализация работает как прелесть, однако есть другая реализация, которую вы можете узнать, если вы не хотите, чтобы ваш сервер выполнял всю тяжелую работу для пользователя.

В этой статье вы узнаете, как преобразовать строку Base64 в BLOB-объект, чтобы загрузить ее в виде файла на наш сервер. Этот подход будет эквивалентен действию, которое пользователь выполняет, когда он перетаскивает файл во входной файл.

Реализация

Для начала нам нужно преобразовать строку base64 в «файл» с помощью Javascript, для этого мы собираемся преобразовать строку Base64 в BLOB-объект, который будет интерпретирован как файл на нашем сервере.

Мы собираемся использовать следующий метод для преобразования строки base64 в 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;
}

Теперь, поскольку вы просто не можете прикрепить большой двоичный объект к видимому входному файлу, нам нужно сделать это с асинхронной формой, используя запрос Post. Мы собираемся использовать следующую форму:


Передача предыдущей формы будет отправлять только текстовый параметр имени файла, но не какой-либо файл. Используйте Javascript для извлечения формы и передачи ее в качестве первого параметра новому экземпляру FormData, это будет включать в себя ввод текста в нашу асинхронную форму, кроме того, обработайте вашу строку Base64, чтобы соответствовать формату, требуемому функцией b64toblob (первый параметр - только строка base64 без типа содержимого и как второй параметр - только тип данных):

// 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);

Теперь, когда FormData имеет файл вам нужно только отправить его асинхронно с помощью 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.");
}
});

От вас зависит, как обрабатывать предыдущий код, вы можете предотвратить событие отправки формы, а затем выполнить код, как показано в следующем примере:


Base64 string to a file in form
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.");
}
});
}

Получить файл на стороне сервера

Вам решать, как получить файл на стороне сервера, так как мы не знаем, используете ли вы PHP, framework, django или другие языки, однако мы предоставляем 2 примера с PHP, которые могут быть полезны для понимания того, что мы только что сделали.

Symfony 2

files->all()); if you need to know if the file is being uploaded.
$file = $request->files->get('image');
$nameInTextInput = $request->request->get("filename");
}

Простой PHP

// Do something with the gif image
$file = $_FILES['image']['tmp_name'];
Ссылка на основную публикацию
Adblock
detector