parallax / jsPDF

Client-side JavaScript PDF generation for everyone.
https://parall.ax/products/jspdf
MIT License
29.2k stars 4.67k forks source link

Image content for phonegap app on android #207

Closed pascalPimaia closed 10 years ago

pascalPimaia commented 10 years ago

I m trying to use addImage in a phonegap application. All the code works perfectly well on a desktop browser but results in a corrupted pdf file on android.

toDataURL returns a png image string instead of a requested jpeg image string. I used the JPEGencoder framework which can be found in this repo https://github.com/owencm/javascript-jpeg-encoder.git to fix this.

There is a copy of the code I use at the end of this post. The code prepares data for addImage and does pretty much what addImage does at the beginning when passing an Image object.

Once the document is written I can observe a /Image object in the pdf script code but it differs from what I obtain on a desktop computer (JPEG algo probably differs). what bother me is that Acrobat reader complains there is an error in the pdf file and no image is displayed.

Here is the piece of code I used

    var getJpegImageDataURI= function(img,w,h) 
    {
        var canvas      = document.createElement("canvas"),                 
            ctx         = canvas.getContext("2d"),
            url         = undefined;

        if( w === undefined )w=img.width;
        if( h === undefined )h=img.height;

        if(ctx!==undefined)
        {
            canvas.width  = w;
            canvas.height = h; 

            ctx.drawImage(img,0,0,w,h,0,0,w,h);

            var toDataURLFailed=false;
            try
            {
                url=canvas.toDataURL("image/jpeg",1.);
            }
            catch(e)
            {
                toDataURLFailed=true; // android may generate png
                alert("toDataURL failed try JPEGEncoder if type deos not match jpeg: " + e);
            }

            if( toDataURLFailed || (url.slice(0,"data:image/jpeg".length) !== "data:image/jpeg") ) // check not jpeg type 
            {
                    // fallback generate jpeg data 
                    try
                    {       
                        var encoder = new JPEGEncoder();
                        url= encoder.encode(ctx.getImageData(0,0,w,h), 100);
                    }
                    catch(e){
                        alert("JPEGEncoder failed"+e);          
                    }
            }
        }
        else 
        {
            console.error("[IZIWATT-CONTOLLER] 2d canvas not supported");
        }
        if(url===undefined)alert("getJpegImageDataURI return undefined");
        return url;
    };
MrRio commented 10 years ago

This PR may be of interest to you as it adds PNG support

https://github.com/MrRio/jsPDF/pull/195

On Wed, Mar 12, 2014 at 9:43 AM, Pascal Sautot notifications@github.comwrote:

I m trying to use addImage in a phonegap application. All the code works perfectly well on a desktop browser but results in a corrupted pdf file on android.

toDataURL returns a png image string instead of a requested jpeg image string. I used the JPEGencoder framework which can be found in this repo https://github.com/owencm/javascript-jpeg-encoder.git to fix this.

There is a copy of the code I use at the end of this post. The code prepares data for addImage and does pretty much what addImage does at the beginning when passing an Image object.

Once the document is written I can observe a /Image object in the pdf script code but it differs from what I obtain on a desktop computer (JPEG algo probably differs). what bother me is that Acrobat reader complains there is an error in the pdf file and no image is displayed.

Here is the piece of code I used var getJpegImageDataURI= function(img,w,h) { var canvas = document.createElement("canvas"),

ctx = canvas.getContext("2d"), url = undefined;

if( w === undefined )w=img.width;
if( h === undefined )h=img.height;

if(ctx!==undefined)
{
    canvas.width  = w;
    canvas.height = h;

    ctx.drawImage(img,0,0,w,h,0,0,w,h);

    var toDataURLFailed=false;
    try
    {
        url=canvas.toDataURL("image/jpeg",1.);
    }
    catch(e)
    {
        toDataURLFailed=true; // android may generate png
        alert("toDataURL failed try JPEGEncoder if type deos not match jpeg: " + e);
    }

    if( toDataURLFailed || (url.slice(0,"data:image/jpeg".length) !== "data:image/jpeg") ) // check not jpeg type
    {
            // fallback generate jpeg data
            try
            {
                var encoder = new JPEGEncoder();
                url= encoder.encode(ctx.getImageData(0,0,w,h), 100);
            }
            catch(e){
                alert("JPEGEncoder failed"+e);
            }
    }
}
else
{
    console.error("[IZIWATT-CONTOLLER] 2d canvas not supported");
}
if(url===undefined)alert("getJpegImageDataURI return undefined");
return url;

};

Reply to this email directly or view it on GitHubhttps://github.com/MrRio/jsPDF/issues/207 .

James Hall Director

Parallax

+44 113 322 6477 http://parall.ax/

Registered office: The Old Brewery, High Court, Leeds, LS2 7ES Registered in England no. 07430032 VAT No. 101 3405 84

pascalPimaia commented 10 years ago

Thx for the info In which context (operating system & browser type x version) was this PR used ?

sebastianperrone commented 10 years ago

Hi, I'm having exactly the same problem as pascalPimaia. I'm using phonegap (with cordova 3.4). I can generate pdf files, text is correctly exposed, but the image is corrupt. I use a similar proc to load image:

function loadImage(photoUrl, cbFunc, cbErr) {
    console.log("image chrome loading");
    var imgLogo = new Image();
    console.log("image created");
    imgLogo.onload = function() {
        var canvas = document.createElement('canvas');
        console.log("image canvas created");
        canvas.width = imgLogo.width;
        canvas.height = imgLogo.height;
        console.log("image size setted");
        var context = canvas.getContext('2d');
        context.drawImage(imgLogo, 0, 0);
        console.log("image context created and image drawn");
        var imgData = canvas.toDataURL('image/jpeg');
        console.log("image data: " + imgData);

        // added from: https://github.com/MrRio/jsPDF/issues/207
        if (imgData.slice(0,"data:image/jpeg".length) !== "data:image/jpeg")  {
            try {       
                var encoder = new JPEGEncoder();
                imgData = encoder.encode(context.getImageData(0, 0, imgLogo.width, imgLogo.height), 100);
            } catch(e) {
                alert("JPEGEncoder failed"+e);          
            }
        }
        cbFunc(imgData);
    };
    imgLogo.src = photoUrl;
    console.log("image order logo file: " + photoUrl);
}

As you can see I added the section to convert image data with jpeg encoder, in this case does not shown any image. The section where I call loadImage function is this:

function createOrder(cbFunc, cbErr) {
    console.log("create pdf object");
    pdf = new jsPDF("p", "mm", "a4")
    var srcFile = "../img/jpg.jpg";
    loadImage(srcFile, function(imgData) {
        try {
            // section A
            pdf.addImage(imgData, "JPG", 14, 28, 44, 18);
            console.log("image added to pdf");
            printText(rows); // function which add text (this works fine)
            $this.pdfOrder = pdf;
            console.log("before callback");
            cbFunc($this.pdfOrder);
        } catch (err) {
            alert(err);
            cbErr(err);
        }
    }, cbErr);
}

am I doing something wrong ? is there any solution to this problem ?

By the way, I test the following scenarios in the section comented "section A":

var imgData = "..."
var imgData = "..."

In these tests the image appears corrupted, is shown more gray or black and white, but never with the appropriate contents. All test were performed with android 4.2 (genymotion emulator and alcatel one touch idol phone). The SDK of jsPDF was 1.0.0 trunk.

Any ideas ?

diegocr commented 10 years ago

jsPDF 1.0.88 supports passing a canvas object directly to addImage(), so how about if you try that way instead?

I.e, replace:

var imgData = canvas.toDataURL('image/jpeg');
....
cbFunc(imgData);

By:

cbFunc(canvas);
sebastianperrone commented 10 years ago

Thanks for the alternative, but still show a corrupted image. I changed the code to this:

function loadImage(photoUrl, cbFunc, cbErr) {
    console.log("image chrome loading");
    var imgLogo = new Image();
    console.log("image created");
    imgLogo.onload = function() {
        var canvas = document.createElement('canvas');
        console.log("image canvas created");
        canvas.width = imgLogo.width;
        canvas.height = imgLogo.height;
        console.log("image size setted");
        cbFunc(canvas);
    };
    imgLogo.src = photoUrl;
    console.log("image order logo file: " + photoUrl);
}

The resulting image was this: ordercanvas

diegocr commented 10 years ago

You have removed the .drawImage() call too, which obviously you shouldn't do :)

Ie, add back (where it was):

    var context = canvas.getContext('2d');
    context.drawImage(imgLogo, 0, 0);
    console.log("image context created and image drawn");
sebastianperrone commented 10 years ago

Ups. Its true sorry. I fixed and try again, but still showing corrupted image. Also I try JPG and PNG params in jsPDF addImage function. The code:

function loadImage(photoUrl, cbFunc, cbErr) {
    console.log("image chrome loading");
    var imgLogo = new Image();
    console.log("image created");
    imgLogo.onload = function() {
        var canvas = document.createElement('canvas');
        console.log("image canvas created");
        canvas.width = imgLogo.width;
        canvas.height = imgLogo.height;
        console.log("image size setted");
        var context = canvas.getContext('2d');
        context.drawImage(imgLogo, 0, 0, imgLogo.width, imgLogo.height, 0, 0, imgLogo.width, imgLogo.height);
        console.log("image context created and image drawn");
        cbFunc(canvas);
    };
    imgLogo.src = photoUrl;
    console.log("image order logo file: " + photoUrl);
}

Resulting image: canvas2

diegocr commented 10 years ago

In fact you could omit the format parameter and it should work too given a canvas, ie pdf.addImage(imgData, 14, 28, 44, 18);

That said, are you sure imgLogo contains a valid image? post what this returns:

console.log(imgLogo.complete, imgLogo.width, imgLogo.height,
      imgLogo.naturalWidth, imgLogo.naturalHeight, imgLogo.clientWidth);

Edit: Use that within imgLogo.onload function.

sebastianperrone commented 10 years ago

I modified the function according to your instructions. Still show corrupted image, I placed all requested log. Image data URI, is truncated to 4K, the log function does not show more data, but, if you cut and paste in browser address field, shows the first part of the logo. Remember, in a computer browser all works fine (image surely are correct and consistent), also I tried with images jpg, png 32 and 24 of this link: https://github.com/MrRio/jsPDF/pull/195

04-11 22:14:06.600: I/Web Console(2460): image data: 

04-11 22:14:06.592: I/Web Console(2460): imgLogo.complete: true at file:///android_asset/www/orderingHistory/orderingHistory.js:714
04-11 22:14:06.592: I/Web Console(2460): imgLogo.width: 165 at file:///android_asset/www/orderingHistory/orderingHistory.js:715
04-11 22:14:06.592: I/Web Console(2460): imgLogo.height: 74 at file:///android_asset/www/orderingHistory/orderingHistory.js:716
04-11 22:14:06.592: I/Web Console(2460): imgLogo.naturalWidth: 165 at file:///android_asset/www/orderingHistory/orderingHistory.js:717
04-11 22:14:06.592: I/Web Console(2460): imgLogo.naturalHeight: 74 at file:///android_asset/www/orderingHistory/orderingHistory.js:718
sebastianperrone commented 10 years ago

I forgot post the used source code:

function getOrderItems() {
    console.log("create pdf object");
    pdf = new jsPDF("p", "mm", "a4")
    var srcFile = "../img/orderLogo.jpg";
    //var srcFile = "../img/32_bit.png";
    //var srcFile = "../img/24_bit.png";
    //var srcFile = "../img/png8_trans.png";
    //var srcFile = "../img/jpg.jpg";
    loadImage(srcFile, function(imgData) {
        try {
            console.log("init loading image: " + srcFile);

            //var imgData = "";
            //var imgData = "";
            //var imgData = "";
            //var imgData = "";

            //pdf.addImage(imgData, "PNG", 14, 28, 44, 18);
            pdf.addImage(imgData, 14, 28, 44, 18);

            console.log("image added to pdf");
            printText(rows); // print text data
            console.log("before callback");
            cbFunc($this.pdfOrder);
        } catch (err) {
            alert(JSON.stringify(err));
        }
    }, showErrors);
};

function loadImage(photoUrl, cbFunc, cbErr) {
    console.log("image chrome loading");
    var imgLogo = new Image();
    console.log("image created");
    imgLogo.onload = function() {
        console.log("imgLogo.complete: " + imgLogo.complete);
        console.log("imgLogo.width: " + imgLogo.width);
        console.log("imgLogo.height: " + imgLogo.height);
        console.log("imgLogo.naturalWidth: " + imgLogo.naturalWidth);
        console.log("imgLogo.naturalHeight: " + imgLogo.naturalHeight);
        var canvas = document.createElement('canvas');
        console.log("image canvas created");
        canvas.width = imgLogo.width;
        canvas.height = imgLogo.height;
        console.log("image size setted");
        var context = canvas.getContext('2d');
        context.drawImage(imgLogo, 0, 0, imgLogo.width, imgLogo.height);
        console.log("image context created and image drawn");
        var imgData = canvas.toDataURL('image/jpeg');
        console.log("image data: " + imgData);

        //cbFunc(imgData);
        cbFunc(canvas);
    };
    imgLogo.src = photoUrl;
    console.log("image order logo file: " + photoUrl);
}
diegocr commented 10 years ago

I'm nearly out of ideas, try using document.body.appendChild(canvas); after the .drawImage() call.

diegocr commented 10 years ago

Btw, did you tested if our live demos works there on Android?

http://mrrio.github.io/jsPDF/

Test the "Images" and "addHTML" examples.

jamesbrobb commented 10 years ago

I'm wondering if this issue may be related to endianness. The logic for png support was written on a mac (OSX 10.7.5), which is little endian.

You can use this gist to check the endianness of the andriod phone you're testing on.

diegocr commented 10 years ago

AFAICT, all ARM cpus are bi-endian, but the Android OS uses little-endian by default.

I think there are some weird limitations on the framework they're using, otherwise .toDataURL("image/jpeg") would return a proper JPEG.

jamesbrobb commented 10 years ago

Thanks Diego.

Yeah very true. Possibly a size limit on the returned data uri and it's being truncated, causing the image to corrupt when added to the PDF. If i remember correctly, there were size limitations on the early dataURL implementations for desktop browsers.

sebastianperrone commented 10 years ago

Thanks Diego and James for the info. Still have not been able to test the phonegap app on iOS, probably in the next few days I will. What I have been able to prove, is the page (http://mrrio.github.io/jsPDF) with the elementary jsPDF tests in iOS (iPad with iOS 7.1) and it worked perfectly; but with android did not work: android 2.2 show characters into iframe, android 3.2 did nothing, and android 4.2, in some devices proposed download the pdf file (and download ok), in the others did nothing too. When I get the results of phonegap on ios, I leave the result here.

pascalPimaia commented 10 years ago

I finaly could get a proper output and use it on android and iOS. The major difficulty was related to the encoding of the jsPDF.output function to save it into a file in the local file system.

Read this link it helped me to understand the problem: https://coderwall.com/p/nc8hia The solution is similar to the jsPDF.save function using Blob or ArrayBuffer. Warning Blobbuilder was not supported on my android device.

Here the code I used. The content is saved in buffer (ArrayBuffer class) before writing the file. In this example my class provides the pdf jsPDF.output by calling this.output() and PdfOutput is a global scope variable pointing ti this content.


PdfOutput   = this.output();    

if ( typeof window.requestFileSystem === "undefined") {
    alert("access to file system can only be performed on a real device : saving document for regular browser");
    //for debug on browser only 
    this.save(filename);        
    onSuccess && onSuccess();       
    return;
}
window.requestFileSystem(LocalFileSystem.PERSISTENT, 0, 
    // onRequestFileSystemReady
    function(fileSystem) 
    {
        getPdfDirectoryEntry(
            function(directoryEntry)
            {
                if(directoryEntry!==undefined)
                {
                    directoryEntry.getFile(filename, 
                    //fileSystem.root.getFile(filename, 
                        {create : true},
                        // onGetFileReady
                        function(entry)
                        {
                            // entry is a "file entry"
                            entry.createWriter(
                                // onWriterReady
                                function(writer) 
                                {
                                    writer.onwrite= onSuccess;
                                    writer.onerror= onError;                                                
                                        try
                                        {
                                            if( Blob )
                                            {
                                                // see https://coderwall.com/p/nc8hia
                                                var buffer = new ArrayBuffer(PdfOutput.length),
                                                    array = new Uint8Array(buffer);
                                                for (var i = 0; i < PdfOutput.length; i++) 
                                                {
                                                    array[i] = PdfOutput.charCodeAt(i);
                                                }                                           
                                                // see http://dev.w3.org/2006/webapi/FileAPI/#dfn-Blob
                                                writer.write(buffer);
                                            }
                                            else  
                                            {
                                                writer.write(PdfOutput); 
                                            }
                                        }
                                        catch(e)
                                        {
                                            onError(e);
                                        }                                                                               
                                },                              
                                onError //onWriterReady
                            );
                        },
                        onError //onGetFileReady
                    );//getFile(filename,
                }//if(directoryEntry !== undefined)
            },
            onError //onGetPdfDirectoryEntry
        );// getPdfDirectoryEntry
    },              
    onError //onRequestFileSystemError
);

writer is obtained using HTML5 createWriter() created from an entry the whole thing depending on requestFileSystem API.

diegocr commented 10 years ago

Thanks Pascal!

Well... let me digest that, there are two problems on cordova/phonegap:

I guess the first problem is the result of using the jspdf.js from the root of the repo, instead of one of the files available on the dist folder, because the jspdf.min.js file includes a Blob polyfill and therefore Blob should always be available, unless there's a problem with that polyfill, of course.

Moreover, the saveAs function we're using to save the PDF to disk comes from another shim/polyfill, FileSaver.js - This one already uses the FileSystem API (window.requestFileSystem) when it's available

So, can you please confirm which jsPDF were you using?

Then, there's the canvas problem, on the pointed link he's using window.canvasplugin instead, i think it's this one. Not much i can add here, if that thing were sync we could add it in addImage() as a workaround when it's available, so you guys will need to manually make use of it.

sebastianperrone commented 10 years ago

Thanks a lot Pascal! yes the key is the encoding. This data transformation that you posted, do the work:

var strPdf = pdf.output();
var buffer = new ArrayBuffer(strPdf.length);
var array = new Uint8Array(buffer);
for (var i = 0; i < strPdf.length; i++) {
    array[i] = strPdf.charCodeAt(i);
} 
writer.write(buffer);

With this add, is working fine on ios and android. Thanks a lot to all !

diegocr commented 10 years ago

Huh? so since you was using our latest version, that means none of the polyfills works there... but, the canvas element has a problem regardless i think... :confused:

Well, meh.

sebastianperrone commented 10 years ago

Yes I'm using your last version (v1.0.88), and the last posted loadImage function (passing the canvas object directly without image type). So yes, its very possible, that occurs that you say.

diegocr commented 10 years ago

Using 1.0.106 you can now do writer.write(pdf.output("arraybuffer"));

lvbeck commented 10 years ago

test on android 4.0.3 with jspdf-1.0.116 and window.canvasplugin, still not working.

also test jspdf 0.90, does not work with image either.

with 1.0.x version, I do like this: writer.write(doc.output("arraybuffer"));

with 0.9.0 version, I do like this: var output = doc.output(); var buffer = new ArrayBuffer(output.length); var array = new Uint8Array(buffer); for (var i = 0; i < output.length; i++) { array[i] = output.charCodeAt(i); }

writer.write( buffer );

here is the error log:

05-05 13:43:33.364: I/Web Console(12839): processMessage failed: Stack: TypeError: Function.prototype.apply: Arguments list has wrong type 05-05 13:43:33.364: I/Web Console(12839): at Function.APPLY_PREPARE (native) 05-05 13:43:33.364: I/Web Console(12839): at androidExec (file:///android_asset/www/scripts/libs/cordova/android/cordova.js:849:55) 05-05 13:43:33.364: I/Web Console(12839): at [object Object].write (file:///android_asset/www/scripts/libs/cordova/android/cordova.js:3103:5) 05-05 13:43:33.364: I/Web Console(12839): at file:///android_asset/www/scripts/controllers/ResultsCtrl.js:1177:28 05-05 13:43:33.364: I/Web Console(12839): at file:///android_asset/www/scripts/libs/cordova/android/cordova.js:2273:32 05-05 13:43:33.364: I/Web Console(12839): at file:///android_asset/www/scripts/libs/cordova/android/cordova.js:2287:9 05-05 13:43:33.364: I/Web Console(12839): at Object.callbackFromNative (file:///android_asset/www/scripts/libs/cordova/android/cordova.js:301:54) 05-05 13:43:33.364: I/Web Console(12839): at processMessage (file:///android_asset/www/scripts/libs/cordova/android/cordova.js:975:21) 05-05 13:43:33.364: I/Web Console(12839): at Function.processMessages (file:///android_asset/www/scripts/libs/cordova/android/cordova.js:1009:13) 05-05 13:43:33.364: I/Web Console(12839): at androidExec (file:///android_asset/www/scripts/libs/cordova/android/cordova.js:872:25) at file:///android_asset/www/scripts/libs/cordova/android/cordova.js:982

Originalin commented 10 years ago

I also have the above problem, can anyone solve it? I think it's the problem of cordova version

jonatasfreitasv commented 9 years ago

WOW its work @pascalPimaia !!!!!!!! THX A LOT MAN !!!

6trading commented 8 years ago

Thanks @pascalPimaia !!