diff --git a/binding.gyp b/binding.gyp index 5e042f1b9..af9c0e399 100644 --- a/binding.gyp +++ b/binding.gyp @@ -66,6 +66,12 @@ "cc/modules/face/EigenFaceRecognizer.cc", "cc/modules/face/FisherFaceRecognizer.cc", "cc/modules/face/LBPHFaceRecognizer.cc", + "cc/modules/face/Facemark.cc", + "cc/modules/face/FacemarkAAM.cc", + "cc/modules/face/FacemarkAAMData.cc", + "cc/modules/face/FacemarkAAMParams.cc", + "cc/modules/face/FacemarkLBF.cc", + "cc/modules/face/FacemarkLBFParams.cc", "cc/modules/text/text.cc", "cc/modules/text/OCRHMMClassifier.cc", "cc/modules/text/OCRHMMDecoder.cc", diff --git a/cc/macros.h b/cc/macros.h index 387d2248e..e7a542614 100644 --- a/cc/macros.h +++ b/cc/macros.h @@ -9,6 +9,16 @@ #define FF_GETTER(clazz, name, prop) \ NAN_GETTER(name) { info.GetReturnValue().Set(Nan::ObjectWrap::Unwrap(info.This())->prop); } +#define FF_GETTER_SIMPLE(clazz, name, prop, converter) \ + NAN_GETTER(name) { \ + v8::Local jsValue = converter::wrap( \ + Nan::ObjectWrap::Unwrap(info.This())->prop \ + ); \ + info.GetReturnValue().Set(jsValue); \ + } + +#define FF_GETTER_COMPLEX(clazz, name, prop, converter) FF_GETTER_SIMPLE(clazz, name, prop, converter) + #define FF_GETTER_JSOBJ(clazz, name, value, unwrapper, ctor) \ NAN_GETTER(name) { \ v8::Local jsObj = FF_NEW_INSTANCE(ctor); \ @@ -96,4 +106,20 @@ static FF_FUNC_TYPE ff_func = FF_FUNC_TYPE(); #define FF_SETTER_NUMBER(clazz, name, prop) FF_SETTER(clazz, name, prop, ff_number) #define FF_SETTER_BOOL(clazz, name, prop) FF_SETTER(clazz, name, prop, ff_bool) -#endif \ No newline at end of file +#define FF_SETTER_SIMPLE(clazz, name, prop, converter) \ + NAN_SETTER(name##Set) { \ + FF_METHOD_CONTEXT(#name); \ + Nan::ObjectWrap::Unwrap(info.This())->prop = converter::unwrap( \ + value \ + ); \ + } + +#define FF_SETTER_COMPLEX(clazz, name, prop, type, converter) \ + NAN_SETTER(name##Set) { \ + FF_METHOD_CONTEXT(#name); \ + type target; \ + converter::unwrap(&target, value); \ + Nan::ObjectWrap::Unwrap(info.This())->prop = target; \ + } + +#endif diff --git a/cc/modules/face/Facemark.cc b/cc/modules/face/Facemark.cc new file mode 100644 index 000000000..70d478fd7 --- /dev/null +++ b/cc/modules/face/Facemark.cc @@ -0,0 +1,174 @@ +#ifdef HAVE_FACE + +#include "Facemark.h" +#include "FacemarkBindings.h" + +#if CV_VERSION_MINOR >= 4 + +NAN_METHOD(Facemark::Save) { + FF_METHOD_CONTEXT("Facemark::Save"); + FF_ARG_STRING(0, std::string path); + FF_UNWRAP(info.This(), Facemark)->save(path); +} + +NAN_METHOD(Facemark::Load) { + FF_METHOD_CONTEXT("Facemark::Load"); + FF_ARG_STRING(0, std::string path); + FF_UNWRAP(info.This(), Facemark)->load(path); +} + +void Facemark::Init(v8::Local ctor) { + Nan::SetPrototypeMethod(ctor, "addTrainingSample", AddTrainingSample); + Nan::SetPrototypeMethod(ctor, "addTrainingSampleAsync", + AddTrainingSampleAsync); + Nan::SetPrototypeMethod(ctor, "loadModel", LoadModel); + Nan::SetPrototypeMethod(ctor, "loadModelAsync", LoadModelAsync); + Nan::SetPrototypeMethod(ctor, "getData", GetData); + Nan::SetPrototypeMethod(ctor, "getDataAsync", GetDataAsync); + Nan::SetPrototypeMethod(ctor, "getFaces", GetFaces); + Nan::SetPrototypeMethod(ctor, "getFacesAsync", GetFacesAsync); + Nan::SetPrototypeMethod(ctor, "setFaceDetector", SetFaceDetector); + Nan::SetPrototypeMethod(ctor, "training", Training); + Nan::SetPrototypeMethod(ctor, "trainingAsync", TrainingAsync); + Nan::SetPrototypeMethod(ctor, "fit", Fit); + Nan::SetPrototypeMethod(ctor, "fitAsync", FitAsync); + Nan::SetPrototypeMethod(ctor, "save", Save); + Nan::SetPrototypeMethod(ctor, "load", Load); +}; + +NAN_METHOD(Facemark::AddTrainingSample) { + FF::SyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::AddTrainingSample", + info + ); +} + +NAN_METHOD(Facemark::AddTrainingSampleAsync) { + FF::AsyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::AddTrainingSampleAsync", + info + ); +} + +NAN_METHOD(Facemark::LoadModel) { + FF::SyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::LoadModel", + info + ); +} + +NAN_METHOD(Facemark::LoadModelAsync) { + FF::AsyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::LoadModelAsync", + info + ); +} + +NAN_METHOD(Facemark::GetData) { + FF::SyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::GetData", + info + ); +} + +NAN_METHOD(Facemark::GetDataAsync) { + FF::AsyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::GetDataAsync", + info + ); +} + +NAN_METHOD(Facemark::GetFaces) { + FF::SyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::GetFaces", + info + ); +} + +NAN_METHOD(Facemark::GetFacesAsync) { + FF::AsyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::GetFacesAsync", + info + ); +} + +NAN_METHOD(Facemark::SetFaceDetector) { + FF_METHOD_CONTEXT("SetFaceDetector"); + + if (!info[0]->IsFunction()) { + return Nan::ThrowError(Nan::New("Facemark::SetFaceDetector - Error: " + "expected argument 0 to be of type") + .ToLocalChecked()); + } + + FF_ARG_FUNC(0, v8::Local cbFunc); + Nan::Callback *callback = new Nan::Callback(cbFunc); + + bool results = FF_UNWRAP(info.This(), Facemark) + ->getFacemark() + ->setFaceDetector((cv::face::FN_FaceDetector)detector, callback); + + info.GetReturnValue().Set(Nan::New(results)); +} + +NAN_METHOD(Facemark::Training) { + FF::SyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::Train", + info + ); +} + +NAN_METHOD(Facemark::TrainingAsync) { + FF::AsyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::TrainAsync", + info + ); +} + +NAN_METHOD(Facemark::Fit) { + FF::SyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::Fit", + info + ); +} + +NAN_METHOD(Facemark::FitAsync) { + FF::AsyncBinding( + std::make_shared(FF_UNWRAP(info.This(), Facemark)->getFacemark()), + "Facemark::FitAsync", + info + ); +} + +bool Facemark::detector(cv::InputArray image, cv::OutputArray faces, + Nan::Callback *callback) { + Nan::HandleScope scope; + + cv::Mat frame = image.getMat().clone(); + v8::Local jsMat = Mat::Converter::wrap(frame); + + v8::Local argv[] = {jsMat}; + FF_OBJ jsObject = callback->Call(1, argv)->ToObject(); + + std::vector _faces; + ObjectArrayConverter::unwrap(&_faces, jsObject); + + cv::Mat(_faces).copyTo(faces); + + return true; +} + +#endif + +#endif diff --git a/cc/modules/face/Facemark.h b/cc/modules/face/Facemark.h new file mode 100644 index 000000000..43de3cd27 --- /dev/null +++ b/cc/modules/face/Facemark.h @@ -0,0 +1,44 @@ +#include "NativeNodeUtils.h" +#include "Mat.h" +#include "Point.h" +#include "Rect.h" +#include "macros.h" +#include +#include + +#if CV_VERSION_MINOR >= 4 + +#ifndef __FF_FACEMARK_H__ +#define __FF_FACEMARK_H__ + +class Facemark : public Nan::ObjectWrap { +public: + virtual cv::Ptr getFacemark() = 0; + virtual void save(std::string) = 0; + virtual void load(std::string) = 0; + + static void Init(v8::Local); + + static NAN_METHOD(AddTrainingSample); + static NAN_METHOD(AddTrainingSampleAsync); + static NAN_METHOD(LoadModel); + static NAN_METHOD(LoadModelAsync); + static NAN_METHOD(GetData); + static NAN_METHOD(GetDataAsync); + static NAN_METHOD(GetFaces); + static NAN_METHOD(GetFacesAsync); + static NAN_METHOD(SetFaceDetector); + static NAN_METHOD(Training); + static NAN_METHOD(TrainingAsync); + static NAN_METHOD(Fit); + static NAN_METHOD(FitAsync); + static NAN_METHOD(Save); + static NAN_METHOD(Load); + + static bool detector(cv::InputArray image, cv::OutputArray faces, + Nan::Callback *callback); +}; + +#endif + +#endif diff --git a/cc/modules/face/FacemarkAAM.cc b/cc/modules/face/FacemarkAAM.cc new file mode 100644 index 000000000..ef0085694 --- /dev/null +++ b/cc/modules/face/FacemarkAAM.cc @@ -0,0 +1,43 @@ +#ifdef HAVE_FACE + +#include "FacemarkAAM.h" +#include "FacemarkAAMParams.h" + +#if CV_VERSION_MINOR >= 4 + +Nan::Persistent FacemarkAAM::constructor; + +NAN_MODULE_INIT(FacemarkAAM::Init) { + v8::Local ctor = + Nan::New(FacemarkAAM::New); + v8::Local instanceTemplate = ctor->InstanceTemplate(); + + Facemark::Init(ctor); + constructor.Reset(ctor); + ctor->SetClassName(Nan::New("FacemarkAAM").ToLocalChecked()); + instanceTemplate->SetInternalFieldCount(1); + + target->Set(Nan::New("FacemarkAAM").ToLocalChecked(), ctor->GetFunction()); +}; + +NAN_METHOD(FacemarkAAM::New) { + FF_METHOD_CONTEXT("FacemarkAAM::New"); + + FF_ARG_INSTANCE_IFDEF( + 0, + cv::face::FacemarkAAM::Params params, + FacemarkAAMParams::constructor, + FF_UNWRAP_FACEMARKAAMPARAMS_AND_GET, + cv::face::FacemarkAAM::Params() + ); + + FacemarkAAM *self = new FacemarkAAM(); + self->Wrap(info.Holder()); + self->facemark = cv::face::FacemarkAAM::create(params); + + FF_RETURN(info.Holder()); +}; + +#endif + +#endif diff --git a/cc/modules/face/FacemarkAAM.h b/cc/modules/face/FacemarkAAM.h new file mode 100644 index 000000000..61351e25e --- /dev/null +++ b/cc/modules/face/FacemarkAAM.h @@ -0,0 +1,27 @@ +#include "Facemark.h" + +#if CV_VERSION_MINOR >= 4 + +#ifndef __FF_FACEMARKAAM_H__ +#define __FF_FACEMARKAAM_H__ + +class FacemarkAAM : public Facemark { +public: + cv::Ptr facemark; + + void save(std::string path) { facemark->save(path); } + + void load(std::string path) { + cv::Algorithm::load(path); + } + + static NAN_MODULE_INIT(Init); + static NAN_METHOD(New); + + static Nan::Persistent constructor; + cv::Ptr getFacemark() { return facemark; } +}; + +#endif + +#endif diff --git a/cc/modules/face/FacemarkAAMData.cc b/cc/modules/face/FacemarkAAMData.cc new file mode 100644 index 000000000..213296156 --- /dev/null +++ b/cc/modules/face/FacemarkAAMData.cc @@ -0,0 +1,33 @@ +#ifdef HAVE_FACE + +#include "FacemarkAAMData.h" + +#if CV_VERSION_MINOR >= 4 + +Nan::Persistent FacemarkAAMData::constructor; + +NAN_MODULE_INIT(FacemarkAAMData::Init) { + v8::Local ctor = + Nan::New(FacemarkAAMData::New); + v8::Local instanceTemplate = ctor->InstanceTemplate(); + + constructor.Reset(ctor); + ctor->SetClassName(FF_NEW_STRING("FacemarkAAMData")); + instanceTemplate->SetInternalFieldCount(1); + + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("s0"), s0Get, s0Set); + + target->Set(FF_NEW_STRING("FacemarkAAMData"), ctor->GetFunction()); +}; + +NAN_METHOD(FacemarkAAMData::New) { + FF_METHOD_CONTEXT("FacemarkAAMData::New"); + FacemarkAAMData *self = new FacemarkAAMData(); + self->data = cv::face::FacemarkAAM::Data(); + self->Wrap(info.Holder()); + FF_RETURN(info.Holder()); +}; + +#endif + +#endif diff --git a/cc/modules/face/FacemarkAAMData.h b/cc/modules/face/FacemarkAAMData.h new file mode 100644 index 000000000..4d8b59465 --- /dev/null +++ b/cc/modules/face/FacemarkAAMData.h @@ -0,0 +1,36 @@ +#include "macros.h" +#include "ArrayConverters.h" +#include "Point2.h" +#include +#include + +#if CV_VERSION_MINOR >= 4 + +#ifndef __FF_FACEMARKAAMDATA_H__ +#define __FF_FACEMARKAAMDATA_H__ + +typedef ObjectArrayConverter s0_converter; + +class FacemarkAAMData : public Nan::ObjectWrap { +public: + cv::face::FacemarkAAM::Data data; + + static NAN_MODULE_INIT(Init); + static NAN_METHOD(New); + + static FF_GETTER_COMPLEX(FacemarkAAMData, s0Get, data.s0, s0_converter); + static FF_SETTER_COMPLEX(FacemarkAAMData, s0, data.s0, std::vector, s0_converter); + + static Nan::Persistent constructor; + + cv::face::FacemarkAAM::Data* getNativeObjectPtr() { return &data; } + cv::face::FacemarkAAM::Data getNativeObject() { return data; } +}; + +#define FF_UNWRAP_FACEMARKAAMDATA(obj) FF_UNWRAP(obj, FacemarkAAMData) +#define FF_UNWRAP_FACEMARKAAMDATA_AND_GET(obj) \ + FF_UNWRAP_FACEMARKAAMDATA(obj)->data + +#endif + +#endif diff --git a/cc/modules/face/FacemarkAAMParams.cc b/cc/modules/face/FacemarkAAMParams.cc new file mode 100644 index 000000000..9a30c0d98 --- /dev/null +++ b/cc/modules/face/FacemarkAAMParams.cc @@ -0,0 +1,42 @@ +#ifdef HAVE_FACE + +#include "FacemarkAAMParams.h" + +#if CV_VERSION_MINOR >= 4 + +Nan::Persistent FacemarkAAMParams::constructor; + +NAN_MODULE_INIT(FacemarkAAMParams::Init) { + v8::Local ctor = + Nan::New(FacemarkAAMParams::New); + v8::Local instanceTemplate = ctor->InstanceTemplate(); + + constructor.Reset(ctor); + ctor->SetClassName(FF_NEW_STRING("FacemarkAAMParams")); + instanceTemplate->SetInternalFieldCount(1); + + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("m"), mGet, mSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("maxM"), maxMGet, maxMSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("maxN"), maxNGet, maxNSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("modelFilename"), modelFilenameGet, modelFilenameSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("n"), nGet, nSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("nIter"), nIterGet, nIterSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("saveModel"), saveModelGet, saveModelSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("scales"), scalesGet, scalesSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("textureMaxM"), textureMaxMGet, textureMaxMSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("verbose"), verboseGet, verboseSet); + + target->Set(FF_NEW_STRING("FacemarkAAMParams"), ctor->GetFunction()); +}; + +NAN_METHOD(FacemarkAAMParams::New) { + FF_METHOD_CONTEXT("FacemarkAAMParams::New"); + FacemarkAAMParams *self = new FacemarkAAMParams(); + self->params = cv::face::FacemarkAAM::Params(); + self->Wrap(info.Holder()); + FF_RETURN(info.Holder()); +}; + +#endif + +#endif diff --git a/cc/modules/face/FacemarkAAMParams.h b/cc/modules/face/FacemarkAAMParams.h new file mode 100644 index 000000000..01282c60b --- /dev/null +++ b/cc/modules/face/FacemarkAAMParams.h @@ -0,0 +1,47 @@ +#include "macros.h" +#include "TypeConverters.h" +#include + +#if CV_VERSION_MINOR >= 4 + +#ifndef __FF_FACEMARKAAMPARAMS_H__ +#define __FF_FACEMARKAAMPARAMS_H__ + +class FacemarkAAMParams : public Nan::ObjectWrap { +public: + cv::face::FacemarkAAM::Params params; + + static NAN_MODULE_INIT(Init); + static NAN_METHOD(New); + + static FF_GETTER(FacemarkAAMParams, mGet, params.m); + static FF_SETTER_INT(FacemarkAAMParams, m, params.m); + static FF_GETTER(FacemarkAAMParams, maxMGet, params.max_m); + static FF_SETTER_INT(FacemarkAAMParams, maxM, params.max_m); + static FF_GETTER(FacemarkAAMParams, maxNGet, params.max_n); + static FF_SETTER_INT(FacemarkAAMParams, maxN, params.max_n); + static FF_GETTER_SIMPLE(FacemarkAAMParams, modelFilenameGet, params.model_filename, StringConverter); + static FF_SETTER_SIMPLE(FacemarkAAMParams, modelFilename, params.model_filename, StringConverter); + static FF_GETTER(FacemarkAAMParams, nGet, params.n); + static FF_SETTER_INT(FacemarkAAMParams, n, params.n); + static FF_GETTER(FacemarkAAMParams, nIterGet, params.n_iter); + static FF_SETTER_INT(FacemarkAAMParams, nIter, params.n_iter); + static FF_GETTER(FacemarkAAMParams, saveModelGet, params.save_model); + static FF_SETTER_BOOL(FacemarkAAMParams, saveModel, params.save_model); + static FF_GETTER_COMPLEX(FacemarkAAMParams, scalesGet, params.scales, FloatArrayConverter); + static FF_SETTER_COMPLEX(FacemarkAAMParams, scales, params.scales, std::vector, FloatArrayConverter); + static FF_GETTER(FacemarkAAMParams, textureMaxMGet, params.texture_max_m); + static FF_SETTER_INT(FacemarkAAMParams, textureMaxM, params.texture_max_m); + static FF_GETTER(FacemarkAAMParams, verboseGet, params.verbose); + static FF_SETTER_BOOL(FacemarkAAMParams, verbose, params.verbose); + + static Nan::Persistent constructor; +}; + +#define FF_UNWRAP_FACEMARKAAMPARAMS(obj) FF_UNWRAP(obj, FacemarkAAMParams) +#define FF_UNWRAP_FACEMARKAAMPARAMS_AND_GET(obj) \ + FF_UNWRAP_FACEMARKAAMPARAMS(obj)->params + +#endif + +#endif diff --git a/cc/modules/face/FacemarkBindings.h b/cc/modules/face/FacemarkBindings.h new file mode 100644 index 000000000..57656b9d3 --- /dev/null +++ b/cc/modules/face/FacemarkBindings.h @@ -0,0 +1,138 @@ +#include "Facemark.h" +#include "FacemarkAAMData.h" + +#if CV_VERSION_MINOR >= 4 + +#ifndef __FF_FACEMARKBINDINGS_H_ +#define __FF_FACEMARKBINDINGS_H_ + +namespace FacemarkBindings { + struct AddTrainingSampleWorker : public CatchCvExceptionWorker { + public: + cv::Ptr self; + AddTrainingSampleWorker(cv::Ptr self) { + this->self = self; + } + + bool results; + cv::Mat image; + std::vector landmarks; + + std::string executeCatchCvExceptionWorker() { + results = self->addTrainingSample(image, landmarks); + return ""; + } + + v8::Local getReturnValue() { + v8::Local ret = Nan::New(results); + return ret; + } + + bool unwrapRequiredArgs(Nan::NAN_METHOD_ARGS_TYPE info) { + return (Mat::Converter::arg(0, &image, info) || + ObjectArrayConverter::arg( + 1, &landmarks, info)); + } + }; + + struct LoadModelWorker : public CatchCvExceptionWorker { + public: + cv::Ptr self; + LoadModelWorker(cv::Ptr self) { this->self = self; } + + std::string model; + + std::string executeCatchCvExceptionWorker() { + self->loadModel(model); + return ""; + } + + bool unwrapRequiredArgs(Nan::NAN_METHOD_ARGS_TYPE info) { + return (StringConverter::arg(0, &model, info)); + } + }; + + struct GetDataWorker : public CatchCvExceptionWorker { + public: + cv::Ptr self; + GetDataWorker(cv::Ptr self) { this->self = self; } + + cv::face::FacemarkAAM::Data data; + + std::string executeCatchCvExceptionWorker() { + self->getData(&data); + return ""; + } + + v8::Local getReturnValue() { + v8::Local ret = InstanceConverter::wrap(data); + return ret; + } + }; + + struct GetFacesWorker : public CatchCvExceptionWorker { + public: + cv::Ptr self; + GetFacesWorker(cv::Ptr self) { this->self = self; } + + cv::Mat image; + std::vector faces; + + std::string executeCatchCvExceptionWorker() { + self->getFaces(image, faces); + return ""; + } + + v8::Local getReturnValue() { + v8::Local ret = + ObjectArrayConverter::wrap(faces); + return ret; + } + + bool unwrapRequiredArgs(Nan::NAN_METHOD_ARGS_TYPE info) { + return (Mat::Converter::arg(0, &image, info)); + } + }; + + struct TrainingWorker : public CatchCvExceptionWorker { + public: + cv::Ptr self; + TrainingWorker(cv::Ptr self) { this->self = self; } + + std::string executeCatchCvExceptionWorker() { + self->training(); + return ""; + } + }; + + struct FitWorker : public CatchCvExceptionWorker { + public: + cv::Ptr self; + FitWorker(cv::Ptr self) { this->self = self; } + + cv::Mat image; + std::vector faces; + std::vector> landmarks; + + std::string executeCatchCvExceptionWorker() { + self->fit(image, faces, landmarks); + return ""; + } + + v8::Local getReturnValue() { + v8::Local ret = + ObjectArrayOfArraysConverter::wrap( + landmarks); + return ret; + } + + bool unwrapRequiredArgs(Nan::NAN_METHOD_ARGS_TYPE info) { + return (Mat::Converter::arg(0, &image, info) || + ObjectArrayConverter::arg(1, &faces, info)); + } + }; +} + +#endif + +#endif diff --git a/cc/modules/face/FacemarkLBF.cc b/cc/modules/face/FacemarkLBF.cc new file mode 100644 index 000000000..2ba5fcd3b --- /dev/null +++ b/cc/modules/face/FacemarkLBF.cc @@ -0,0 +1,43 @@ +#ifdef HAVE_FACE + +#include "FacemarkLBF.h" +#include "FacemarkLBFParams.h" + +#if CV_VERSION_MINOR >= 4 + +Nan::Persistent FacemarkLBF::constructor; + +NAN_MODULE_INIT(FacemarkLBF::Init) { + v8::Local ctor = + Nan::New(FacemarkLBF::New); + v8::Local instanceTemplate = ctor->InstanceTemplate(); + + Facemark::Init(ctor); + constructor.Reset(ctor); + ctor->SetClassName(Nan::New("FacemarkLBF").ToLocalChecked()); + instanceTemplate->SetInternalFieldCount(1); + + target->Set(Nan::New("FacemarkLBF").ToLocalChecked(), ctor->GetFunction()); +}; + +NAN_METHOD(FacemarkLBF::New) { + FF_METHOD_CONTEXT("FacemarkLBF::New"); + + FF_ARG_INSTANCE_IFDEF( + 0, + cv::face::FacemarkLBF::Params params, + FacemarkLBFParams::constructor, + FF_UNWRAP_FACEMARKLBFPARAMS_AND_GET, + cv::face::FacemarkLBF::Params() + ); + + FacemarkLBF *self = new FacemarkLBF(); + self->Wrap(info.Holder()); + self->facemark = cv::face::FacemarkLBF::create(params); + + FF_RETURN(info.Holder()); +}; + +#endif + +#endif diff --git a/cc/modules/face/FacemarkLBF.h b/cc/modules/face/FacemarkLBF.h new file mode 100644 index 000000000..ab621ecd9 --- /dev/null +++ b/cc/modules/face/FacemarkLBF.h @@ -0,0 +1,27 @@ +#include "Facemark.h" + +#if CV_VERSION_MINOR >= 4 + +#ifndef __FF_FACEMARKLBF_H__ +#define __FF_FACEMARKLBF_H__ + +class FacemarkLBF : public Facemark { +public: + cv::Ptr facemark; + + void save(std::string path) { facemark->save(path); } + + void load(std::string path) { + cv::Algorithm::load(path); + } + + static NAN_MODULE_INIT(Init); + static NAN_METHOD(New); + + static Nan::Persistent constructor; + cv::Ptr getFacemark() { return facemark; } +}; + +#endif + +#endif diff --git a/cc/modules/face/FacemarkLBFParams.cc b/cc/modules/face/FacemarkLBFParams.cc new file mode 100644 index 000000000..1dc97ada5 --- /dev/null +++ b/cc/modules/face/FacemarkLBFParams.cc @@ -0,0 +1,48 @@ +#ifdef HAVE_FACE + +#include "FacemarkLBFParams.h" + +#if CV_VERSION_MINOR >= 4 + +Nan::Persistent FacemarkLBFParams::constructor; + +NAN_MODULE_INIT(FacemarkLBFParams::Init) { + v8::Local ctor = + Nan::New(FacemarkLBFParams::New); + v8::Local instanceTemplate = ctor->InstanceTemplate(); + + constructor.Reset(ctor); + ctor->SetClassName(FF_NEW_STRING("FacemarkLBFParams")); + instanceTemplate->SetInternalFieldCount(1); + + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("baggingOverlap"), baggingOverlapGet, baggingOverlapSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("cascadeFace"), cascadeFaceGet, cascadeFaceSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("detectROI"), detectROIGet, detectROISet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("featsM"), featsMGet, featsMSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("initShapeN"), initShapeNGet, initShapeNSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("modelFilename"), modelFilenameGet, modelFilenameSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("nLandmarks"), nLandmarksGet, nLandmarksSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("pupils"), pupilsGet, pupilsSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("radiusM"), radiusMGet, radiusMSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("saveModel"), saveModelGet, saveModelSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("seed"), seedGet, seedSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("shapeOffset"), shapeOffsetGet, shapeOffsetSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("stagesN"), stagesNGet, stagesNSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("treeDepth"), treeDepthGet, treeDepthSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("treeN"), treeNGet, treeNSet); + Nan::SetAccessor(instanceTemplate, FF_NEW_STRING("verbose"), verboseGet, verboseSet); + + target->Set(FF_NEW_STRING("FacemarkLBFParams"), ctor->GetFunction()); +}; + +NAN_METHOD(FacemarkLBFParams::New) { + FF_METHOD_CONTEXT("FacemarkLBFParams::New"); + FacemarkLBFParams *self = new FacemarkLBFParams(); + self->params = cv::face::FacemarkLBF::Params(); + self->Wrap(info.Holder()); + FF_RETURN(info.Holder()); +}; + +#endif + +#endif diff --git a/cc/modules/face/FacemarkLBFParams.h b/cc/modules/face/FacemarkLBFParams.h new file mode 100644 index 000000000..bb3cd86f1 --- /dev/null +++ b/cc/modules/face/FacemarkLBFParams.h @@ -0,0 +1,83 @@ +#include "macros.h" +#include "TypeConverters.h" +#include "ArrayConverters.h" +#include "Rect.h" +#include + +#if CV_VERSION_MINOR >= 4 + +#ifndef __FF_FACEMARKLBFPARAMS_H__ +#define __FF_FACEMARKLBFPARAMS_H__ + +class FacemarkLBFParams : public Nan::ObjectWrap { +public: + cv::face::FacemarkLBF::Params params; + + static NAN_MODULE_INIT(Init); + static NAN_METHOD(New); + + static FF_GETTER(FacemarkLBFParams, baggingOverlapGet, params.bagging_overlap); + static FF_SETTER_NUMBER(FacemarkLBFParams, baggingOverlap, params.bagging_overlap); + static FF_GETTER_SIMPLE(FacemarkLBFParams, cascadeFaceGet, params.cascade_face, StringConverter); + static FF_SETTER_SIMPLE(FacemarkLBFParams, cascadeFace, params.cascade_face, StringConverter); + static FF_GETTER_SIMPLE(FacemarkLBFParams, detectROIGet, params.detectROI, Rect::Converter); + static FF_SETTER_SIMPLE(FacemarkLBFParams, detectROI, params.detectROI, Rect::Converter); + static FF_GETTER_COMPLEX(FacemarkLBFParams, featsMGet, params.feats_m, IntArrayConverter); + static FF_SETTER_COMPLEX(FacemarkLBFParams, featsM, params.feats_m, std::vector, IntArrayConverter); + static FF_GETTER(FacemarkLBFParams, initShapeNGet, params.initShape_n); + static FF_SETTER_INT(FacemarkLBFParams, initShapeN, params.initShape_n); + static FF_GETTER_SIMPLE(FacemarkLBFParams, modelFilenameGet, params.model_filename, StringConverter); + static FF_SETTER_SIMPLE(FacemarkLBFParams, modelFilename, params.model_filename, StringConverter); + static FF_GETTER(FacemarkLBFParams, nLandmarksGet, params.n_landmarks); + static FF_SETTER_INT(FacemarkLBFParams, nLandmarks, params.n_landmarks); + + static NAN_GETTER(pupilsGet) { + v8::Local jsArr = Nan::New(2); + for (int i = 0; i < jsArr->Length(); i++) { + jsArr->Set(i, ArrayConverterType::wrap( + Nan::ObjectWrap::Unwrap(info.This())->params.pupils[i]) + ); + } + info.GetReturnValue().Set(jsArr); + } + + static NAN_SETTER(pupilsSet) { + FF_METHOD_CONTEXT("pupils"); + v8::Local jsArr = v8::Local::Cast(value); + for (int i = 0; i < jsArr->Length(); i++) { + std::vector vec; + Nan::TryCatch tryCatch; + if (ArrayConverterType::unwrap(&vec, jsArr->Get(i))) { + tryCatch.ReThrow(); + } + Nan::ObjectWrap::Unwrap(info.This())->params.pupils[i] = vec; + } + } + + static FF_GETTER_COMPLEX(FacemarkLBFParams, radiusMGet, params.radius_m, DoubleArrayConverter); + static FF_SETTER_COMPLEX(FacemarkLBFParams, radiusM, params.radius_m, std::vector, DoubleArrayConverter); + static FF_GETTER(FacemarkLBFParams, saveModelGet, params.save_model); + static FF_SETTER_BOOL(FacemarkLBFParams, saveModel, params.save_model); + static FF_GETTER(FacemarkLBFParams, seedGet, params.seed); + static FF_SETTER_UINT(FacemarkLBFParams, seed, params.seed); + static FF_GETTER(FacemarkLBFParams, shapeOffsetGet, params.shape_offset); + static FF_SETTER_NUMBER(FacemarkLBFParams, shapeOffset, params.shape_offset); + static FF_GETTER(FacemarkLBFParams, stagesNGet, params.stages_n); + static FF_SETTER_INT(FacemarkLBFParams, stagesN, params.stages_n); + static FF_GETTER(FacemarkLBFParams, treeDepthGet, params.tree_depth); + static FF_SETTER_INT(FacemarkLBFParams, treeDepth, params.tree_depth); + static FF_GETTER(FacemarkLBFParams, treeNGet, params.tree_n); + static FF_SETTER_INT(FacemarkLBFParams, treeN, params.tree_n); + static FF_GETTER(FacemarkLBFParams, verboseGet, params.verbose); + static FF_SETTER_BOOL(FacemarkLBFParams, verbose, params.verbose); + + static Nan::Persistent constructor; +}; + +#define FF_UNWRAP_FACEMARKLBFPARAMS(obj) FF_UNWRAP(obj, FacemarkLBFParams) +#define FF_UNWRAP_FACEMARKLBFPARAMS_AND_GET(obj) \ + FF_UNWRAP_FACEMARKLBFPARAMS(obj)->params + +#endif + +#endif diff --git a/cc/modules/face/face.cc b/cc/modules/face/face.cc index 658f4cfc1..a388cfd79 100644 --- a/cc/modules/face/face.cc +++ b/cc/modules/face/face.cc @@ -5,10 +5,25 @@ #include "FisherFaceRecognizer.h" #include "LBPHFaceRecognizer.h" +#if CV_VERSION_MINOR >= 4 +#include "FacemarkAAM.h" +#include "FacemarkAAMData.h" +#include "FacemarkAAMParams.h" +#include "FacemarkLBF.h" +#include "FacemarkLBFParams.h" +#endif + NAN_MODULE_INIT(Face::Init) { - EigenFaceRecognizer::Init(target); - FisherFaceRecognizer::Init(target); - LBPHFaceRecognizer::Init(target); + EigenFaceRecognizer::Init(target); + FisherFaceRecognizer::Init(target); + LBPHFaceRecognizer::Init(target); +#if CV_VERSION_MINOR >= 4 + FacemarkAAM::Init(target); + FacemarkAAMData::Init(target); + FacemarkAAMParams::Init(target); + FacemarkLBF::Init(target); + FacemarkLBFParams::Init(target); +#endif }; -#endif \ No newline at end of file +#endif diff --git a/examples/facemark.js b/examples/facemark.js new file mode 100644 index 000000000..db4d36c51 --- /dev/null +++ b/examples/facemark.js @@ -0,0 +1,47 @@ +const cv = require("../"); +const fs = require("fs"); +const path = require("path"); + +if (!cv.xmodules.face) { + throw new Error("exiting: opencv4nodejs compiled without face module"); +} + +const facemarkModelPath = "../data/face/"; +const modelFile = path.resolve(facemarkModelPath, "lbfmodel.yaml"); + +if (!fs.existsSync(modelFile)) { + console.log("could not find landmarks model"); + console.log( + "download the model from: https://raw.githubusercontent.com/kurnianggoro/GSOC2017/master/data/lbfmodel.yaml" + ); + throw new Error("exiting: could not find landmarks model"); +} + +const classifier = new cv.CascadeClassifier(cv.HAAR_FRONTALFACE_ALT2); + +// create the facemark object with the landmarks model +const facemark = new cv.FacemarkLBF(); +facemark.loadModel(modelFile); + +// give the facemark object it's face detection callback +facemark.setFaceDetector(frame => { + const { objects } = classifier.detectMultiScale(frame, 1.12); + return objects; +}); + +// retrieve faces using the facemark face detector callback +const image = cv.imread("../data/got.jpg"); +const gray = image.bgrToGray(); +const faces = facemark.getFaces(gray); + +// use the detected faces to detect the landmarks +const faceLandmarks = facemark.fit(gray, faces); + +for (let i = 0; i < faceLandmarks.length; i++) { + const landmarks = faceLandmarks[i]; + for (let x = 0; x < landmarks.length; x++) { + image.drawCircle(landmarks[x], 1, new cv.Vec(0, 255, 0), 1, cv.LINE_8); + } +} + +cv.imshowWait("VideoCapture", image); diff --git a/lib/typings/Facemark.d.ts b/lib/typings/Facemark.d.ts new file mode 100644 index 000000000..fce3c7930 --- /dev/null +++ b/lib/typings/Facemark.d.ts @@ -0,0 +1,19 @@ +import { Mat } from "./Mat.d"; +import { Rect } from "./Rect.d"; +import { Point2 } from "./Point2.d"; + +export class Facemark { + addTrainingSample(image: Mat, landmarks: number[][]): boolean; + addTrainingSampleAsync(image: Mat, landmarks: number[][]): Promise; + loadModel(model: string): void; + loadModelAsync(model: string): Promise; + getFaces(image: Mat): Rect[]; + getFacesAsync(image: Mat): Promise; + setFaceDetector(callback: Function): boolean; + training(): void; + trainingAsync(): Promise; + fit(image: Mat, faces: Rect[]): Point2[][]; + fitAsync(image: Mat, faces: Rect[]): Promise; + save(file: string): void; + load(file: string): void; +} diff --git a/lib/typings/FacemarkAAMParams.d.ts b/lib/typings/FacemarkAAMParams.d.ts new file mode 100644 index 000000000..af99fd30e --- /dev/null +++ b/lib/typings/FacemarkAAMParams.d.ts @@ -0,0 +1,13 @@ +export class FacemarkAAMParams { + readonly m: number; + readonly maxM: number; + readonly maxN: number; + readonly modelFilename: string; + readonly n: number; + readonly nIter: number; + readonly saveModel: boolean; + readonly scales: number[]; + readonly textureMaxM: number; + readonly verbose: boolean; + constructor(); +} diff --git a/lib/typings/FacemarkLBF.d.ts b/lib/typings/FacemarkLBF.d.ts new file mode 100644 index 000000000..adf6485ab --- /dev/null +++ b/lib/typings/FacemarkLBF.d.ts @@ -0,0 +1,3 @@ +import { Facemark } from "./Facemark"; + +export class FacemarkLBF extends Facemark {} diff --git a/lib/typings/FacemarkLBFParams.d.ts b/lib/typings/FacemarkLBFParams.d.ts new file mode 100644 index 000000000..95aef54bd --- /dev/null +++ b/lib/typings/FacemarkLBFParams.d.ts @@ -0,0 +1,21 @@ +import { Rect } from "./Rect.d"; + +export class FacemarkLBFParams { + readonly baggingOverlap: number; + readonly cascadeFace: string; + readonly detectROI: Rect; + readonly featsM: number[]; + readonly initShapeN: number; + readonly modelFilename: string; + readonly nLandmarks: number; + readonly pupils: number[]; + readonly radiusM: number[]; + readonly saveModel: boolean; + readonly seed: number; + readonly shapeOffset: number; + readonly stagesN: number; + readonly treeDepth: number; + readonly treeN: number; + readonly verbose: boolean; + constructor(); +} diff --git a/lib/typings/FacemarkrAAM.d.ts b/lib/typings/FacemarkrAAM.d.ts new file mode 100644 index 000000000..31b50b4ae --- /dev/null +++ b/lib/typings/FacemarkrAAM.d.ts @@ -0,0 +1,3 @@ +import { Facemark } from "./Facemark"; + +export class FacemarkAAM extends Facemark {} diff --git a/test/tests/modules/face/face.test.js b/test/tests/modules/face/face.test.js index 5c156568e..4cdcb5b0b 100644 --- a/test/tests/modules/face/face.test.js +++ b/test/tests/modules/face/face.test.js @@ -1,34 +1,46 @@ -const cv = global.dut; -const { readTestImage } = global.utils; -const recognizerTests = require('./recognizerTests'); - -describe('face', () => { - if (!cv.xmodules.face) { - it('compiled without face'); - return; - } - - let testImg; - - before(() => { - testImg = readTestImage().resizeToMax(250); - }); - - describe('EigenFaceRecognizer', () => { - const args = ['num_components', 'threshold']; - const values = [10, 0.8]; - recognizerTests(() => testImg, args, values, cv.EigenFaceRecognizer); - }); - - describe('FisherFaceRecognizer', () => { - const args = ['num_components', 'threshold']; - const values = [10, 0.8]; - recognizerTests(() => testImg, args, values, cv.FisherFaceRecognizer); - }); - - describe('LBPHFaceRecognizer', () => { - const args = ['radius', 'neighbors', 'grid_x', 'grid_y']; - const values = [2, 16, 16, 16]; - recognizerTests(() => testImg, args, values, cv.LBPHFaceRecognizer); - }); -}); +const cv = global.dut; +const { readTestImage } = global.utils; +const recognizerTests = require('./recognizerTests'); +const facemarkTests = require('./facemarkTests'); +const facemarkStructsTests = require('./facemarkStructsTests'); + +describe('face', () => { + if (!cv.xmodules.face) { + it('compiled without face'); + return; + } + + let testImg; + + before(() => { + testImg = readTestImage().resizeToMax(250); + }); + + describe('EigenFaceRecognizer', () => { + const args = ['num_components', 'threshold']; + const values = [10, 0.8]; + recognizerTests(() => testImg, args, values, cv.EigenFaceRecognizer); + }); + + describe('FisherFaceRecognizer', () => { + const args = ['num_components', 'threshold']; + const values = [10, 0.8]; + recognizerTests(() => testImg, args, values, cv.FisherFaceRecognizer); + }); + + describe('LBPHFaceRecognizer', () => { + const args = ['radius', 'neighbors', 'grid_x', 'grid_y']; + const values = [2, 16, 16, 16]; + recognizerTests(() => testImg, args, values, cv.LBPHFaceRecognizer); + }); + + facemarkStructsTests(); + + describe('FacemarkLBF', () => { + facemarkTests(() => testImg, cv.FacemarkLBF, cv.FacemarkLBFParams); + }); + + describe('FacemarkAAM', () => { + facemarkTests(() => testImg, cv.FacemarkAAM, cv.FacemarkAAMParams); + }); +}); diff --git a/test/tests/modules/face/facemarkStructsTests.js b/test/tests/modules/face/facemarkStructsTests.js new file mode 100644 index 000000000..3a7be48cb --- /dev/null +++ b/test/tests/modules/face/facemarkStructsTests.js @@ -0,0 +1,69 @@ +const cv = global.dut; +const { assertPropsWithValue } = global.utils; + +module.exports = () => { + describe('Facemark structures', () => { + it('FacemarkAAMData', () => { + const data = { + s0: [new cv.Point2(0, 0), new cv.Point2(0, 0)] + }; + + const facemarkData = new cv.FacemarkAAMData(); + Object.keys(data).forEach((item) => { + facemarkData[item] = data[item]; + }); + + assertPropsWithValue(facemarkData)(data); + }); + + it('FacemarkAAMParams', () => { + const params = { + m: 10, + n: 10, + maxM: 30, + maxN: 30, + modelFilename: 'filename.xml', + nIter: 4, + saveModel: true, + scales: [3.0, 2.0], + textureMaxM: 12, + verbose: true + }; + + const facemarkParams = new cv.FacemarkAAMParams(); + Object.keys(params).forEach((param) => { + facemarkParams[param] = params[param]; + }); + + assertPropsWithValue(facemarkParams)(params); + }); + + it('FacemarkLBFParams', () => { + const params = { + baggingOverlap: 2.5, + cascadeFile: 'cascadeFile.xml', + detectROI: new cv.Rect(0, 0, 10, 10), + featsM: [5, 4, 3, 2, 1], + initShapeN: 32, + modelFilename: 'modelFilename.xml', + nLandmarks: 68, + pupils: [[1, 2, 3, 4], [4, 3, 2, 1]], + radiusM: [2.5, 5.5], + saveModel: true, + seed: 1000, + shapeOffset: 5.4, + stagesN: 4, + treeDepth: 3, + treeN: 2, + verbose: true + }; + + const facemarkParams = new cv.FacemarkLBFParams(); + Object.keys(params).forEach((param) => { + facemarkParams[param] = params[param]; + }); + + assertPropsWithValue(facemarkParams)(params); + }); + }); +}; diff --git a/test/tests/modules/face/facemarkTests.js b/test/tests/modules/face/facemarkTests.js new file mode 100644 index 000000000..1df5e054a --- /dev/null +++ b/test/tests/modules/face/facemarkTests.js @@ -0,0 +1,144 @@ +const cv = global.dut; +const { generateAPITests, clearTmpData, getTmpDataFilePath } = global.utils; +const { expect } = require('chai'); + +module.exports = (getTestImg, Facemark, FacemarkParams) => { + let testImg; + + before(() => { + testImg = getTestImg().bgrToGray(); + }); + + describe('constructor', () => { + it('is constructable without args', () => { + expect(() => new Facemark()).to.not.throw(); + }); + + it('is constructable from args', () => { + const params = new FacemarkParams(); + expect(() => new Facemark(params).to.not.throw()); + }); + }); + + describe('face detection tests', () => { + let facemark; + before(() => { + facemark = new Facemark(); + }); + + describe('setFaceDetector', () => { + const expectOutput = () => {}; + const callback = () => {}; + + generateAPITests({ + getDut: () => facemark, + methodName: 'setFaceDetector', + methodNameSpace: 'Facemark', + getRequiredArgs: () => [callback], + hasAsync: false, + expectOutput + }); + }); + + describe('getData', () => { + const expectOutput = () => {}; + + generateAPITests({ + getDut: () => facemark, + methodName: 'getData', + methodNameSpace: 'Facemark', + hasAsync: true, + expectOutput + }); + }); + + describe('getFaces', () => { + const expectOutput = () => {}; + + it('setFaceDetector', () => { + facemark.setFaceDetector(() => []); + }); + + generateAPITests({ + getDut: () => facemark, + methodName: 'getFaces', + methodNameSpace: 'Facemark', + getRequiredArgs: () => [testImg], + hasAsync: false, + expectOutput + }); + }); + }); + + describe('train', () => { + let facemark; + + const landmarks = []; + for (let i = 0; i < 68; i++) { + landmarks[i] = new cv.Point2(Math.random() * 250, Math.random() * 250); + } + + before(() => { + const params = new FacemarkParams(); + params.cascadeFace = '../lib/haarcascades/haarcascade_frontalcatface.xml'; + params.modelFilename = 'modelFilename.model'; + params.nLandmarks = 68; + params.initShapeN = 10; + params.stagesN = 5; + params.treeN = 6; + params.treeDepth = 5; + + facemark = new Facemark(params); + }); + + describe('addTrainingSample', () => { + generateAPITests({ + getDut: () => facemark, + methodName: 'addTrainingSample', + methodNameSpace: 'Facemark', + getRequiredArgs: () => [testImg, landmarks], + expectOutput: () => {} + }); + }); + }); + + describe('trained model tests', () => { + let facemark; + + before(() => { + facemark = new Facemark(); + }); + + describe('fit', () => { + const expectOutput = (res) => { + expect(res).to.be.an('array'); + }; + + const faces = []; + + generateAPITests({ + getDut: () => facemark, + methodName: 'fit', + methodNameSpace: 'Facemark', + getRequiredArgs: () => [testImg, faces], + expectOutput + }); + }); + + describe('save and load', () => { + beforeEach(() => { + clearTmpData(); + }); + afterEach(() => { + clearTmpData(); + }); + + it('should save and load from xml', () => { + const file = getTmpDataFilePath('testFacemark.xml'); + facemark.save(file); + const facemarkNew = new Facemark(); + facemarkNew.load(file); + }); + }); + }); +}; diff --git a/test/utils/testUtils.js b/test/utils/testUtils.js index c068047fe..b38d06385 100644 --- a/test/utils/testUtils.js +++ b/test/utils/testUtils.js @@ -34,7 +34,10 @@ exports.assertError = assertError; const makeCompareValues = floatSafe => (val1, val2) => { if (floatSafe && typeof val1 === 'number' && typeof val2 === 'number') { return Math.abs(val1 - val2) < 0.001; - } + } else if (typeof val1 === 'object' && typeof val2 === 'object') { + return JSON.stringify(val1) === JSON.stringify(val2); + } + return val1 === val2; };