0

So basically I'm trying to send WebGL frames/screenshots to a server which may save these frames to the hard drive so I'll be able to merge these saved frames to video file. I have found this: Exporting video from webgl

I don't like to bother you guys with the WebGL related details... The only thing you should know that I would like to send the content of two client side JavaScript variables to the localhost node.js server. So these two client side variables are the following:

            var frame = frame + 1; //this is the frame counter
            var dataUrl = renderer.domElement.toDataURL("image/png"); //this is the encoded screenshot/frame

the receiver server side code is given by Joe Turner (this is running in node.js). So my question is: how to send the above mentioned two variable's content to this node.js code?

var http = require('http');
var querystring = require('querystring');
var fs = require('fs');

// Override so we don't decode spaces, and mess up the base64 encoding
querystring.unescape = function(s, decodeSpaces) {
    return s;
};

// Pad to follow the processing export format
function pad(num) {
    var s = "000" + num;
    return s.substr(s.length-4);
}

http.createServer(function (request, response) {
    request.content = '';
    request.addListener("data", function(data) {
        request.content += data;
    });

    request.addListener("end", function() {
        if (request.content.trim()) {
            request.content = querystring.parse(request.content);
            var data = unescape(request.content['data']);
            var frame = request.content['frame'];
            // Remove data:image/png;base64,
            data = data.substr(data.indexOf(',') + 1);
            var buffer = new Buffer(data, 'base64');
            fs.writeFile('screen-' + pad(frame) + '.png',
                         buffer.toString('binary'), 'binary');
        }
    });
    response.writeHead(200, {
        'Content-Type': 'text/plain',
        'Access-Control-Allow-Origin': '*',
        'Access-Control-Allow-Headers': 'X-Requested-With'
    });
    response.end();
}).listen(8080, "127.0.0.1");

Should I do something like this?

            ...
            ...
            //WebGL related code
            ...
            ...
                            var frame = frame + 1;
                            var dataUrl = renderer.domElement.toDataURL("image/png");

                                $.ajax({
                                    url: "http://127.0.0.1:8080",
                                    type: "POST",
                                    data: {"data": dataUrl, "frame": frame}

                                });
            ...
            ...

For example let's see this experiment:

Three.js basic example

I changed this piece of code in the above experiment:

        pointLight.position.copy( camera.position );

        progress += frameTime;
        lastTimestamp = timestamp;
        renderer.render( scene, camera );
        stats.update();
        requestAnimationFrame( animate );

To this:

        pointLight.position.copy( camera.position );

        progress += frameTime;
        lastTimestamp = timestamp;
        renderer.render( scene, camera );
        stats.update();
        requestAnimationFrame( animate );

        var dataUrl = "";
        var frame = 10;

        $.ajax({
                url: "http://127.0.0.1:8080",
                type: "POST",
                data: {"data": dataUrl, "frame": frame}    
        });

But still cannot see any saved image file.

jbduzan
  • 1,106
  • 1
  • 14
  • 30
Fract
  • 323
  • 1
  • 6
  • 22
  • possible duplicate of [Pass array to ajax request in $.ajax()](http://stackoverflow.com/questions/8890524/pass-array-to-ajax-request-in-ajax) – bwright Jul 10 '15 at 11:22

1 Answers1

1

The code you wrote only needs to have the correct protocol in its url (you were missing "http://") :

var frame = frame + 1;
var dataUrl = renderer.domElement.toDataURL("image/png");

$.ajax({
        url: "http://127.0.0.1:8080",
        type: "POST",
        data: {"data": dataUrl, "frame": frame}    
});

From the code you posted, you can see the server is already parsing the request and extracting the two parameters you send in POST

if (request.content.trim()) {    
    request.content = querystring.parse(request.content);
    var data = request.content['data'];
    var frame = request.content['frame'];
    //...

Anything you add to the POST request in $.ajax can be retrieved by name when calling

request.content['name']

Just make sure the names you use in the client and server code match.

Also, the node.js code is receiving an escaped base64 string, which doesn't make this function work

data.substr(data.indexOf(',') + 1);

Since the ',' character is escaped.

To solve this just use the node built-in function unescape on the data string like this

var data = unescape(request.content['data']);
Vale
  • 1,912
  • 16
  • 22
  • It's fine, corrected my code, but node.js still not displaying anything or does not save any picture. Where should the pictures to be saved by node.js? – Fract Jul 10 '15 at 13:21
  • 1
    I updated my answer, after unescaping the data variable the server saves images correctly. The pictures get saved in the directory where you **execute** node.js. If you want to use a specific directory just add it before the filename `fs.writeFile(directoryPath + 'screen-' + pad(frame) + '.png', ...)` – Vale Jul 10 '15 at 14:00
  • Made these changes you suggested, but still no image on the directory path I've specified. I'm a newcomer to node.js. So just please doublecheck how to start the node.js with the script above. I copied and pasted the above node.js related code into a text file (xy.txt), then went to cmd (where xy.txt takes place) and typed "node xy.txt". I only see a black screen. No image created. – Fract Jul 10 '15 at 15:09
  • 1
    That's correct, you don't see any output on the server. After you start it with cmd you need to execute `$.ajax` from your javascript and send the parameters **while the server is still running**. If you're unsure try this [snippet](http://pastebin.com/2EsSuket) I was using to test your code right from your browser developer's console, it requires jquery, of course. – Vale Jul 10 '15 at 15:40
  • Please see: I've updated my original question. Unescaped the node.js code, tested all your codes. Ran a native three.js example with your code included meanwhile the node.js code was running. Still no saved image. May I ask you please to download the three.js experiment I just linked in my original question and implement the AJAX part into it? May I ask you please to give it a try on your side? – Fract Jul 13 '15 at 09:05