Перетворення зображення в Base64


85
<input type="file" id="asd"/>

Я хотів би отримати зображення в base64, як тільки користувач вибрав це (перед подачею форми)

Щось на зразок :

$(input).on('change',function(){
  var data = $(this).val().base64file(); // it is not a plugin is just an example
  alert(data);
});

Я читав про File API та інші матеріали, я хотів би отримати просте рішення для крос-браузерів (IE6 / IE7, очевидно, виключено)

Будь-яка допомога вдячна спасибі.


1
І що ви не зрозуміли про API файлу HTML5? Що ви спробували? Що не вийшло?
epascarello

@epascarello, вони не повністю підтримуються, насправді caniuse.com/#feat=fileapi мені потрібно обійтись , особливо тому, що версії Android все ще використовуються (старі версії), а також для старих версій iOS, і я також хотів би залучити IE9, який все ще використовується багато: P
бомбастичний

обхідний шлях для чого? Що ви намагаєтесь зробити з файлом? base64file()- це плагін?
Девід Хеллсінг,

@David, я просто хотів би отримати файл base64, як тільки користувач вибере файл зі свого ПК, base64file () - лише приклад
бомбастичний

1
@epascarello так, і це було саме моє питання, як підтримати всі браузери: D
бомбастичний

Відповіді:


210

function readFile() {
  
  if (this.files && this.files[0]) {
    
    var FR= new FileReader();
    
    FR.addEventListener("load", function(e) {
      document.getElementById("img").src       = e.target.result;
      document.getElementById("b64").innerHTML = e.target.result;
    }); 
    
    FR.readAsDataURL( this.files[0] );
  }
  
}

document.getElementById("inp").addEventListener("change", readFile);
<input id="inp" type='file'>
<p id="b64"></p>
<img id="img" height="150">

( PS: Закодоване зображення base64 (рядок) 4/3 розміру вихідних даних зображення)

Перевірте цю відповідь для завантаження декількох зображень .

Підтримка браузера: http://caniuse.com/#search=file%20api
Детальніше тут: https://developer.mozilla.org/en-US/docs/Web/API/FileReader


2
так, дякую, так що насправді я бачу, що програма для читання файлів не підтримується повністю, як я можу зробити те саме, що підтримує також старі пристрої Android / iOS?
бомбастичний

1
чи є інші браузери? xD
RicardoE

Це був справді акуратний код для перетворення зображення на base64 .. Чи можу я перетворити його назад на зображення у javascript? Я раніше використовував декодування base64 для кодування, але не маю уявлення про перетворення зображень ..
Кодер

@TheCoder Ви можете помістити його прямо в img src як рядок base64, щоб витягти з нього дані зображень, використовуйте контекст полотна.
Qwerty

1
@bombastic Я використовував JS для створення додатків для iOS та Android для обох платформ, вони працювали нормально. IOS нормальна і на андроїді, мені потрібен був власний код, щоб отримати дозвіл на галерею ...
Осман Гані Хан Масум,

36

Саме те, що вам потрібно :) Ви можете вибрати версію зворотного дзвінка або версію Promise. Зверніть увагу, що обіцянки працюватимуть в IE лише за допомогою програми заповнення polyf lib. Цей код можна розмістити один раз на сторінці, і ця функція з’явиться у всіх ваших файлах.

Подія завантаження запускається, коли прогрес зупиняється при завантаженні ресурсу (наприклад, після відправлення помилки, переривання або завантаження)

Версія зворотного дзвінка

        File.prototype.convertToBase64 = function(callback){
                var reader = new FileReader();
                reader.onloadend = function (e) {
                    callback(e.target.result, e.target.error);
                };   
                reader.readAsDataURL(this);
        };

        $("#asd").on('change',function(){
          var selectedFile = this.files[0];
          selectedFile.convertToBase64(function(base64){
               alert(base64);
          }) 
        });

Обіцяна версія

    File.prototype.convertToBase64 = function(){
         return new Promise(function(resolve, reject) {
                var reader = new FileReader();
                reader.onloadend = function (e) {
                    resolve({
                      fileName: this.name,
                      result: e.target.result, 
                      error: e.target.error
                    });
                };   
                reader.readAsDataURL(this);
        }.bind(this)); 
    };

    FileList.prototype.convertAllToBase64 = function(regexp){
      // empty regexp if not set
      regexp = regexp || /.*/;
      //making array from FileList
      var filesArray = Array.prototype.slice.call(this);
      var base64PromisesArray = filesArray.
           filter(function(file){
             return (regexp).test(file.name)
           }).map(function(file){
             return file.convertToBase64();
           });
      return Promise.all(base64PromisesArray);
    };

    $("#asd").on('change',function(){
      //for one file
      var selectedFile = this.files[0];
      selectedFile.convertToBase64().
          then(function(obj){
            alert(obj.result);
          });
      });
      //for all files that have file extention png, jpeg, jpg, gif
      this.files.convertAllToBase64(/\.(png|jpeg|jpg|gif)$/i).then(function(objArray){
            objArray.forEach(function(obj, i){
                  console.log("result[" + obj.fileName + "][" + i + "] = " + obj.result);
            });
      });
    })

html

<input type="file" id="asd" multiple/>

8
<input type="file" onchange="getBaseUrl()">
function getBaseUrl ()  {
    var file = document.querySelector('input[type=file]')['files'][0];
    var reader = new FileReader();
    var baseString;
    reader.onloadend = function () {
        baseString = reader.result;
        console.log(baseString); 
    };
    reader.readAsDataURL(file);
}

1
"файл" оголошено, але його значення ніколи не читається всередині функції getBaseUrl ().
Biranchi

@Biranchi fileвикористовується в останньому рядкуreader.readAsDataURL(file);
Ахім,

7

У цьому випадку корисно працювати з відкладеним об’єктом і повернути обіцянку:

function readImage(inputElement) {
    var deferred = $.Deferred();

    var files = inputElement.get(0).files;
    if (files && files[0]) {
        var fr= new FileReader();
        fr.onload = function(e) {
            deferred.resolve(e.target.result);
        };
        fr.readAsDataURL( files[0] );
    } else {
        deferred.resolve(undefined);
    }

    return deferred.promise();
}

І вищезазначену функцію можна використовувати таким чином:

var inputElement = $("input[name=file]");
readImage(inputElement).done(function(base64Data){
    alert(base64Data);
});

Або у вашому випадку:

$(input).on('change',function(){
  readImage($(this)).done(function(base64Data){ alert(base64Data); });
});

мені це дуже потрібно, я маю на увазі відкладене, оскільки я повертаю base64 до методу виклику. Я хочу знати, чи буде ваше рішення працювати у всіх браузерах?
Thameem
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.