/** * Created by G-Canvas Open Source Team. * Copyright (c) 2017, Alibaba, Inc. All rights reserved. * * This source code is licensed under the Apache Licence 2.0. * For the full copyright and license information, please view * the LICENSE file in the root directory of this source tree. */ #include "Canvas.h" #include "TextMetrics.h" namespace NodeBinding { Napi::FunctionReference Canvas::constructor; Canvas::Canvas(const Napi::CallbackInfo &info) : Napi::ObjectWrap(info), mDataRaw(nullptr) { Napi::Env env = info.Env(); Napi::HandleScope scope(env); checkArgs(info, 2); mWidth = info[0].As().Int32Value(); mHeight = info[1].As().Int32Value(); mRenderContext = std::make_shared(mWidth, mHeight,2.0); mRenderContext->initRenderEnviroment(); } int Canvas::getWidth() { return mWidth; } int Canvas::getHeight() { return mHeight; } Napi::Value Canvas::getWidth(const Napi::CallbackInfo &info) { return Napi::Number::New(info.Env(), mWidth); } Napi::Value Canvas::getHeight(const Napi::CallbackInfo &info) { return Napi::Number::New(info.Env(), mHeight); } void Canvas::Init(Napi::Env env, Napi::Object exports) { Napi::HandleScope scope(env); Napi::Function func = DefineClass(env, "Canvas", { InstanceAccessor("width", &Canvas::getWidth, nullptr), InstanceAccessor("height", &Canvas::getHeight, nullptr), InstanceMethod("getContext", &Canvas::getContext), InstanceMethod("createPNG", &Canvas::createPNG), InstanceMethod("createJPEG", &Canvas::createJPEG), InstanceMethod("createPNGStreamSync", &Canvas::createPNGStreamSync), InstanceMethod("createJPGStreamSync", &Canvas::createJPGStreamSync), InstanceMethod("toBuffer", &Canvas::ToBuffer), }); constructor = Napi::Persistent(func); constructor.SuppressDestruct(); return; } Napi::Object Canvas::NewInstance(Napi::Env env, Napi::Value arg, Napi::Value arg2) { Napi::Object obj = constructor.New({arg, arg2}); obj.Set("name", Napi::String::New(env, "canvas")); Canvas *canvas = Napi::ObjectWrap::Unwrap(obj); canvas->mRef = Napi::ObjectReference::New(obj); return obj; } Napi::Value Canvas::getContext(const Napi::CallbackInfo &info) { Napi::Env env = info.Env(); checkArgs(info, 1); std::string type = info[0].As().Utf8Value(); if (type == "2d") { if (this->context2dRef.IsEmpty()) { Napi::Object obj = Context2D::NewInstance(env); this->context2dRef = Napi::ObjectReference::New(obj); Context2D *ctx = Napi::ObjectWrap::Unwrap(obj); ctx->setRenderContext(this->mRenderContext); ctx->setCanvasRef(this); return obj; } else { return this->context2dRef.Value(); } } else { throwError(info, "only support 2d now"); return Napi::Object::New(env); } } void Canvas::createPNG(const Napi::CallbackInfo &info) { NodeBinding::checkArgs(info, 1); std::string arg = info[0].As().Utf8Value(); if (this->mRenderContext) { this->mRenderContext->makeCurrent(); this->mRenderContext->drawFrame(); this->mRenderContext->render2file(arg.c_str(), PNG_FORAMT); } return; } void Canvas::createJPEG(const Napi::CallbackInfo &info) { NodeBinding::checkArgs(info, 1); std::string arg = info[0].As().Utf8Value(); if (this->mRenderContext) { this->mRenderContext->makeCurrent(); this->mRenderContext->drawFrame(); this->mRenderContext->render2file(arg.c_str(), JPEG_FORMAT); } return; } Napi::Value Canvas::createJPGStreamSync(const Napi::CallbackInfo &info) { NodeBinding::checkArgs(info, 2); unsigned long size = 0; Napi::Buffer buffer = this->getJPGBuffer(info, size); if (size >= 0) { Napi::Function callback = info[0].As(); //handlescope 表示作用域,一般调用callback函数时使用 Napi::HandleScope scope(info.Env()); callback.Call({info.Env().Null(), buffer, Napi::Number::New(info.Env(), size)}); } else { Napi::Function callback = info[0].As(); Napi::HandleScope scope(info.Env()); callback.Call({Napi::String::New(Env(), "createJPGStreamFail"), info.Env().Null(), info.Env().Null()}); } return info.Env().Undefined(); } Napi::Value Canvas::createPNGStreamSync(const Napi::CallbackInfo &info) { NodeBinding::checkArgs(info, 2); unsigned long size = 0; Napi::Buffer buffer = this->getPNGBuffer(info, size); if (size >= 0) { Napi::Function callback = info[0].As(); //handlescope 表示作用域,一般调用callback函数时使用 Napi::HandleScope scope(info.Env()); callback.Call({info.Env().Null(), buffer, Napi::Number::New(info.Env(), size)}); } else { Napi::Function callback = info[0].As(); Napi::HandleScope scope(info.Env()); callback.Call({Napi::String::New(Env(), "createPNGStreamFail"), info.Env().Null(), info.Env().Null()}); } return info.Env().Undefined(); } Napi::Buffer Canvas::getPNGBuffer(const Napi::CallbackInfo &info, unsigned long &size) { if (this->mRenderContext) { this->mRenderContext->makeCurrent(); this->mRenderContext->drawFrame(); } std::vector dataPNGFormat; int ret = this->mRenderContext->getImagePixelPNG(dataPNGFormat); if (ret == 0) { size = dataPNGFormat.size(); return Napi::Buffer::Copy(info.Env(), &dataPNGFormat[0], dataPNGFormat.size()); } else { return Napi::Buffer::New(info.Env(), nullptr, 0); } } Napi::Buffer Canvas::getJPGBuffer(const Napi::CallbackInfo &info, unsigned long &size) { if (this->mRenderContext) { this->mRenderContext->makeCurrent(); this->mRenderContext->drawFrame(); } unsigned char *dataJPGFormat = nullptr; int ret = this->mRenderContext->getImagePixelJPG(&dataJPGFormat, size); if (ret == 0) { return Napi::Buffer::Copy(info.Env(), dataJPGFormat, size); } else { size = -1; return Napi::Buffer::New(info.Env(), nullptr, 0); } } Napi::Buffer Canvas::getRawDataBuffer(const Napi::CallbackInfo &info, unsigned long &size) { if (this->mDataRaw == nullptr) { this->mDataRaw = new unsigned char[4 * mWidth * mHeight]; } int ret = this->mRenderContext->readPixelAndSampleFromCurrentCtx(mDataRaw); if (ret == 0) { return Napi::Buffer::Copy(info.Env(), this->mDataRaw, 4 * mWidth * mHeight); } else { size = -1; return Napi::Buffer::Copy(info.Env(), nullptr, 0); } } Napi::Value Canvas::ToBuffer(const Napi::CallbackInfo &info) { unsigned long size = 0; //默认输出png 编码 if (info.Length() == 0) { return this->getPNGBuffer(info, size); } else { Napi::Buffer ret; if (info.Length() == 1) { std::string mimeType = info[0].As().Utf8Value(); if (mimeType == "image/png") { ret = this->getPNGBuffer(info, size); } else if (mimeType == "image/jpeg") { ret = this->getJPGBuffer(info, size); } else if (mimeType == "raw") { ret = this->getRawDataBuffer(info, size); } } if (size < 0) { return info.Env().Null(); } else { return ret; } } } Canvas::~Canvas() { this->mRenderContext = nullptr; if (this->mDataRaw != nullptr) { free(this->mDataRaw); this->mDataRaw = nullptr; } printf("canvas destroy called \n"); } } // namespace NodeBinding