1

First part of my application loads picture with file chooser from my file system and draws image to canvas. I take that canvas and convert it to data URL and also save it in my database.

var imgData = canvas.toDataURL("image/jpg");
factory.saveData(imgData);                      // execute some java code

Here is the problem. I can't redraw that image, my javascript:

var pic = new Image();
pic.onload = function() {
    ctx.drawImage(pic, 0, 0);                  // ctx = canvas.getContext("2d")
};
// This data url is copy/pasted from database for simplicity
pic.src = "..........ElFTkSuQmCC";

Onload function is called but I just get transparent image.

milos
  • 77
  • 3
  • 10
  • 1
    have you tried setting the width and height specifically, e. g. `ctx.drawImage(pic, 0, 0, canvas.width, canvas.height)`? – Simon Hänisch Jun 24 '18 at 23:42
  • @SimonHänisch yes, I tried different values and image is still transparent – milos Jun 25 '18 at 00:03
  • 1
    When do you call canvas.toDataURL? Are you sure the `pic.onload` handler has already fired at this time? i.e are you sure something has been drawn on the canvas when you do export its content? – Kaiido Jun 25 '18 at 01:17
  • @Kaiido I think you are right! `canvas.toDataUrl()` is called before anything is drawn to canvas. Brb trying to fix it. – milos Jun 25 '18 at 01:27

1 Answers1

0

Probably has something to do with the fact that getDataFromDatabase() is asynchronous so you're returning a promise object to pic.src instead of the data url. Fix this by using a then call on getDataFromDatabase and using the result on pic.src like this:

var pic = new Image();
var pic.onload = function() {
    ctx.drawImage(pic, 0, 0);
};
getDataFromDatabase().then(dataUrl => {
    pic.src = dataUrl;
});

Here's a full example of this in action. I draw to a canvas, store the canvas data as a base64 data url into a variable, save it into a mock DB using closures, then reload the data from a simulated asynchronous function call by using a promise.

const canvas = document.getElementById('myCanvas');
const ctx = canvas.getContext('2d');

// Initialize canvas to a random image
const imageUrl = 'https://cdn.sstatic.net/Sites/stackoverflow/company/img/logos/so/so-logo.svg?v=a010291124bf';
const initialImage = new Image();
initialImage.src = imageUrl;
initialImage.onload = function() {
  ctx.drawImage(initialImage, 0, 0);
}

// Convert canvas to base64 data url
const imageData = canvas.toDataURL('image/png');
const getDataFromDatabase = saveData(imageData);

// Clear the canvas
ctx.clearRect(0, 0, canvas.width, canvas.height);

// Redraw the image
const pic = new Image();
getDataFromDatabase().then(data => {
  pic.src = data;
});
pic.onload = function() {
  ctx.drawImage(pic, 0, 0);
}

function saveData(image) {
  return function getDataFromDatabase() {
    return new Promise(resolve => {
      resolve(image);
    });
  }
}
<canvas id="myCanvas">
  
</canvas>

Edit: It shouldn't really matter where you get the image data from. If the data is not malformed, it should draw to the canvas. Here's an example identical to the way you're loading image data (from a hardcoded data url):

const imageData = '';

const canvas = document.getElementById('myCanvas');
const ctx = canvas.getContext('2d');

const image = new Image();
image.onload = function() {
  ctx.drawImage(image, 0, 0);
}
image.src = imageData;
<canvas id="myCanvas">
</canvas>

I suggest you check your image data using a base64 image decoder to see if your image data is malformed or not. If your data url is valid, then something else I can suggest is ensuring your canvas width and height are equal to or greater than the image's width and height or else it'll be hidden (if the image you're supplying has empty space / padding).

Jake Miller
  • 2,432
  • 2
  • 24
  • 39
  • I may have omitted some important info in my post I apologize. I didn't really call `getDataFromDatabase()` I just pasted base64 data code in quotations marks, so no promise object there. Also, I loaded my picture with file chooser from my file system, not from web like in your example. – milos Jun 25 '18 at 00:22
  • I tried [this](https://codebeautify.org/base64-to-image-converter) site and I see transparent image. Did run [this](https://stackoverflow.com/a/31245864/5059176) code and it returns FALSE. I guess I have malformed data, probably gonna open another post thanks for help. – milos Jun 25 '18 at 00:58
  • 1
    @vesa Yeah your base64 image data is malformed then. I’ll help you sort it out. Let me create a StackOverflow chat and I’ll invite you. Hang on. – Jake Miller Jun 25 '18 at 01:19
  • 1
    If the image data was not correct, then the onload event would not fire. – Kaiido Jun 25 '18 at 01:28
  • @Jake Miller can't join, 2 more reputaion needed to join – milos Jun 25 '18 at 01:30
  • @Kaiido Exactly. Which is why his canvas isn't drawing most likely. His image doesn't load properly on a base64 image decoder so I'm assuming his base64 data url is malformed in some way. – Jake Miller Jun 25 '18 at 01:32
  • 2
    "*Onload function is called but I just get transparent image.*" The problem is that they call toDataURL when the canvas is still empty. – Kaiido Jun 25 '18 at 01:35
  • @Kaiido Hmm, that makes sense. I overlooked that sentence. – Jake Miller Jun 25 '18 at 01:37
  • @vesa Can you post your code where you draw an image from the file system to the canvas as well? Kaiido made a good point that you're probably creating your data url before the canvas has a chance to draw the image that you select from your file system. – Jake Miller Jun 25 '18 at 01:37
  • you solved the problem, thanks. My `canvas.toDataURL("image/png")` was outside of Onload function. I just put that line inside Onload function and image is correctly drawn. – milos Jun 25 '18 at 01:40