This question already has answers here:
HTML5 Canvas toDataURL returns blank
(2 answers)
Closed 1 year ago.
I'm using a dataUrl and setting an image's src as the url but the image is not showing up when I do that. Space is created for the image but it's blank.
var rUrl = received.pngUrl;
var image = new Image();
image.onload = function(){
};
image.src = rUrl;
document.body.appendChild(image);
Should setting an image src to data URL be available immediately?
I saw this post about using image.onload. I tried setting image.src = rUrl inside the onload function but then it wouldn't execute. My rUrl is of the form "..."
Other relevant code:
$('#submit').click(function(){
var pickedImage = new Image();
pickedImage.src = "{% static '/draw/hurricane.PNG' %}"
pickedImage.onload = function() {
context.drawImage(pickedImage, 0, 0);
}
var pngUrl = canvas.toDataURL();
socket.send(JSON.stringify({pngUrl}));
});
socket.onmessage = function(receivedMessage) {
var received = JSON.parse(receivedMessage.data);
...(code from above)...
}
I'm using this gray hurricane image. It should show under the car image and there's a blank space for it but I don't see the image. Dev tools shows there's an img src for it though.
Gray hurricane pic
Dev Tools
Image src is this. Is there anything wrong with it?

Well, that base64 code doesn't work for me either. I converted that image into base64 and worked fine. Maybe there is something wrong with your converter. Try to convert your image with a different one. Here is an example for that.
Anyway, the relevant code you attached is not the best, a lot of things missing from that. As others linked before, there are some basic tutorials on how to ask efficiently. I recommend jsfiddle to make a basic variation of your codes so we can edit it and we'll find solution faster.
I hope I could help!
Related
New to Cordova (and JS), please be gentle.
I'm trying to get my app taking photos, and everything works fine until it's time to upload them to the server. Pics take fine, edit works (if edit selected).
What I am getting though, is imageData (variable from documentation) with what appears to be a url encoded guid. (I tried an atob on it just to be sure - it is not b64 - which is clear, but just in case). The documentation even has a section that says:
function onPhotoDataSuccess(imageData) {
// Uncomment to view the base64 encoded image data //<--- this...
// console.log(imageData);
....
}
but if I look at it right there, before anything happens, I get something like:
blob:http%3A//localhost%3A4400/5d142c27-3908-4db4-87dd-0a22d6e5fd62
I've tried sending Camera.DestinationType.DATA_URL, FILE_URI, etc.. all of them. I always get this - the options don't seem to matter.
So, I have a blob. If I try to get it with an xhr request, I get a 500 error. If I try to get it with the file helper addon (window.FilePath.... I can't remember the name, but it's cordova), it doesn't work. I don't think that one likes blobs or something.
If I try to navigate to the location, I simply get:
Cannot GET /5d142c27-3908-4db4-87dd-0a22d6e5fd62
If I try to navigate to something like:
content://media/external/images/media/2/d085fb44-de36-46ed-93a4-73f140f4bc3d
it also fails.
if I split the GUID and atob it(replacing the "-"), I wind up with something like:
q×xÓMz-÷_u-ã®´-iÞ
So I don't think that's it...
I've been trying to get this figured out for hours now. Can someone point me in the right direction please? I have looked at other articles, but nothing seems to be working. I'm pretty sure it's because of the format I'm getting returned.
If this is simple, my apologies. I'm new to the OS, language, and cordova, and trying my best here. Just not sure my next steps.
If more code is needed, please let me know and I'll update it. It's all still pretty much boilerplate though.
If anyone else winds up here, this is A correct fix (as of today). I found it at: Using PhoneGap, How to get base64 image data of the photo chosen from photo library in iPhone, credit to revathy-durairajan:
create this function.
function encodeImageUri(imageUri) {
var c = document.createElement('canvas');
var ctx = c.getContext("2d");
var img = new Image();
img.onload = function () {
c.width = this.width;
c.height = this.height;
ctx.drawImage(img, 0, 0);
};
img.src = imageUri;
var dataURL = c.toDataURL("image/jpeg");
return dataURL;
}
Call this function with the imageData
var testtest = encodeImageUri(imageData);
done.
Well, I need some help about convert .svg file/image to .png file/image...
I have a .svg image displayed on my page. It is saved on my server (as a .png file). I need to convert it to a .png file on demand (on click on a button) and save the .png file on the server (I will do this with an .ajax request).
But the problem is the conversion.
I read many things about the html5 Canvas, which can probably help doing what I need to do now, but I can't find any clear solution to my problem, and, tbh, I do not understand everything I found... So I need some clear advices/help about the way I have to do it.
Here is the "html idea" template :
<html>
<body>
<svg id="mySvg" width="300px" height="300px">
<!-- my svg data -->
</svg>
<label id="button">Click to convert</label>
<canvas id="myCanvas"></canvas>
</body>
</html>
and some js :
<script>
$("body").on("click","#button",function(){
var svgText = $("#myViewer").outerHTML;
var myCanvas = document.getElementById("canvas");
var ctxt = myCanvas.getContext("2d");
});
</script>
Then, I need to draw the svg into the Canvas, get back the base64 data, and save it in a .png file on my server... but... how? I read about so many different solutions that I'm actually... lost... I'm working on a jsfiddle, but I'm actually... nowhere... http://jsfiddle.net/xfh7nctk/6/ ... Thanks for reading / help
For inline SVG you'll need to:
Convert the SVG string to a Blob
Get an URL for the Blob
Create an image element and set the URL as src
When loaded (onload) you can draw the SVG as an image on canvas
Use toDataURL() to get the PNG file from canvas.
For example:
function drawInlineSVG(ctx, rawSVG, callback) {
var svg = new Blob([rawSVG], {type:"image/svg+xml;charset=utf-8"}),
domURL = self.URL || self.webkitURL || self,
url = domURL.createObjectURL(svg),
img = new Image;
img.onload = function () {
ctx.drawImage(this, 0, 0);
domURL.revokeObjectURL(url);
callback(this);
};
img.src = url;
}
// usage:
drawInlineSVG(ctxt, svgText, function() {
console.log(canvas.toDataURL()); // -> PNG data-uri
});
Of course, console.log here is just for example. Store/transfer the string instead here. (I would also recommend adding an onerror handler inside the method).
Also remember to set canvas size using either the width and height attributes, or from JavaScript using properties.
I come long after since some other question raised from this one because the accepted answer may produce undesirable behavior.
#K3N solution is almost right, but I would go against the use of svgElement.outerHTML.
Instead, one should prefer new XMLSerializer().serializeToString(svgElement).
Also, the use of blob and of the URL API is not necessary and a simple dataURI has more compatibility accross browsers.
So a complete version of this would be :
function drawInlineSVG(svgElement, ctx, callback) {
var svgURL = new XMLSerializer().serializeToString(svgElement);
var img = new Image();
img.onload = function() {
ctx.drawImage(this, 0, 0);
callback();
}
img.src = 'data:image/svg+xml; charset=utf8, ' + encodeURIComponent(svgURL);
}
// usage:
drawInlineSVG(document.querySelector('svg'), ctxt, function() {
console.log(canvas.toDataURL())
});
I'm using fabric.js for my canvas application, toDataURL method works properly except when there is a image on canvas. When i add an image to canvas and call toDataURL it shows me a blank page.
//When i call it from chrome console
canvas.toDataURL();
//It returns a working data url with no problem.
"…fpmwgogX1TrjoqP0FACewngtZh+iYCSmDflKuOyk8Q+H+CKCqUW0spTgAAAABJRU5ErkJggg=="
//When i execute same code in a .js file it returns a data url which shows a blank image.
"…sBAmEBAw6XJzoBA/YDBMICBhwuT3QCBuwHCIQFDDhcnugEHt/IAaW9dzALAAAAAElFTkSuQmCC"
It's interesting that it's working on chrome dev console but not works in .js file even if it's same code. I noticed that working data url finishes with '==' but other one not. However i don't know what this means.
You didn't give much to analyze but I'll go from there on my gut feeling.
The image you are using is probably violating Cross-Origin Resource Sharing (CORS) requirements. When this happens the canvas will return a blank image when you try to get the pixel data either by using canvas.toDataURL() or context.getImageData().
It basically happens when the image is not located on the same domain as the page or is loaded from the file:// protocol.
You can try to add the following to the image object before setting the source:
image.crossOrigin = 'anonymous';
image.src = '...';
From tag:
<img crossOrigin='anonymous' src='...' alt='' />
This will request permission from the server to use the image cross-origin. If the server allows you should be fine.
If not you will either have to copy the image to your own server so it loads from the same domain as your page, or create a proxy page that loads the image externally and serves it to your page from the proxy page (it sounds complicated but really isn't).
If the image does not show up at all on the canvas you are probably not using load callback which you need since image loading is asynchronous. If so just add:
image.onload = function() {
/// now you can draw the image to canvas
}
image.crossOrigin = 'anonymous';
image.src = '...';
The problem is solved. The point i missed is, i was calling toDataURL function before canvas is loaded, that's why it was showing me a blank page.
canvas.loadFromJSON(json,function(){
var dataURL = canvas.toDataURL();
});
This solved my problem when i gave toDataURL() function as a callback to loadFromJSON function.
But after a while i had a different issue about CORS when i tried to upload my images from s3 bucket and i solved this problem as upward solution.
I was facing the same issues when I was trying to generate images from the canvas using Fabricsjs and generate PDF from images using JSPDF so below is my case I also spend hours on this may this can help someone to save time.
Load the canvas from JSON that i.e
canvas.loadFromJSON(json, canvas.renderAll.bind(canvas), function(obj, object) {
//fabric.log(obj, object);
});
Canvas was drawing images all fine then I was generating the images from that canvas it was a case of multiple images in a single canvas and I was generating PDF based on each page of the canvas.
Instead of this
canvas.toDataURL('image/png', 0.1)
I used this and it starts returning me the propper images dataurl
var imgData = document.getElementById('canvas_0').toDataURL();
Below are snippets
$("#pdfSelector .canvas-container").each(function(index, value){ // canvas selector
html2canvas($("#canvas_"+index), { // html2canvas used to get images
onrendered: function(canvas) { // on successfully render of images
//var imgData = canvas.toDataURL('image/png', 0.1);
var imgData = document.getElementById('canvas_0').toDataURL();
const imgProps= doc.getImageProperties(imgData);
const pdfWidth = doc.internal.pageSize.getWidth();
const pdfHeight = (imgProps.height * pdfWidth) / imgProps.width;
doc.addImage(imgData, 'PNG', 0, 0, pdfWidth, pdfHeight, 'page_'+index, 'FAST');
}
});
});
Is there any way to get a callback on background-image is loaded with a base 64 DataURL. I can cache this with image src attribute, but with data url ? How ?
Should work the way it usually does:
var image = new Image();
image.src = ".......";
image.onload = function() {
//image was loaded
};
Caching by the browser is of course disabled for Base64 strings, that's why it's normally only used for small images, like icons and stuff.
You can convert images to Base64 online here : http://base64img.com/#encode
Note that some browsers can have limitations on size for Base64.
Seems that style.BackgroundImage property do not set the value asynchronously. So my problem is solved.
I have image data (either JPEG or PNG) in a Javascript variable. How do I display the image in an HTML document? These are very large images and code like this doesn't work because the URI is too long:
// doesn't work because the URI is too long
$('img#target').attr('src', 'data:...');
Using canvas is probably the answer but I cannot figure out how to load it with the image data.
In case you are wondering: no, I cannot just download the image data directly to the img tag. It came from the server encrypted and was decrypted within the browser using Javascript.
Thanks,
-- Art Z.
To use a data URL to draw on a canvas:
var img = new Image;
img.onload = function(){
myCanvasContext.drawImage(img,0,0);
};
img.src = "data:...";
Per this question/answer be sure that you set the onload before the src.
You say that "the URI is too long", but it is not clear what you mean by this. Only IE8 has a 32kB limit on the data URI; for other browsers it should work fine. If you are experiencing an error, please describe it.
It turns out that
$('img#target').attr('src', 'data:...');
does work in all except IE. My problem originated elsewhere.