7

I have imagedata in base64 format returned from Expo ImagePicker component, and I want to convert it to An Uint8ClampedArray of RGBA pixel values in the form [r0, g0, b0, a0, r1, g1, b1, a1, ...], cause it's the only input accepted by jsQR library

I tried this, but did not work:

const dataURItoBlob = byteString  => {

  // write the bytes of the string to a typed array
  var ia = new Uint8ClampedArray(byteString.length);
  for (var i = 0; i < byteString.length; i++) {
    ia[i] = byteString.charCodeAt(i);
  }

  return ia;
};

any help will be highly appreciated

Hend El-Sahli
  • 6,268
  • 2
  • 25
  • 42

3 Answers3

6

I'm guessing your base64-encoded URI is actually a PNG, so you could do the following:

const {PNG} = require('pngjs');
const jsqr = require('jsqr');

const dataUri = '';
const png = PNG.sync.read(Buffer.from(dataUri.slice('data:image/png;base64,'.length), 'base64'));
const code = jsqr(Uint8ClampedArray.from(png.data), png.width, png.height);

// code.data now contains the URL string encoded by the QR code
Oleg Vaskevich
  • 12,444
  • 6
  • 63
  • 80
4

The solution that I found is as follows:

it requires the usage of typedarray, Buffer, jpeg-js

var Uint8ClampedArray = require('typedarray').Uint8ClampedArray;
const Buffer = require('buffer').Buffer;
global.Buffer = Buffer; // very important
const jpeg = require('jpeg-js');


const jpegData = Buffer.from(base64, 'base64');
var rawImageData = jpeg.decode(jpegData);

var clampedArray = new Uint8ClampedArray(rawImageData.data.length);
// manually fill Uint8ClampedArray, cause Uint8ClampedArray.from function is not available in react-native
var i;
for (i = 0; i < rawImageData.data.length; i++) {
  clampedArray[i] = rawImageData.data[i];
}
Hend El-Sahli
  • 6,268
  • 2
  • 25
  • 42
0

New answer for react native:

this package should do the trick. let me know if it works.

https://www.npmjs.com/package/get-image-data

OLD ASNWER FOR BROWSER:

First put the base64 img in a canvas like described here:

Base64 PNG data to HTML5 canvas

Then apply ImageData on your canvas (https://developer.mozilla.org/en-US/docs/Web/API/ImageData)

This should give you the desired result.

Domenik Reitzner
  • 1,583
  • 1
  • 12
  • 21
  • Thank you very much for your help ... but the thing is I'm in react-native nor react :) – Hend El-Sahli Aug 21 '18 at 13:07
  • Is it the only way to use react-native WebView to get access to Canvas to get image data in RGBA format? – Hend El-Sahli Aug 21 '18 at 13:09
  • Oh, that is interresting :) Then that might help: https://www.npmjs.com/package/get-image-data – Domenik Reitzner Aug 21 '18 at 13:42
  • While this link might provide some limited, immediate help, an answer [should include sufficient context around the link](//meta.stackoverflow.com/a/8259) so your fellow users will have some idea what it is and why it’s there. Always quote the most relevant part of an important link, to make it more useful to future readers with other, similar questions. In addition, other users tend to respond negatively to answers which are [barely more than a link to an external site](//meta.stackexchange.com/q/225370), and they [might be deleted](//stackoverflow.com/help/deleted-answers). – Cerbrus Aug 21 '18 at 14:06