9

I have a image that needs to converted to Uint8Array.

For Example:

var image = new Image();
image.src="image13.jpg";

Now i need to transform this image variable to Uint8Array. Is there any function in JS, which allows to achieve this functionality?

3
  • @david Hi david, thanks for pointing to some other similar question. But the type of the variable we are dealing with is different. Commented Aug 16, 2018 at 4:21
  • 2
    @Learner The question you pointed out to is exactly opposite of what i am asking here. Commented Aug 16, 2018 at 4:23
  • 1
    Sorry didn't sleep and its morning, can you reverse it if its opposite? :D Commented Aug 16, 2018 at 4:25

1 Answer 1

7

You may use the Canvas API to retrieve the image data:

  1. Retrieving the data in the binary form of various image formats (PNG, JPEG, etc.) using canvas.toBlob() and blob.arrayBuffer():
async function imageToUint8Array(image, context) {
  context.width = image.width;
  context.height = image.height;
  context.drawImage(image, 0, 0);
  const blob = await context.canvas.toBlob(
    callback,
    "image/jpeg" // the MIME type of the image format
    1 // quality (in the range 0..1)
  );
  return new Uint8Array(await blob.arrayBuffer());
}

function toBlob(canvas, type = "image/png", quality = 1) {
  return new Promise((resolve) => canvas.toBlob(blob => resolve(blob)))
}

async function imageToUint8Array(image, context) {
  context.width = image.width;
  context.height = image.height;
  context.drawImage(image, 0, 0);
  const blob = await toBlob(context.canvas, "image/png");
  return new Uint8Array(await blob.arrayBuffer());
}

const image = new Image();
// stackoverflow logo
image.src = ``;

const canvas = document.createElement("canvas");
const context = canvas.getContext("2d");

image.onload = async() => console.log(await imageToUint8Array(image, context));

  1. Retrieving the raw image pixels using canvas.getImageData():
function imageToUint8Array(image, context) {
  context.width = image.width;
  context.height = image.height;
  context.drawImage(image, 0, 0);
  // `getImageData().data` is a `Uint8ClampedArray`, which differs from `Uint8Array` only in
  // how data is treated when values are being *set*, so it is valid to perform the conversion
  // into a `Uint8Array`.
  return new Uint8Array(context.getImageData(0, 0, image.width, image.height).data.buffer);
}

function imageToUint8Array(image, context) {
  context.width = image.width;
  context.height = image.height;
  context.drawImage(image, 0, 0);
  // `getImageData().data` is a `Uint8ClampedArray`, which differs from `Uint8Array` only in
  // how data is treated when values are being *set*, so it is valid to perform the conversion
  // into a `Uint8Array`.
  return new Uint8Array(context.getImageData(0, 0, image.width, image.height).data.buffer);
}

const image = new Image();
// stackoverflow logo
image.src = ``;

const canvas = document.createElement("canvas");
const context = canvas.getContext("2d");

image.onload = async() => console.log(imageToUint8Array(image, context));

Sign up to request clarification or add additional context in comments.

6 Comments

"There is no extra processing" gosh there is... If it's synchronous it's because this method is old and because the slow operation is the reading of the buffer anyway, which needs to be done before the next call draws something new on it. Also, your first snippet does create a completely new image, which will "look" the same, but won't be the actual image's data. See stackoverflow.com/a/42916772/3702797
Not sure what you mean. There isn't any processing done once you've drawn the image onto the canvas. The fact that it creates a new image is not important, since all we care about is the image's contents, so as long as it looks the same, that's good enough.
Yes, there is a lot of processing being done when you call ctx.getImageData(): The bitmap that was until there stored on the GPU buffer has to be moved to the CPU, from there every pixels have to be read, and unmultiplied to RGBA before being stored in the ImageData's buffer. That's literally one of the slowest operations in the canvas API. And yes, it does matter that you are now generating a new bigger image with more encoding artifacts.
I think you must've misunderstood what I meant. What you're describing is a consequence of how the data is stored. Of course it has to be read, unpacked, and serialized into a JS object. What I mean is that it isn't additionally converted to PNG/JPEG/etc on top of that. That's why I wrote extra processing. The extra artifacts only matter if you're trying to represent the exact same image. OP's question is a little vague in this regard.
"There is no extra processing, so this method is fully synchronous." This sentence clearly states that this method is synchronous because there is no extra processing. That's just not true, it's synchronous for other reasons. And "no extra processing" is in contrast with your first paragraph which stated "the data has to be processed, which may take a long time, and should not block the main thread." Here too the data has to be processed, a different processing but still a processing, that still "may take a long time" and could also "block the main thread".
|

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.