处理canvas转base64/jpeg时全透明地区变为黑色情况的

2021-02-24 05:14 jianzhan

在用canvas将png照片转jpeg时,发现全透明地区被填充成黑色。

编码以下:

XML/HTML Code拷贝內容到剪贴板
  1. <p>Canvas:</p>  
  2. <canvas id="canvas" style="border: 1px solid #ccc;"></canvas>  
  3. <br>  
  4. <p>Base64转码后的照片:</p>  
  5. <div id="base64Img"></div>  
  6.   
  7. <script type="text/javascript">  
  8.     var base64Img = document.getElementById("base64Img"),   
  9.         canvas = document.getElementById("canvas"),   
  10.         context = canvas.getContext("2d");   
  11.   
  12.     // 建立新照片   
  13.     var img = new Image();   
  14.     img.src = "1.png";   
  15.   
  16.     img.addEventListener("load", function() {   
  17.         // 绘图照片到canvas上   
  18.         canvas.width = img.width;   
  19.         canvas.height = img.height;   
  20.   
  21.         context.drawImage(img, 0, 0);   
  22.   
  23.         getBase64(canvas, function(dataUrl) {   
  24.             // 展现base64位的照片   
  25.             var newImg = document.createElement("img");   
  26.                 newImg.src = dataUrl;   
  27.   
  28.             base64Img.appendChild(newImg);   
  29.         });   
  30.     }, false);   
  31.   
  32.     // 获得canvas的base64照片的dataURL(照片文件格式为image/jpeg)   
  33.     function getBase64(canvas, callback) {   
  34.         var dataURL = canvas.toDataURL("image/jpeg");   
  35.   
  36.         if(typeof callback !== undefined) {   
  37.             callback(dataURL);   
  38.         }   
  39.     }   
  40. </script>  

实际效果以下:

为何canvas会png的全透明地区转成黑色呢?

canvas变换成jpeg以前移除alpha安全通道,因此全透明地区被填充变成黑色。

可是,大家期待的是,canvas能够将png的全透明地区填充成白色。

那末如何将canvas中的全透明地区填充成白色呢?

下列是我实践活动过的两种处理计划方案,期待对你有协助。

处理计划方案1:将全透明的pixel设成白色

由于png照片的情况全是全透明的,因此大家能够找寻全透明的pixel,随后将其所有设定成白色,关键编码以下:

JavaScript Code拷贝內容到剪贴板
  1. // 将canvas的全透明情况设定成白色   
  2. var imageData = context.getImageData(0, 0, canvas.width, canvas.height);   
  3. for(var i = 0; i < imageData.data.length; i += 4) {   
  4.     // 当该像素是全透明的,则设定成白色   
  5.     if(imageData.data[i + 3] == 0) {   
  6.         imageData.data[i] = 255;   
  7.         imageData.data[i + 1] = 255;   
  8.         imageData.data[i + 2] = 255;   
  9.         imageData.data[i + 3] = 255;    
  10.     }   
  11. }   
  12. context.putImageData(imageData, 0, 0);  

详细编码以下:

XML/HTML Code拷贝內容到剪贴板
  1. <p>Canvas:</p>  
  2. <canvas id="canvas" style="border: 1px solid #ccc;"></canvas>  
  3. <br>  
  4. <p>Base64转码后的照片:</p>  
  5. <div id="base64Img"></div>  
  6.   
  7. <script type="text/javascript">  
  8.     var base64Img = document.getElementById("base64Img"),   
  9.         canvas = document.getElementById("canvas"),   
  10.         context = canvas.getContext("2d");   
  11.   
  12.     // 建立新照片   
  13.     var img = new Image();   
  14.     img.src = "1.png";   
  15.   
  16.     img.addEventListener("load", function() {   
  17.         // 绘图照片到canvas上   
  18.         canvas.width = img.width;   
  19.         canvas.height = img.height;   
  20.   
  21.         context.drawImage(img, 0, 0);   
  22.   
  23.         // 将canvas的全透明情况设定成白色   
  24.         var imageData = context.getImageData(0, 0, canvas.width, canvas.height);   
  25.         for(var i = 0; i < imageData.data.length; i += 4) {   
  26.             // 当该像素是全透明的,则设定成白色   
  27.             if(imageData.data[i + 3] == 0) {   
  28.                 imageData.data[i] = 255;   
  29.                 imageData.data[i + 1] = 255;   
  30.                 imageData.data[i + 2] = 255;   
  31.                 imageData.data[i + 3] = 255;    
  32.             }   
  33.         }   
  34.         context.putImageData(imageData, 0, 0);   
  35.   
  36.         // 展现base64位的照片   
  37.         getBase64(canvas, function(dataUrl) {   
  38.             var newImg = document.createElement("img");   
  39.                 newImg.src = dataUrl;   
  40.   
  41.             base64Img.appendChild(newImg);   
  42.         });   
  43.     }, false);   
  44.   
  45.     // 获得canvas的base64照片的dataURL(照片文件格式为image/jpeg)   
  46.     function getBase64(canvas, callback) {   
  47.         var dataURL = canvas.toDataURL("image/jpeg");   
  48.   
  49.         if(typeof callback !== undefined) {   
  50.             callback(dataURL);   
  51.         }   
  52.     }   
  53. </script>  

实际效果以下:

缺陷不言而喻。当png照片上存在半全透明地区时,会将其填充为黑色。这是大家不期待的。

处理计划方案2:在canvas绘图前填充白色情况

关键编码以下:

JavaScript Code拷贝內容到剪贴板
  1. // 在canvas绘图前填充白色情况   
  2. context.fillStyle = "#fff";   
  3. context.fillRect(0, 0, canvas.width, canvas.height);  

详细编码以下:

XML/HTML Code拷贝內容到剪贴板
  1. <p>Canvas:</p>  
  2. <canvas id="canvas" style="border: 1px solid #ccc;"></canvas>  
  3. <br>  
  4. <p>Base64转码后的照片:</p>  
  5. <div id="base64Img"></div>  
  6.   
  7. <script type="text/javascript">  
  8.     var base64Img = document.getElementById("base64Img"),   
  9.         canvas = document.getElementById("canvas"),   
  10.         context = canvas.getContext("2d");   
  11.   
  12.     // 建立新照片   
  13.     var img = new Image();   
  14.     img.src = "1.png";   
  15.   
  16.     img.addEventListener("load", function() {   
  17.         // 绘图照片到canvas上   
  18.         canvas.width = img.width;   
  19.         canvas.height = img.height;   
  20.   
  21.         // 在canvas绘图前填充白色情况   
  22.         context.fillStyle = "#fff";   
  23.         context.fillRect(0, 0, canvas.width, canvas.height);   
  24.   
  25.         context.drawImage(img, 0, 0);   
  26.   
  27.         // 展现base64位的照片   
  28.         getBase64(canvas, function(dataUrl) {   
  29.             var newImg = document.createElement("img");   
  30.                 newImg.src = dataUrl;   
  31.   
  32.             base64Img.appendChild(newImg);   
  33.         });   
  34.     }, false);   
  35.   
  36.     // 获得canvas的base64照片的dataURL(照片文件格式为image/jpeg)   
  37.     function getBase64(canvas, callback) {   
  38.         var dataURL = canvas.toDataURL("image/jpeg");   
  39.   
  40.         if(typeof callback !== undefined) {   
  41.             callback(dataURL);   
  42.         }   
  43.     }   
  44. </script>  

实际效果以下:

Perfect!

明显,在canvas绘图前填充白色情况这类方式,不但简易,并且对png照片的半全透明地区填充不好看的黑色块。强烈推荐这类处理计划方案。

另:canvas.toDataURL()方式不容许解决跨域照片。不然会出错。

总结

以上便是这篇文章内容的所有內容了,期待本文的內容对大伙儿的学习培训或工作中能带来1定的协助,假如有疑惑大伙儿能够留言沟通交流。