#include "gdal_common.hpp"
#include "gdal_feature.hpp"
#include "gdal_feature_defn.hpp"
#include "gdal_geometry.hpp"
#include "gdal_field_defn.hpp"
#include "gdal_layer.hpp"
#include "collections/feature_fields.hpp"
namespace node_gdal {
Nan::Persistent<FunctionTemplate> Feature::constructor;
void Feature::Initialize(Local<Object> target)
{
Nan::HandleScope scope;
Local<FunctionTemplate> lcons = Nan::New<FunctionTemplate>(Feature::New);
lcons->InstanceTemplate()->SetInternalFieldCount(1);
lcons->SetClassName(Nan::New("Feature").ToLocalChecked());
Nan::SetPrototypeMethod(lcons, "toString", toString);
Nan::SetPrototypeMethod(lcons, "getGeometry", getGeometry);
//Nan::SetPrototypeMethod(lcons, "setGeometryDirectly", setGeometryDirectly);
Nan::SetPrototypeMethod(lcons, "setGeometry", setGeometry);
// Nan::SetPrototypeMethod(lcons, "stealGeometry", stealGeometry);
Nan::SetPrototypeMethod(lcons, "clone", clone);
//Nan::SetPrototypeMethod(lcons, "equals", equals);
//Nan::SetPrototypeMethod(lcons, "getFieldDefn", getFieldDefn); (use defn.fields.get() instead)
Nan::SetPrototypeMethod(lcons, "setFrom", setFrom);
//Note: We should let node GC handle destroying features when they arent being used
//TODO: Give node more info on the amount of memory a feature is using
// Nan::AdjustExternalMemory()
//Nan::SetPrototypeMethod(lcons, "destroy", destroy);
ATTR(lcons, "fields", fieldsGetter, READ_ONLY_SETTER);
ATTR(lcons, "defn", defnGetter, READ_ONLY_SETTER);
ATTR(lcons, "fid", fidGetter, fidSetter);
Nan::Set(target, Nan::New("Feature").ToLocalChecked(), Nan::GetFunction(lcons).ToLocalChecked());
constructor.Reset(lcons);
}
Feature::Feature(OGRFeature *feature)
: Nan::ObjectWrap(),
this_(feature),
owned_(true)
{
LOG("Created Feature[%p]", feature);
}
Feature::Feature()
: Nan::ObjectWrap(),
this_(0),
owned_(true)
{
}
Feature::~Feature()
{
dispose();
}
void Feature::dispose()
{
if(this_) {
LOG("Disposing Feature [%p] (%s)", this_, owned_ ? "owned" : "unowned");
if(owned_) OGRFeature::DestroyFeature(this_);
LOG("Disposed Feature [%p]", this_);
this_ = NULL;
}
}
/**
* A simple feature, including geometry and attributes. Its fields and geometry type is defined by the given definition.
*
* ```
* //create layer and specify geometry type
* var layer = dataset.layers.create('mylayer', null, gdal.Point);
*
* //setup fields for the given layer
* layer.fields.add(new gdal.FieldDefn('elevation', gdal.OFTInteger));
* layer.fields.add(new gdal.FieldDefn('name', gdal.OFTString));
*
* //create feature using layer definition and then add it to the layer
* var feature = new gdal.Feature(layer);
* feature.fields.set('elevation', 13775);
* feature.fields.set('name', 'Grand Teton');
* feature.setGeometry(new gdal.Point(43.741208, -110.802414));
* layer.features.add(feature);```
*
* @constructor
* @class gdal.Feature
* @param {gdal.Layer|gdal.FeatureDefn} definition
*/
NAN_METHOD(Feature::New)
{
Nan::HandleScope scope;
Feature* f;
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();
f = static_cast<Feature *>(ptr);
} else {
if (info.Length() < 1) {
Nan::ThrowError("Constructor expects Layer or FeatureDefn object");
return;
}
OGRFeatureDefn *def;
if (IS_WRAPPED(info[0], Layer)) {
Layer *layer = Nan::ObjectWrap::Unwrap<Layer>(info[0].As<Object>());
if (!layer->isAlive()) {
Nan::ThrowError("Layer object already destroyed");
return;
}
def = layer->get()->GetLayerDefn();
} else if(IS_WRAPPED(info[0], FeatureDefn)) {
FeatureDefn *feature_def = Nan::ObjectWrap::Unwrap<FeatureDefn>(info[0].As<Object>());
if (!feature_def->isAlive()) {
Nan::ThrowError("FeatureDefn object already destroyed");
return;
}
def = feature_def->get();
} else {
Nan::ThrowError("Constructor expects Layer or FeatureDefn object");
return;
}
OGRFeature *ogr_f = new OGRFeature(def);
f = new Feature(ogr_f);
}
Local<Value> fields = FeatureFields::New(info.This());
Nan::SetPrivate(info.This(), Nan::New("fields_").ToLocalChecked(), fields);
f->Wrap(info.This());
info.GetReturnValue().Set(info.This());
}
Local<Value> Feature::New(OGRFeature *feature)
{
Nan::EscapableHandleScope scope;
return scope.Escape(Feature::New(feature, true));
}
Local<Value> Feature::New(OGRFeature *feature, bool owned)
{
Nan::EscapableHandleScope scope;
if (!feature) {
return scope.Escape(Nan::Null());
}
Feature *wrapped = new Feature(feature);
wrapped->owned_ = owned;
Local<Value> ext = Nan::New<External>(wrapped);
Local<Object> obj = Nan::NewInstance(Nan::GetFunction(Nan::New(Feature::constructor)).ToLocalChecked(), 1, &ext).ToLocalChecked();
return scope.Escape(obj);
}
NAN_METHOD(Feature::toString)
{
Nan::HandleScope scope;
info.GetReturnValue().Set(Nan::New("Feature").ToLocalChecked());
}
/**
* Returns the geometry of the feature.
*
* @method getGeometry
* @return {gdal.Geometry}
*/
NAN_METHOD(Feature::getGeometry)
{
Nan::HandleScope scope;
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
OGRGeometry* geom = feature->this_->GetGeometryRef();
if (!geom) {
info.GetReturnValue().Set(Nan::Null());
return;
}
info.GetReturnValue().Set(Geometry::New(geom, false));
}
/**
* Returns the definition of a particular field at an index.
*
* @method getFieldDefn
* @param {Integer} index Field index (0-based)
* @return {gdal.FieldDefn}
*/
NAN_METHOD(Feature::getFieldDefn)
{
Nan::HandleScope scope;
int field_index;
NODE_ARG_INT(0, "field index", field_index);
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
if (field_index < 0 || field_index >= feature->this_->GetFieldCount()) {
Nan::ThrowError("Invalid field index");
return;
}
info.GetReturnValue().Set(FieldDefn::New(feature->this_->GetFieldDefnRef(field_index), false));
}
//NODE_WRAPPED_METHOD_WITH_RESULT(Feature, stealGeometry, Geometry, StealGeometry);
/**
* Sets the feature's geometry.
*
* @throws Error
* @method setGeometry
* @param {gdal.Geometry} geometry
*/
NAN_METHOD(Feature::setGeometry)
{
Nan::HandleScope scope;
Geometry *geom = NULL;
NODE_ARG_WRAPPED_OPT(0, "geometry", Geometry, geom);
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
OGRErr err = feature->this_->SetGeometry(geom ? geom->get() : NULL);
if(err){
NODE_THROW_OGRERR(err);
}
return;
}
/**
* Determines if the features are the same.
*
* @method equals
* @param {gdal.Feature} feature
* @return {Boolean} `true` if the features are the same, `false` if different
*/
NODE_WRAPPED_METHOD_WITH_RESULT_1_WRAPPED_PARAM(Feature, equals, Boolean, Equal, Feature, "feature");
/**
* Clones the feature.
*
* @method clone
* @return {gdal.Feature}
*/
NAN_METHOD(Feature::clone)
{
Nan::HandleScope scope;
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
info.GetReturnValue().Set(Feature::New(feature->this_->Clone()));
}
/**
* Releases the feature from memory.
*
* @method destroy
*/
NAN_METHOD(Feature::destroy)
{
Nan::HandleScope scope;
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
feature->dispose();
return;
}
/**
* Set one feature from another. Overwrites the contents of this feature
* from the geometry and attributes of another.
*
* @example
* ```
* var feature1 = new gdal.Feature(defn);
* var feature2 = new gdal.Feature(defn);
* feature1.setGeometry(new gdal.Point(5, 10));
* feature1.fields.set([5, 'test', 3.14]);
* feature2.setFrom(feature1);```
*
* @throws Error
* @method setFrom
* @param {gdal.Feature} feature
* @param {Array} [*index_map] Array of the indices (integers) of the feature's fields stored at the corresponding index of the source feature's fields. A value of -1 should be used to ignore the source's field. The array should not be `null` and be as long as the number of fields in the source feature.
* @param {Boolean} [forgiving=true] `true` if the operation should continue despite lacking output fields matching some of the source fields.
*/
NAN_METHOD(Feature::setFrom)
{
Nan::HandleScope scope;
Feature *other_feature;
int forgiving = 1;
Local<Array> index_map;
OGRErr err = 0;
NODE_ARG_WRAPPED(0, "feature", Feature, other_feature);
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
if (info.Length() <= 2) {
NODE_ARG_BOOL_OPT(1, "forgiving", forgiving);
err = feature->this_->SetFrom(other_feature->this_, forgiving ? TRUE : FALSE);
} else {
NODE_ARG_ARRAY(1, "index map", index_map);
NODE_ARG_BOOL_OPT(2, "forgiving", forgiving);
if (index_map->Length() < 1) {
Nan::ThrowError("index map must contain at least 1 index");
return;
}
int *index_map_ptr = new int[index_map->Length()];
for (unsigned index = 0; index < index_map->Length(); index++) {
Local<Value> field_index(Nan::Get(index_map, Nan::New<Integer>(index)).ToLocalChecked());
if (!field_index->IsUint32()) {
delete [] index_map_ptr;
Nan::ThrowError("index map must contain only integer values");
return;
}
int val = (int)Nan::To<uint32_t>(field_index).ToChecked(); //todo: validate index? perhaps ogr already does this and throws an error
index_map_ptr[index] = val;
}
err = feature->this_->SetFrom(other_feature->this_, index_map_ptr, forgiving ? TRUE : FALSE);
delete [] index_map_ptr;
}
if(err) {
NODE_THROW_OGRERR(err);
return;
}
return;
}
/**
* @readOnly
* @attribute fields
* @type {gdal.FeatureFields}
*/
NAN_GETTER(Feature::fieldsGetter)
{
Nan::HandleScope scope;
info.GetReturnValue().Set(Nan::GetPrivate(info.This(), Nan::New("fields_").ToLocalChecked()).ToLocalChecked());
}
/**
* @attribute fid
* @type {Number}
*/
NAN_GETTER(Feature::fidGetter)
{
Nan::HandleScope scope;
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
info.GetReturnValue().Set(Nan::New<Number>(feature->this_->GetFID()));
}
/**
* @readOnly
* @attribute defn
* @type {gdal.FeatureDefn}
*/
NAN_GETTER(Feature::defnGetter)
{
Nan::HandleScope scope;
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
info.GetReturnValue().Set(FeatureDefn::New(feature->this_->GetDefnRef(), false));
}
NAN_SETTER(Feature::fidSetter)
{
Nan::HandleScope scope;
Feature *feature = Nan::ObjectWrap::Unwrap<Feature>(info.This());
if (!feature->isAlive()) {
Nan::ThrowError("Feature object already destroyed");
return;
}
if(!value->IsInt32()){
Nan::ThrowError("fid must be an integer");
return;
}
feature->this_->SetFID(Nan::To<int64_t>(value).ToChecked());
}
} // namespace node_gdal