Show:
#include "linestring_points.hpp"
#include "../gdal_common.hpp"
#include "../gdal_geometry.hpp"
#include "../gdal_linestring.hpp"
#include "../gdal_point.hpp"

namespace node_gdal {

Nan::Persistent<FunctionTemplate> LineStringPoints::constructor;

void LineStringPoints::Initialize(Local<Object> target) {
  Nan::HandleScope scope;

  Local<FunctionTemplate> lcons = Nan::New<FunctionTemplate>(LineStringPoints::New);
  lcons->InstanceTemplate()->SetInternalFieldCount(1);
  lcons->SetClassName(Nan::New("LineStringPoints").ToLocalChecked());

  Nan::SetPrototypeMethod(lcons, "toString", toString);
  Nan::SetPrototypeMethod(lcons, "count", count);
  Nan::SetPrototypeMethod(lcons, "get", get);
  Nan::SetPrototypeMethod(lcons, "set", set);
  Nan::SetPrototypeMethod(lcons, "add", add);
  Nan::SetPrototypeMethod(lcons, "reverse", reverse);
  Nan::SetPrototypeMethod(lcons, "resize", resize);

  Nan::Set(target, Nan::New("LineStringPoints").ToLocalChecked(), Nan::GetFunction(lcons).ToLocalChecked());

  constructor.Reset(lcons);
}

LineStringPoints::LineStringPoints() : Nan::ObjectWrap() {
}

LineStringPoints::~LineStringPoints() {
}

/**
 * An encapsulation of a {{#crossLink
 * "gdal.LineString"}}LineString{{/crossLink}}'s points.
 *
 * @class gdal.LineStringPoints
 */
NAN_METHOD(LineStringPoints::New) {
  Nan::HandleScope scope;

  if (!info.IsConstructCall()) {
    Nan::ThrowError("Cannot call constructor as function, you need to use 'new' keyword");
    return;
  }
  if (info[0]->IsExternal()) {
    Local<External> ext = info[0].As<External>();
    void *ptr = ext->Value();
    LineStringPoints *geom = static_cast<LineStringPoints *>(ptr);
    geom->Wrap(info.This());
    info.GetReturnValue().Set(info.This());
    return;
  } else {
    Nan::ThrowError("Cannot create LineStringPoints directly");
    return;
  }
}

Local<Value> LineStringPoints::New(Local<Value> geom) {
  Nan::EscapableHandleScope scope;

  LineStringPoints *wrapped = new LineStringPoints();

  v8::Local<v8::Value> ext = Nan::New<External>(wrapped);
  v8::Local<v8::Object> obj =
    Nan::NewInstance(Nan::GetFunction(Nan::New(LineStringPoints::constructor)).ToLocalChecked(), 1, &ext)
      .ToLocalChecked();
  Nan::SetPrivate(obj, Nan::New("parent_").ToLocalChecked(), geom);

  return scope.Escape(obj);
}

NAN_METHOD(LineStringPoints::toString) {
  Nan::HandleScope scope;
  info.GetReturnValue().Set(Nan::New("LineStringPoints").ToLocalChecked());
}

/**
 * Returns the number of points that are part of the line string.
 *
 * @method count
 * @return {Integer}
 */
NAN_METHOD(LineStringPoints::count) {
  Nan::HandleScope scope;

  Local<Object> parent =
    Nan::GetPrivate(info.This(), Nan::New("parent_").ToLocalChecked()).ToLocalChecked().As<Object>();
  LineString *geom = Nan::ObjectWrap::Unwrap<LineString>(parent);

  info.GetReturnValue().Set(Nan::New<Integer>(geom->get()->getNumPoints()));
}

/**
 * Reverses the order of all the points.
 *
 * @method reverse
 */
NAN_METHOD(LineStringPoints::reverse) {
  Nan::HandleScope scope;

  Local<Object> parent =
    Nan::GetPrivate(info.This(), Nan::New("parent_").ToLocalChecked()).ToLocalChecked().As<Object>();
  LineString *geom = Nan::ObjectWrap::Unwrap<LineString>(parent);

  geom->get()->reversePoints();

  return;
}

/**
 * Adjusts the number of points that make up the line string.
 *
 * @method resize
 * @param {Integer} count
 */
NAN_METHOD(LineStringPoints::resize) {
  Nan::HandleScope scope;

  Local<Object> parent =
    Nan::GetPrivate(info.This(), Nan::New("parent_").ToLocalChecked()).ToLocalChecked().As<Object>();
  LineString *geom = Nan::ObjectWrap::Unwrap<LineString>(parent);

  int count;
  NODE_ARG_INT(0, "point count", count)
  geom->get()->setNumPoints(count);

  return;
}

/**
 * Returns the point at the specified index.
 *
 * @method get
 * @param {Integer} index 0-based index
 * @return {gdal.Point}
 */
NAN_METHOD(LineStringPoints::get) {
  Nan::HandleScope scope;

  Local<Object> parent =
    Nan::GetPrivate(info.This(), Nan::New("parent_").ToLocalChecked()).ToLocalChecked().As<Object>();
  LineString *geom = Nan::ObjectWrap::Unwrap<LineString>(parent);

  OGRPoint *pt = new OGRPoint();
  int i;

  NODE_ARG_INT(0, "index", i);
  if (i < 0 || i >= geom->get()->getNumPoints()) {
    info.GetReturnValue().Set(Nan::Null());
    return;
  }

  geom->get()->getPoint(i, pt);

  info.GetReturnValue().Set(Point::New(pt));
}

/**
 * Sets the point at the specified index.
 *
 * @example
 * ```
 * lineString.points.set(0, new gdal.Point(1, 2));```
 *
 * @method set
 * @throws Error
 * @param {Integer} index 0-based index
 * @param {gdal.Point} point
 */
NAN_METHOD(LineStringPoints::set) {
  Nan::HandleScope scope;

  Local<Object> parent =
    Nan::GetPrivate(info.This(), Nan::New("parent_").ToLocalChecked()).ToLocalChecked().As<Object>();
  LineString *geom = Nan::ObjectWrap::Unwrap<LineString>(parent);

  int i;
  NODE_ARG_INT(0, "index", i);
  if (i < 0 || i >= geom->get()->getNumPoints()) {
    Nan::ThrowError("Point index out of range");
    return;
  }

  int n = info.Length() - 1;

  if (n == 0) {
    Nan::ThrowError("Point must be given");
    return;
  } else if (n == 1) {
    if (!info[1]->IsObject()) {
      Nan::ThrowError("Point or object expected for second argument");
      return;
    }
    if (IS_WRAPPED(info[1], Point)) {
      // set from Point object
      Point *pt = Nan::ObjectWrap::Unwrap<Point>(info[1].As<Object>());
      geom->get()->setPoint(i, pt->get());
    } else {
      Local<Object> obj = info[1].As<Object>();
      // set from object {x: 0, y: 5}
      double x, y;
      NODE_DOUBLE_FROM_OBJ(obj, "x", x);
      NODE_DOUBLE_FROM_OBJ(obj, "y", y);

      Local<String> z_prop_name = Nan::New("z").ToLocalChecked();
      if (Nan::HasOwnProperty(obj, z_prop_name).FromMaybe(false)) {
        Local<Value> z_val = Nan::Get(obj, z_prop_name).ToLocalChecked();
        if (!z_val->IsNumber()) {
          Nan::ThrowError("z property must be number");
          return;
        }
        geom->get()->setPoint(i, x, y, Nan::To<double>(z_val).ToChecked());
      } else {
        geom->get()->setPoint(i, x, y);
      }
    }
  } else {
    // set x, y, z from numeric arguments
    if (!info[1]->IsNumber()) {
      Nan::ThrowError("Number expected for second argument");
      return;
    }
    if (!info[2]->IsNumber()) {
      Nan::ThrowError("Number expected for third argument");
      return;
    }
    if (n == 2) {
      geom->get()->setPoint(i, Nan::To<double>(info[1]).ToChecked(), Nan::To<double>(info[2]).ToChecked());
    } else {
      if (!info[3]->IsNumber()) {
        Nan::ThrowError("Number expected for fourth argument");
        return;
      }

      geom->get()->setPoint(
        i,
        Nan::To<double>(info[1]).ToChecked(),
        Nan::To<double>(info[2]).ToChecked(),
        Nan::To<double>(info[3]).ToChecked());
    }
  }

  return;
}

/**
 * Adds point(s) to the line string. Also accepts any object with an x and y
 * property.
 *
 * @example
 * ```
 * lineString.points.add(new gdal.Point(1, 2));
 * lineString.points.add([
 *     new gdal.Point(1, 2)
 *     new gdal.Point(3, 4)
 * ]);```
 *
 * @method add
 * @throws Error
 * @param {gdal.Point|object|Array} point(s)
 */
NAN_METHOD(LineStringPoints::add) {
  Nan::HandleScope scope;

  Local<Object> parent =
    Nan::GetPrivate(info.This(), Nan::New("parent_").ToLocalChecked()).ToLocalChecked().As<Object>();
  LineString *geom = Nan::ObjectWrap::Unwrap<LineString>(parent);

  int n = info.Length();

  if (n == 0) {
    Nan::ThrowError("Point must be given");
    return;
  } else if (n == 1) {
    if (!info[0]->IsObject()) {
      Nan::ThrowError("Point, object, or array of points expected");
      return;
    }
    if (IS_WRAPPED(info[0], Point)) {
      // set from Point object
      Point *pt = Nan::ObjectWrap::Unwrap<Point>(info[0].As<Object>());
      geom->get()->addPoint(pt->get());
    } else if (info[0]->IsArray()) {
      // set from array of points
      Local<Array> array = info[0].As<Array>();
      int length = array->Length();
      for (int i = 0; i < length; i++) {
        Local<Value> element = Nan::Get(array, i).ToLocalChecked();
        if (!element->IsObject()) {
          Nan::ThrowError("All points must be Point objects or objects");
          return;
        }
        Local<Object> element_obj = element.As<Object>();
        if (IS_WRAPPED(element_obj, Point)) {
          // set from Point object
          Point *pt = Nan::ObjectWrap::Unwrap<Point>(element_obj);
          geom->get()->addPoint(pt->get());
        } else {
          // set from object {x: 0, y: 5}
          double x, y;
          NODE_DOUBLE_FROM_OBJ(element_obj, "x", x);
          NODE_DOUBLE_FROM_OBJ(element_obj, "y", y);

          Local<String> z_prop_name = Nan::New("z").ToLocalChecked();
          if (Nan::HasOwnProperty(element_obj, z_prop_name).FromMaybe(false)) {
            Local<Value> z_val = Nan::Get(element_obj, z_prop_name).ToLocalChecked();
            if (!z_val->IsNumber()) {
              Nan::ThrowError("z property must be number");
              return;
            }
            geom->get()->addPoint(x, y, Nan::To<double>(z_val).ToChecked());
          } else {
            geom->get()->addPoint(x, y);
          }
        }
      }
    } else {
      // set from object {x: 0, y: 5}
      Local<Object> obj = info[0].As<Object>();
      double x, y;
      NODE_DOUBLE_FROM_OBJ(obj, "x", x);
      NODE_DOUBLE_FROM_OBJ(obj, "y", y);

      Local<String> z_prop_name = Nan::New("z").ToLocalChecked();
      if (Nan::HasOwnProperty(obj, z_prop_name).FromMaybe(false)) {
        Local<Value> z_val = Nan::Get(obj, z_prop_name).ToLocalChecked();
        if (!z_val->IsNumber()) {
          Nan::ThrowError("z property must be number");
          return;
        }
        geom->get()->addPoint(x, y, Nan::To<double>(z_val).ToChecked());
      } else {
        geom->get()->addPoint(x, y);
      }
    }
  } else {
    // set x, y, z from numeric arguments
    if (!info[0]->IsNumber()) {
      Nan::ThrowError("Number expected for first argument");
      return;
    }
    if (!info[1]->IsNumber()) {
      Nan::ThrowError("Number expected for second argument");
      return;
    }
    if (n == 2) {
      geom->get()->addPoint(Nan::To<double>(info[0]).ToChecked(), Nan::To<double>(info[1]).ToChecked());
    } else {
      if (!info[2]->IsNumber()) {
        Nan::ThrowError("Number expected for third argument");
        return;
      }

      geom->get()->addPoint(
        Nan::To<double>(info[0]).ToChecked(),
        Nan::To<double>(info[1]).ToChecked(),
        Nan::To<double>(info[2]).ToChecked());
    }
  }

  return;
}

} // namespace node_gdal