Використання методу jQuery ajax для отримання зображень у вигляді BLOB-об'єкта


84

Нещодавно я задав інше (пов’язане) запитання, яке призвело до наступного запитання: Надання даних замість файлу для форми введення

Читаючи документацію jQuery.ajax () ( http://api.jquery.com/jQuery.ajax/ ), здається, список прийнятих типів даних не включає зображення.

Я намагаюся отримати зображення за допомогою jQuery.get (або jQuery.ajax, якщо мені потрібно), зберігаю це зображення в Blob і завантажую його на інший сервер у запиті POST. На даний момент, схоже, через невідповідність типів даних мої зображення в кінцевому підсумку пошкоджені (невідповідність розміру в байтах тощо).

Код, щоб виконати це, виглядає наступним чином (він написаний coffeescript, але синтаксичний аналіз не повинен бути складним):

handler = (data,status) ->
  fd = new FormData
  fd.append("file", new Blob([data], { "type" : "image/png" }))
  jQuery.ajax {
    url: target_url,
    data: fd,
    processData: false,
    contentType: "multipart/form-data",
    type: "POST",
    complete: (xhr,status) ->
      console.log xhr.status
      console.log xhr.statusCode
      console.log xhr.responseText

  }
jQuery.get(image_source_url, null, handler)

Як я можу замість цього отримати це зображення як крапку?


Я думаю, вам доведеться змінити тип відповіді на стороні сервера.
Ерік Фрік,

Я намагаюся витягнути зображення з будь-якої URL-адреси, а не з сервера, яким я володію.
jabalsad


Здається, що три рішення у цій відповіді полягають у тому, щоб (1) використовувати тег <img>, (2) змусити сервер подавати зображення у вигляді кодованого байта64 або (3) використовувати кеш браузера. (2) виключається, оскільки я хочу, щоб сценарій працював на будь-якій графічній графіці. Я не впевнений, як використовувати (1) або (3), оскільки після завантаження зображення мені потрібно перетворити його на Blob.
jabalsad

варіант 3 буде працювати лише в тому випадку, якщо ви вже завантажили зображення. Вперше вам знадобиться щось інше. Може вибрати 1?
Ерік Фрік,

Відповіді:


146

Ви не можете зробити це за допомогою jQuery ajax, але за допомогою власного XMLHttpRequest.

var xhr = new XMLHttpRequest();
xhr.onreadystatechange = function(){
    if (this.readyState == 4 && this.status == 200){
        //this.response is what you're looking for
        handler(this.response);
        console.log(this.response, typeof this.response);
        var img = document.getElementById('img');
        var url = window.URL || window.webkitURL;
        img.src = url.createObjectURL(this.response);
    }
}
xhr.open('GET', 'http://jsfiddle.net/img/logo.png');
xhr.responseType = 'blob';
xhr.send();      

РЕДАГУВАТИ

Тож переглядаючи цю тему, здається, це дійсно можливо зробити за допомогою jQuery 3

jQuery.ajax({
        url:'https://images.unsplash.com/photo-1465101108990-e5eac17cf76d?ixlib=rb-0.3.5&q=85&fm=jpg&crop=entropy&cs=srgb&ixid=eyJhcHBfaWQiOjE0NTg5fQ%3D%3D&s=471ae675a6140db97fea32b55781479e',
        cache:false,
        xhr:function(){// Seems like the only way to get access to the xhr object
            var xhr = new XMLHttpRequest();
            xhr.responseType= 'blob'
            return xhr;
        },
        success: function(data){
            var img = document.getElementById('img');
            var url = window.URL || window.webkitURL;
            img.src = url.createObjectURL(data);
        },
        error:function(){
            
        }
    });
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.0.0/jquery.min.js"></script>
<img id="img" width=100%>

або

використовуйте xhrFields для встановлення responseType

    jQuery.ajax({
            url:'https://images.unsplash.com/photo-1465101108990-e5eac17cf76d?ixlib=rb-0.3.5&q=85&fm=jpg&crop=entropy&cs=srgb&ixid=eyJhcHBfaWQiOjE0NTg5fQ%3D%3D&s=471ae675a6140db97fea32b55781479e',
            cache:false,
            xhrFields:{
                responseType: 'blob'
            },
            success: function(data){
                var img = document.getElementById('img');
                var url = window.URL || window.webkitURL;
                img.src = url.createObjectURL(data);
            },
            error:function(){
                
            }
        });
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.0.0/jquery.min.js"></script>
    <img id="img" width=100%>


Дякую. Я просто зрозумів це і побачив вашу відповідь. Це схоже на моє (крім того, що я публікую його у формі, а не встановлюю на об’єкті <img>). Я все одно
позначу

@jabalsad поставлене запитання How can I retrieve this image as a blob instead?, у будь-якому випадку, яке мало бути лише для демонстрації, handlerвізьме this.responseі додасть його до об’єкта datadata form і надішле через ajax.
Муса

2
+1, чудово працює! І FYI, це, можливо, нарешті скоро буде додано до jQuery: github.com/jquery/jquery/pull/1525
lambshaanxy

9
2017: jQuery все ще не може мати справу з типом `` BLOB ''?
robsch

1
'xhrFields' також працює в jQuery 2 (протестовано в 2.2.4)
Bampfer

15

Якщо вам потрібно обробляти повідомлення про помилки за допомогою jQuery.AJAX, вам потрібно буде змінити xhrфункцію, щоб responseTypeвона не змінювалася, коли трапляється помилка.

Отже, вам доведеться змінити responseType" BLOB " лише у випадку успішного виклику:

$.ajax({
    ...
    xhr: function() {
        var xhr = new XMLHttpRequest();
        xhr.onreadystatechange = function() {
            if (xhr.readyState == 2) {
                if (xhr.status == 200) {
                    xhr.responseType = "blob";
                } else {
                    xhr.responseType = "text";
                }
            }
        };
        return xhr;
    },
    ...
    error: function(xhr, textStatus, errorThrown) {
        // Here you are able now to access to the property "responseText"
        // as you have the type set to "text" instead of "blob".
        console.error(xhr.responseText);
    },
    success: function(data) {
        console.log(data); // Here is "blob" type
    }
});

Примітка

Якщо ви налагоджуєте та розміщуєте точку зупинки в точці відразу після встановлення значення xhr.responseType" краплина ", ви можете зазначити, що при спробі отримати значення responseTextви отримаєте таке повідомлення:

Значення доступне лише в тому випадку, якщо 'responseType' об'єкта є '' або 'text' (було 'blob').


1
Дуже дякую! Це було рішення, яке я шукав. Мені потрібен був AJAX, щоб обробляти успішну відповідь як "BLOB" (у моєму випадку ZIP-архів) у .done()методі, і якщо щось пішло не так, відповідь у .fail()методі повинна оброблятися як "текст", оскільки в іншому випадку значення responseTextбуло порожнім тощо. рішення для мене прекрасно працювало!
informatik01

4

Велике спасибі @Musa, і ось чудова функція, яка перетворює дані в рядок base64. Це може стати вам у пригоді при обробці файлу двійкового файлу (pdf, png, jpeg, docx, ...) у WebView, який отримує двійковий файл, але вам потрібно безпечно перенести дані файлу у ваш додаток.

// runs a get/post on url with post variables, where:
// url ... your url
// post ... {'key1':'value1', 'key2':'value2', ...}
//          set to null if you need a GET instead of POST req
// done ... function(t) called when request returns
function getFile(url, post, done)
{
   var postEnc, method;
   if (post == null)
   {
      postEnc = '';
      method = 'GET';
   }
   else
   {
      method = 'POST';
      postEnc = new FormData();
      for(var i in post)
         postEnc.append(i, post[i]);
   }
   var xhr = new XMLHttpRequest();
   xhr.onreadystatechange = function() {
      if (this.readyState == 4 && this.status == 200)
      {
         var res = this.response;
         var reader = new window.FileReader();
         reader.readAsDataURL(res); 
         reader.onloadend = function() { done(reader.result.split('base64,')[1]); }
      }
   }
   xhr.open(method, url);
   xhr.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
   xhr.send('fname=Henry&lname=Ford');
   xhr.responseType = 'blob';
   xhr.send(postEnc);
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.