将图片转换为Base64

108
<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);
});

我了解了文件API和其他相关内容,希望能有一个简单且跨浏览器的解决方案(排除IE6/IE7)。

非常感谢任何帮助。


1
你对HTML5文件API有什么不明白的地方?你尝试了什么?什么没有起作用? - epascarello
@epascarello,实际上它们并没有得到完全支持http://caniuse.com/#feat=fileapi。我需要一个解决方法,特别是因为仍然使用Android版本(旧版本)以及旧的iOS版本,而且我也想涉及仍然被广泛使用的IE9:P - bombastic
一个解决方法是为了什么?你想用这个文件做什么?base64file() - 那是一个插件吗? - David Hellsing
@David,我只想让用户从他的电脑上选择文件后获取base64文件,base64file()仅为示例。 - bombastic
1
@epascarello 是的,那正是我的问题,如何支持所有浏览器 :D - bombastic
显示剩余5条评论
8个回答

252

function readFile() {
  
  if (!this.files || !this.files[0]) return;
    
  const FR = new FileReader();
    
  FR.addEventListener("load", function(evt) {
    document.querySelector("#img").src         = evt.target.result;
    document.querySelector("#b64").textContent = evt.target.result;
  }); 
    
  FR.readAsDataURL(this.files[0]);
  
}

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

(P.S: Base64编码的图片(字符串),大小为原始图片数据的4/3)

查看此答案以进行多图上传

浏览器支持:http://caniuse.com/#search=file%20api
更多信息请参见:https://developer.mozilla.org/en-US/docs/Web/API/FileReader


2
谢谢,我可以看到文件阅读器并不完全支持,那么我该如何使其支持旧的Android/iOS设备? - bombastic
2
还有其他浏览器吗?xD - RicardoE
那段用于图像转换为base64的代码真的很棒。我能否在javascript中将其转换回图像?我以前使用过base64编码解码,但对图像转换没有任何想法。 - The Coder
1
@TheCoder 你可以将它直接放入img src作为base64字符串,要获取图像数据,请使用画布上下文。 - Qwerty
1
@bombastic 我使用JS在iOS和Android平台上构建了应用程序,它们都运行良好。iOS正常,而在Android上,我需要一些本地代码来获取访问相册的权限... - Osman Gani Khan Masum
显示剩余2条评论

38

正是您所需要的:)您可以选择回调版本或Promise版本。请注意,只有在使用Promise polyfill库时,才能在IE中使用Promise。您可以将此代码放置在页面上一次,然后这个函数将出现在所有文件中。

当资源加载已停止(例如分派“error”,“abort”或“load”之后),会触发loadend事件

回调版本

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

Promise 版本

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

当应用于使用Angular的程序时,此方法会产生错误,因为Angular无法容忍File.prototype.convertToBase64 = function(){..}和FileList.prototype.convertAllToBase64 = function(regexp){...}。 - Wisarut Bholsithi
谢谢!你可以将它重写为Angular服务... - Alex Nikulin
@N3K0100183的方法可以修改为适用于Angular平台的TypeScript文件。 - Wisarut Bholsithi

14
<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() 函数中声明了 'file',但从未读取其值。 - Biranchi
1
@Biranchi file 在最后一行 reader.readAsDataURL(file); 中被使用。 - Achim
即使使用Angular编写的程序,该方法仍然有效,尽管根据程序编写风格需要更改定义文件变量的方式。 - Wisarut Bholsithi
2
我希望我能够为这个答案点赞两次,因为它非常有效和简洁。 - Jose Manuel Abarca Rodríguez

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

1

使用jQuery将图像转换为base64格式的函数(也可以使用原生JavaScript)。希望能对您有所帮助!

用法:输入是您的nameId,输入具有文件图像。

<input type="file" id="asd"/>
<button onclick="proccessData()">Submit</button>

<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
<script>

async function converImageToBase64(inputId) {
  let image = $('#'+inputId)[0]['files']

  if (image && image[0]) {
    const reader = new FileReader();

    return new Promise(resolve => {
      reader.onload = ev => {
        resolve(ev.target.result)
      }
      reader.readAsDataURL(image[0])
    })
  }
}

async function proccessData() {
  const image = await converImageToBase64('asd')
  console.log(image)
}

</script>

例子:将图像转换为Base64('yourFileInputId')

https://codepen.io/mariohandsome/pen/yLadmVb


0

适用于现代浏览器(不支持IE)

HTML文件输入

<style>
.upload-button {
  background-color: grey;
}

.upload-button input{
  display:none;
}
</style>
<label for="upload-photo" class="upload-button">
    Upload file
    <input
     type="file"
     id="upload-photo"
    </input>
</label>

JS

document.getElementById("upload-photo").addEventListener("change", function({target}){
 if (target.files && target.files.length) {
      try {
        const uploadedImageBase64 = await convertFileToBase64(target.files[0]); 
        //do something with above data string 
      } catch() {
        //handle error
      }
    }
})

function convertFileToBase64(file) {
  return new Promise((resolve, reject) => {
    const reader = new FileReader();
    reader.readAsDataURL(file);
    reader.onload = () => resolve(reader.result);
    reader.onerror = reject;
  });
}

-1

// https://developer.mozilla.org/en-US/docs/Web/API/FileReader/readAsDataURL

/* Simple */
function previewImage( image, preview, string )
{

    var preview     = document.querySelector( preview );
    var fileImage   = image.files[0];

    var reader      = new FileReader();

    reader.addEventListener( "load", function() {

        preview.style.height    = "100";
        preview.title           = fileImage.name;

        // convert image file to base64 string
        preview.src             = reader.result;

        /* --- */

        document.querySelector( string ).value = reader.result;                    

    }, false );

    if ( fileImage )
    {
        reader.readAsDataURL( fileImage );
    }

}

document.querySelector( "#imageID" ).addEventListener( "change", function() {

    previewImage( this, "#imagePreviewID", "#imageStringID" );

} )
/* Simple || */
<form>

    File Upload: <input type="file" id="imageID" /><br />
    Preview: <img src="#" id="imagePreviewID" /><br />    
    String base64: <textarea id="imageStringID" rows="10" cols="50"></textarea>

</form>

codesanbox

{{链接1:codesandbox}}


-1
  const FR = new FileReader();
    
  FR.addEventListener("load", function(evt) {
    document.querySelector("#img").src         = evt.target.result;
    document.querySelector("#b64").`textContent` = evt.target.result;
  }); 
    
  FR.readAsDataURL(this.files[0]);
  

< img id="inp" src="https://tlc.ca/pricing-tool/images/logo.png">
<p id="b64"></p>
<img id="img" height="150">
  • 列表项

1
你的回答可以通过提供更多支持信息来改进。请编辑以添加进一步的细节,例如引用或文档,以便他人可以确认你的答案是正确的。您可以在帮助中心中找到有关如何编写良好答案的更多信息。 - Community
1
你只是复制了接受的答案,并在没有解释的情况下添加了一些东西,这有什么用? - tyzion

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接