Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to bridge a JavaScript (ragged) array and an std::vector<std::vector<T>> object?

In JavaScript I have a list of "lines", each of which consists of indefinite number of "points", each of which has a form of [x, y]. So it is a 3D ragged array. Now I need to pass it to my C++ code with the help from emscripten (embind). Here's declaration of the C++ function:

Eigen::MatrixXd f(const std::vector<std::vector<std::vector<double>>>& lines);

And I would like to get a list of lists ([[m11, m12],[m22, m22],...]) in JavaScript after calling f. How to write the binding code in this case (the stuff inside EMSCRIPTEN_BINDINGS, for example)?

UPDATE: I can now pass the JavaScript array to C++. The binding part is something like

typedef std::vector<double> Point;
typedef std::vector<Point> Line;
EMSCRIPTEN_BINDINGS(module) {
  register_vector<Line>("LineArray");
  register_vector<Point>("Line");
  register_vector<double>("Point");
  emscripten::function("f_wrapper", &f_wrapper);
}

where f_wrapper calls f but returns vector<vector<double>> instead of MatrixXd. Now the problem is that I can only get an empty JavaScript object after calling f_wrapper. The JavaScript is

var Module = require('./bind.js'); // the output of em++
var cppAllLines = new Module.LineArray();
// some initialization
var result = Module.f_wrapper(cppAllLines); // an empty "Line" object

Any ideas?

like image 720
ziyuang Avatar asked May 23 '26 19:05

ziyuang


1 Answers

When passing an embound vector from a C++ function, such as

std::vector<std::vector<double>> f_wrapper(...);

to Javascript

var result = Module.f_wrapper(...);

The result object is not a Javascript array that implements length property or array-indexed access, so it can appear "empty" if using these to access its data.

But it does expose get and size functions to access the vector:

console.log('Back from C++', result.size(), result.get(0).get(1));

(The double get is because we're returning a vector of vectors)

For completeness, looking into the prototype of the object returned, it seems to expose the following functions.

  • get
  • push_back
  • resize
  • set
  • size

Slightly inconsistently it exposes get and set functions rather than an equivalent of the C++ at function. I suspect that it's not possible for an exactly equivalent function since at returns a reference which allows it to be used as a setter, which isn't possible in Javascript.

like image 119
Michal Charemza Avatar answered May 26 '26 07:05

Michal Charemza



Donate For Us

If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!