0

I am trying to extract base64 HTML image data after clicking a button which loads the base64 data from the server.

When testing locally, I have to click twice initially to get the expected results. And every subsequent click works as expected.

How can I shape this code so that it triggers on only one click?

<html>
<button id="t-load" type="button" data-board="g" data-tid="0" style="font-size: 11px; padding: 0px; width: 90px; box-sizing: border-box; margin: 0px 6px 0px 0px; vertical-align: middle; height: 18px;">Get Captcha</button>

<div id="t-fg" style="width: 100%; height: 100%; position: absolute; background-repeat: no-repeat; background-position: left top; background-image: url(&quot;&quot;);"></div>
</html>


<script>
// turn string to image
function b64toImageData(datas) {
    return new Promise(async function (resolve, reject) {

        // We create an image to receive the Data URI
        var img = document.createElement('img');

        // When the event "onload" is triggered we can resize the image.
        img.onload = function () {
        // We create a canvas and get its context.
        var canvas = document.createElement('canvas');
        var ctx = canvas.getContext('2d');

        // We set the dimensions at the wanted size.
        canvas.width = img.width;
        canvas.height = img.height;

        // We resize the image with the canvas method drawImage();
        ctx.drawImage(this, 0, 0, img.width, img.height);
        // get image data
        var imageDatas = ctx.getImageData(0, 0, img.width, img.height);

        // This is the return of the Promise
        resolve(imageDatas);
        };
        // add data to img
        img.src = datas;
    })
    }

// event listener;
document.addEventListener("click", function () {
  document.getElementById("t-load").onclick = async function () {
        // get img
        var fg = document.getElementById("t-fg").style.backgroundImage.slice(5, -2);
        console.log(fg)
        // get img data
        var base_img = await b64toImageData(fg);
        // log results
        console.log(base_img.data);
  }

});
</script>

John Stud
  • 1,506
  • 23
  • 46
  • Because you bind event only on first click. Remove outer `document.addEventListener("click")` – Justinas Jan 13 '23 at 21:36
  • This leads to the following error if I strip that out; `TypeError: Cannot set properties of null (setting 'onclick')` – John Stud Jan 13 '23 at 21:39
  • 1
    @JohnStud It works fine: https://jsfiddle.net/skg5ow69/ – Unmitigated Jan 13 '23 at 21:40
  • Interesting; I am trying to run this in Requestly/Chrome and it is yielding that error! – John Stud Jan 13 '23 at 21:42
  • 2
    @JohnStud Depending on where you placed your script, you may need to replace `document.addEventListener("click"` with `document.addEventListener("DOMContentLoaded"` – Unmitigated Jan 13 '23 at 21:43
  • @Unmitigated appreciate the help; that resolves the error but when using this on the live website I want to work on, it emits no results in my console; any thoughts? – John Stud Jan 13 '23 at 21:47
  • 1
    @JohnStud Where exactly did you place your script? If you place it right before the closing body tag, then it should work fine. – Unmitigated Jan 13 '23 at 21:48
  • Let us [continue this discussion in chat](https://chat.stackoverflow.com/rooms/251129/discussion-between-john-stud-and-unmitigated). – John Stud Jan 13 '23 at 21:48
  • @JohnStud As to why you get `Cannot set properties of null`, see [Why does jQuery or a DOM method such as getElementById not find the element?](https://stackoverflow.com/questions/14028959/why-does-jquery-or-a-dom-method-such-as-getelementbyid-not-find-the-element) – Darryl Noakes Jan 13 '23 at 21:58

1 Answers1

1

Instead of the event handlers that you are using, you can delegate the click handler to the body and use still async.

function b64toImageData(datas) {
  return new Promise(async function(resolve, reject) {

    // We create an image to receive the Data URI
    var img = document.createElement('img');

    // When the event "onload" is triggered we can resize the image.
    img.onload = function() {
      // We create a canvas and get its context.
      var canvas = document.createElement('canvas');
      var ctx = canvas.getContext('2d');

      // We set the dimensions at the wanted size.
      canvas.width = img.width;
      canvas.height = img.height;

      // We resize the image with the canvas method drawImage();
      ctx.drawImage(this, 0, 0, img.width, img.height);
      // get image data
      var imageDatas = ctx.getImageData(0, 0, img.width, img.height);

      // This is the return of the Promise
      resolve(imageDatas);
    };
    // add data to img
    img.src = datas;
  })
}

document.body.addEventListener("click", async(e) => {
  let el = e.target;
  if (el.id == "t-load") {
    var fg = document.getElementById("t-fg").style.backgroundImage.slice(5, -2);
    console.log(fg)
    console.log("WAITING!! don't click again!")
    var base_img = await b64toImageData(fg);
    console.log(base_img.data);
  }
});
<button id="t-load" type="button" data-board="g" data-tid="0" style="font-size: 11px; padding: 0px; width: 90px; box-sizing: border-box; margin: 0px 6px 0px 0px; vertical-align: middle; height: 18px;">Get Captcha</button>

<div id="t-fg" style="width: 100%; height: 100%; position: absolute; background-repeat: no-repeat; background-position: left top; background-image: url(&quot;&quot;);"></div>
imvain2
  • 15,480
  • 1
  • 16
  • 21
  • Thanks this is super helpful and a nice improvement it seems. I am trying to then run this code on https://boards.4channel.org/ck/ to grab captcha; any idea why console.log(fg) still remains empty in it's output? – John Stud Jan 13 '23 at 22:38