HTML5画布的toDataURL方法返回空白图片

4

从一块空白的画布开始:

<canvas id='myCanvas' width='800' height='600'></canvas>

接下来要初始化画布:

  function init_canvas(){
    var canvas = document.getElementById('myCanvas');
    context = canvas.getContext('2d');
    context.lineCap = 'round';
    // fill it with white background
    context.save();
    context.fillStyle = '#fff';
    context.fillRect(0, 0, context.canvas.width, context.canvas.height);
    context.restore();
    return;
  }

接下来在画布上进行一系列的绘制。

然后尝试使用后端的ajax和PHP将其保存到服务器上。

在客户端:

var img = canvas.toDataURL('image/png');
// strip the leading garbage.
img.substr(img.indexOf(',')+1).toString();

将生成的字符串直接带入PHP并使用base64_decode()解密该字符串即可... 图像总是正确的大小,但它没有任何绘图 - 只有一个透明的图像。这似乎不是PHP中base64_decode()的问题,它似乎是安全问题或其他什么原因导致的。这在Firefox 4和最新的Chrome浏览器都会失败。
使用"image/png"头将结果png图像转储到Firefox时,在错误控制台中显示如下内容:
Error: Image corrupt or truncated: http://somewhere.com/showimage.php
Source file: http://somewhere.com/showimage.php

但是...就我所知,该图像并未损坏或截断,除非toDataURL()在所有地方都无法正常工作,因为php部分只是对toDataURL()的结果进行base64_decode()。

有什么想法吗?

谢谢!


你可能需要查看此示例的源代码。http://motyarblog.000space.com/imagetouri.php使用了jQuery。你的问题可能在于`canvas.toDataURL`是图像的base64字符串,如果超过255个字符,则无法作为$_GET传递。 - Lawrence Cherone
已经考虑过这一点并确认我正在进行XHR的POST操作。在后端,我将base64编码的图像转储到文本文件中,长度正常,是一个真正的b64字符串,可以正常解码,只是一个空白图像。 - Erick
2
你在 PHP 文档中看到了“base64_decode”函数的这个注释吗? - robertc
robertc... 哇,真是个笑话。整个将空格替换为加号的事情就是这样。把那条评论作为回复,我会将其标记为解决方案。谢谢大家。 - Erick
4个回答

7

您是否看到了PHP文档中base64_decode函数的这条评论

If you want to save data that is derived from a Javascript canvas.toDataURL() function, you have to convert blanks into plusses. If you do not do that, the decoded data is corrupted:

<?php
  $encodedData = str_replace(' ','+',$encodedData);
  $decocedData = base64_decode($encodedData);
?>

0

不是的。robertc实际上已经给出了答案,但只是在评论中,而不是在回答中。谢谢你的关注。 - Erick
1
@Erick 我会将其添加为答案 ;) - robertc

0

这个可以正常工作,不会返回空白图像。

saveImage.php

 <?php 
        $result = array();  
        $encodedData = explode("," , $_POST['imgData'])[1];
        $encodedData = str_replace(' ','+',$encodedData);
        $decocedData = base64_decode($encodedData); 
        //Location to where you want to save image
        $NewFileName = './photos/'.$_POST['imgName'].'.jpg';
        file_put_contents($NewFileName,$decocedData);
        $result['status'] = 1;
        $result['saveImage'] = $NewFileName;
        echo json_encode($result);
    ?>

script.js

<script>
//This post the data to server on fly

function saveImage(canvasData,FileName) {
  var xmlHttpReq = false;

  if (window.XMLHttpRequest) {
    ajax = new XMLHttpRequest();
  }
  else if (window.ActiveXObject) {
    ajax = new ActiveXObject("Microsoft.XMLHTTP");
  }

  ajax.open("POST", "saveImage.php", false);
  ajax.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
  ajax.onreadystatechange = function() {

    if (this.readyState == 4 && this.status == 200) {
        var rawArr = JSON.parse(this.responseText);
        var ImagePreviewer= document.getElementById("ImagePreviewer");
    ImagePreviewer.src=rawArr.saveImage;
    //console.log(rawArr.saveImage);
    }

  }
  ajax.send("imgData=" + canvasData+"&imgName=" + FileName);
}
// Call this with button click or whatever you want 
function saveJPG(filename) {
//do other logic here
var dataURL = TheCanvas.toDataURL("image/jpg");
    saveImage(dataURL,filename);
}
<script>

-1
在JavaScript中,对于那些支持该方法的浏览器,使用默认类型"image/png"发送canvas.toDataURL()的结果。
var imageInfo = canvas.toDataURL();

创建 PNG 文件的直接方法:

<?php
$imageInfo = imageInfoFromBrowser(); // Your method to get the data
$image = fopen($imageInfo, 'r');
file_put_contents("fileName.png", $image);
fclose($image);
?>

我已经在PHP 5.3中使这个工作正常,对于其他版本,请检查。

用于在PHP上进行测试的图像数据示例。

$imageDataInfoSample = ""

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