2015-02-11 08:03:19 +00:00
|
|
|
|
// Copyright (c) 2015 GitHub, Inc.
|
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
2020-02-04 20:19:40 +00:00
|
|
|
|
#include "shell/common/api/electron_api_native_image.h"
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
2018-09-13 00:25:56 +00:00
|
|
|
|
#include <memory>
|
2015-02-11 13:59:08 +00:00
|
|
|
|
#include <string>
|
2018-09-13 00:25:56 +00:00
|
|
|
|
#include <utility>
|
2015-02-11 13:59:08 +00:00
|
|
|
|
#include <vector>
|
|
|
|
|
|
2016-03-04 20:05:36 +00:00
|
|
|
|
#include "base/files/file_util.h"
|
2015-09-02 07:16:49 +00:00
|
|
|
|
#include "base/strings/pattern.h"
|
2016-08-26 22:30:02 +00:00
|
|
|
|
#include "base/strings/string_util.h"
|
2019-11-23 03:16:43 +00:00
|
|
|
|
#include "base/strings/utf_string_conversions.h"
|
2017-12-10 17:28:06 +00:00
|
|
|
|
#include "base/threading/thread_restrictions.h"
|
2015-02-12 06:27:53 +00:00
|
|
|
|
#include "net/base/data_url.h"
|
2019-06-19 20:46:59 +00:00
|
|
|
|
#include "shell/common/asar/asar_util.h"
|
2019-10-18 00:31:29 +00:00
|
|
|
|
#include "shell/common/gin_converters/file_path_converter.h"
|
|
|
|
|
#include "shell/common/gin_converters/gfx_converter.h"
|
|
|
|
|
#include "shell/common/gin_converters/gurl_converter.h"
|
2019-10-31 07:56:00 +00:00
|
|
|
|
#include "shell/common/gin_converters/value_converter.h"
|
2019-10-18 00:31:29 +00:00
|
|
|
|
#include "shell/common/gin_helper/dictionary.h"
|
|
|
|
|
#include "shell/common/gin_helper/object_template_builder.h"
|
2019-06-19 20:46:59 +00:00
|
|
|
|
#include "shell/common/node_includes.h"
|
2019-07-30 20:49:25 +00:00
|
|
|
|
#include "shell/common/skia_util.h"
|
2017-11-30 16:56:32 +00:00
|
|
|
|
#include "third_party/skia/include/core/SkBitmap.h"
|
|
|
|
|
#include "third_party/skia/include/core/SkImageInfo.h"
|
2016-08-26 22:30:02 +00:00
|
|
|
|
#include "third_party/skia/include/core/SkPixelRef.h"
|
2015-02-11 10:41:06 +00:00
|
|
|
|
#include "ui/base/layout.h"
|
2017-03-06 21:24:50 +00:00
|
|
|
|
#include "ui/base/webui/web_ui_util.h"
|
2015-02-11 10:41:06 +00:00
|
|
|
|
#include "ui/gfx/codec/jpeg_codec.h"
|
|
|
|
|
#include "ui/gfx/codec/png_codec.h"
|
2015-02-11 08:49:43 +00:00
|
|
|
|
#include "ui/gfx/geometry/size.h"
|
2015-02-11 10:41:06 +00:00
|
|
|
|
#include "ui/gfx/image/image_skia.h"
|
2016-10-04 17:26:01 +00:00
|
|
|
|
#include "ui/gfx/image/image_skia_operations.h"
|
2015-02-11 09:47:54 +00:00
|
|
|
|
#include "ui/gfx/image/image_util.h"
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
2015-08-08 14:58:05 +00:00
|
|
|
|
#if defined(OS_WIN)
|
2015-08-20 16:26:20 +00:00
|
|
|
|
#include "base/win/scoped_gdi_object.h"
|
2019-06-19 20:46:59 +00:00
|
|
|
|
#include "shell/common/asar/archive.h"
|
2015-08-08 14:58:05 +00:00
|
|
|
|
#include "ui/gfx/icon_util.h"
|
|
|
|
|
#endif
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
2019-06-19 21:23:04 +00:00
|
|
|
|
namespace electron {
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
|
|
|
|
namespace api {
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
2017-03-07 00:29:39 +00:00
|
|
|
|
// Get the scale factor from options object at the first argument
|
2019-10-18 00:31:29 +00:00
|
|
|
|
float GetScaleFactorFromOptions(gin::Arguments* args) {
|
2017-03-07 00:29:39 +00:00
|
|
|
|
float scale_factor = 1.0f;
|
2019-10-18 00:31:29 +00:00
|
|
|
|
gin_helper::Dictionary options;
|
2017-03-07 00:29:39 +00:00
|
|
|
|
if (args->GetNext(&options))
|
|
|
|
|
options.Get("scaleFactor", &scale_factor);
|
|
|
|
|
return scale_factor;
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-07 22:21:24 +00:00
|
|
|
|
base::FilePath NormalizePath(const base::FilePath& path) {
|
2016-03-07 17:35:35 +00:00
|
|
|
|
if (!path.ReferencesParent()) {
|
|
|
|
|
return path;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-30 17:01:55 +00:00
|
|
|
|
base::ThreadRestrictions::ScopedAllowIO allow_blocking;
|
2016-03-07 17:35:35 +00:00
|
|
|
|
base::FilePath absolute_path = MakeAbsoluteFilePath(path);
|
|
|
|
|
// MakeAbsoluteFilePath returns an empty path on failures so use original path
|
|
|
|
|
if (absolute_path.empty()) {
|
|
|
|
|
return path;
|
|
|
|
|
} else {
|
|
|
|
|
return absolute_path;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 04:24:18 +00:00
|
|
|
|
#if defined(OS_MACOSX)
|
2015-07-27 04:58:48 +00:00
|
|
|
|
bool IsTemplateFilename(const base::FilePath& path) {
|
2015-09-02 07:16:49 +00:00
|
|
|
|
return (base::MatchPattern(path.value(), "*Template.*") ||
|
|
|
|
|
base::MatchPattern(path.value(), "*Template@*x.*"));
|
2015-02-12 04:24:18 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
2015-02-11 10:41:06 +00:00
|
|
|
|
|
2015-08-21 04:06:38 +00:00
|
|
|
|
#if defined(OS_WIN)
|
2016-05-20 08:51:05 +00:00
|
|
|
|
base::win::ScopedHICON ReadICOFromPath(int size, const base::FilePath& path) {
|
2015-08-21 04:06:38 +00:00
|
|
|
|
// If file is in asar archive, we extract it to a temp file so LoadImage can
|
|
|
|
|
// load it.
|
|
|
|
|
base::FilePath asar_path, relative_path;
|
|
|
|
|
base::FilePath image_path(path);
|
|
|
|
|
if (asar::GetAsarArchivePath(image_path, &asar_path, &relative_path)) {
|
|
|
|
|
std::shared_ptr<asar::Archive> archive =
|
|
|
|
|
asar::GetOrCreateAsarArchive(asar_path);
|
|
|
|
|
if (archive)
|
|
|
|
|
archive->CopyFileOut(relative_path, &image_path);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Load the icon from file.
|
2018-04-18 01:55:30 +00:00
|
|
|
|
return base::win::ScopedHICON(
|
|
|
|
|
static_cast<HICON>(LoadImage(NULL, image_path.value().c_str(), IMAGE_ICON,
|
|
|
|
|
size, size, LR_LOADFROMFILE)));
|
2016-05-20 02:31:02 +00:00
|
|
|
|
}
|
2015-08-21 04:06:38 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
|
void Noop(char*, void*) {}
|
2016-08-05 08:55:57 +00:00
|
|
|
|
|
2015-02-11 08:03:19 +00:00
|
|
|
|
} // namespace
|
|
|
|
|
|
2016-04-25 01:17:54 +00:00
|
|
|
|
NativeImage::NativeImage(v8::Isolate* isolate, const gfx::Image& image)
|
2017-05-02 02:29:16 +00:00
|
|
|
|
: image_(image) {
|
|
|
|
|
Init(isolate);
|
|
|
|
|
if (image_.HasRepresentation(gfx::Image::kImageRepSkia)) {
|
|
|
|
|
isolate->AdjustAmountOfExternalAllocatedMemory(
|
2018-04-18 01:55:30 +00:00
|
|
|
|
image_.ToImageSkia()->bitmap()->computeByteSize());
|
2017-05-02 02:29:16 +00:00
|
|
|
|
}
|
2016-04-25 01:17:54 +00:00
|
|
|
|
}
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
2016-05-20 02:31:02 +00:00
|
|
|
|
#if defined(OS_WIN)
|
2016-05-20 08:51:05 +00:00
|
|
|
|
NativeImage::NativeImage(v8::Isolate* isolate, const base::FilePath& hicon_path)
|
2017-05-02 02:29:16 +00:00
|
|
|
|
: hicon_path_(hicon_path) {
|
2016-05-20 08:51:05 +00:00
|
|
|
|
// Use the 256x256 icon as fallback icon.
|
|
|
|
|
gfx::ImageSkia image_skia;
|
2019-07-30 20:49:25 +00:00
|
|
|
|
electron::util::ReadImageSkiaFromICO(&image_skia, GetHICON(256));
|
2016-05-20 08:51:05 +00:00
|
|
|
|
image_ = gfx::Image(image_skia);
|
2016-05-20 02:31:02 +00:00
|
|
|
|
Init(isolate);
|
2017-05-11 09:38:34 +00:00
|
|
|
|
if (image_.HasRepresentation(gfx::Image::kImageRepSkia)) {
|
|
|
|
|
isolate->AdjustAmountOfExternalAllocatedMemory(
|
2018-04-18 01:55:30 +00:00
|
|
|
|
image_.ToImageSkia()->bitmap()->computeByteSize());
|
2017-05-11 09:38:34 +00:00
|
|
|
|
}
|
2016-05-20 02:31:02 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2017-04-12 14:17:07 +00:00
|
|
|
|
NativeImage::~NativeImage() {
|
2017-05-02 02:29:16 +00:00
|
|
|
|
if (image_.HasRepresentation(gfx::Image::kImageRepSkia)) {
|
2018-04-18 01:55:30 +00:00
|
|
|
|
isolate()->AdjustAmountOfExternalAllocatedMemory(-static_cast<int64_t>(
|
|
|
|
|
image_.ToImageSkia()->bitmap()->computeByteSize()));
|
2017-05-02 02:29:16 +00:00
|
|
|
|
}
|
2017-04-12 14:17:07 +00:00
|
|
|
|
}
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
2016-05-20 07:55:22 +00:00
|
|
|
|
#if defined(OS_WIN)
|
2016-05-20 08:51:05 +00:00
|
|
|
|
HICON NativeImage::GetHICON(int size) {
|
|
|
|
|
auto iter = hicons_.find(size);
|
|
|
|
|
if (iter != hicons_.end())
|
|
|
|
|
return iter->second.get();
|
|
|
|
|
|
|
|
|
|
// First try loading the icon with specified size.
|
|
|
|
|
if (!hicon_path_.empty()) {
|
2018-06-25 20:30:00 +00:00
|
|
|
|
hicons_[size] = ReadICOFromPath(size, hicon_path_);
|
2016-05-20 08:51:05 +00:00
|
|
|
|
return hicons_[size].get();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Then convert the image to ICO.
|
2016-05-20 07:55:22 +00:00
|
|
|
|
if (image_.IsEmpty())
|
|
|
|
|
return NULL;
|
2018-06-25 20:30:00 +00:00
|
|
|
|
hicons_[size] = IconUtil::CreateHICONFromSkBitmap(image_.AsBitmap());
|
2016-05-20 08:51:05 +00:00
|
|
|
|
return hicons_[size].get();
|
2016-05-20 07:55:22 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
v8::Local<v8::Value> NativeImage::ToPNG(gin::Arguments* args) {
|
2017-03-07 00:29:39 +00:00
|
|
|
|
float scale_factor = GetScaleFactorFromOptions(args);
|
2017-03-07 00:19:16 +00:00
|
|
|
|
|
|
|
|
|
if (scale_factor == 1.0f) {
|
|
|
|
|
// Use raw 1x PNG bytes when available
|
|
|
|
|
scoped_refptr<base::RefCountedMemory> png = image_.As1xPNGBytes();
|
|
|
|
|
if (png->size() > 0) {
|
|
|
|
|
const char* data = reinterpret_cast<const char*>(png->front());
|
|
|
|
|
size_t size = png->size();
|
|
|
|
|
return node::Buffer::Copy(args->isolate(), data, size).ToLocalChecked();
|
|
|
|
|
}
|
2017-03-06 21:47:37 +00:00
|
|
|
|
}
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
2017-03-07 00:19:16 +00:00
|
|
|
|
const SkBitmap bitmap =
|
2019-01-12 01:00:43 +00:00
|
|
|
|
image_.AsImageSkia().GetRepresentation(scale_factor).GetBitmap();
|
2017-07-31 06:45:59 +00:00
|
|
|
|
std::vector<unsigned char> encoded;
|
|
|
|
|
gfx::PNGCodec::EncodeBGRASkBitmap(bitmap, false, &encoded);
|
|
|
|
|
const char* data = reinterpret_cast<char*>(encoded.data());
|
|
|
|
|
size_t size = encoded.size();
|
2017-03-07 00:19:16 +00:00
|
|
|
|
return node::Buffer::Copy(args->isolate(), data, size).ToLocalChecked();
|
|
|
|
|
}
|
2017-01-25 21:22:44 +00:00
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
v8::Local<v8::Value> NativeImage::ToBitmap(gin::Arguments* args) {
|
2017-03-07 00:29:39 +00:00
|
|
|
|
float scale_factor = GetScaleFactorFromOptions(args);
|
2017-03-07 00:19:16 +00:00
|
|
|
|
|
|
|
|
|
const SkBitmap bitmap =
|
2019-01-12 01:00:43 +00:00
|
|
|
|
image_.AsImageSkia().GetRepresentation(scale_factor).GetBitmap();
|
2017-03-07 00:19:16 +00:00
|
|
|
|
SkPixelRef* ref = bitmap.pixelRef();
|
|
|
|
|
if (!ref)
|
|
|
|
|
return node::Buffer::New(args->isolate(), 0).ToLocalChecked();
|
|
|
|
|
return node::Buffer::Copy(args->isolate(),
|
2016-07-31 03:11:18 +00:00
|
|
|
|
reinterpret_cast<const char*>(ref->pixels()),
|
2018-04-18 01:55:30 +00:00
|
|
|
|
bitmap.computeByteSize())
|
|
|
|
|
.ToLocalChecked();
|
2016-07-31 03:11:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-05-22 11:11:22 +00:00
|
|
|
|
v8::Local<v8::Value> NativeImage::ToJPEG(v8::Isolate* isolate, int quality) {
|
2015-02-11 09:47:54 +00:00
|
|
|
|
std::vector<unsigned char> output;
|
|
|
|
|
gfx::JPEG1xEncodedDataFromImage(image_, quality, &output);
|
2017-07-31 06:48:54 +00:00
|
|
|
|
if (output.empty())
|
|
|
|
|
return node::Buffer::New(isolate, 0).ToLocalChecked();
|
2018-04-18 01:55:30 +00:00
|
|
|
|
return node::Buffer::Copy(isolate,
|
|
|
|
|
reinterpret_cast<const char*>(&output.front()),
|
|
|
|
|
output.size())
|
|
|
|
|
.ToLocalChecked();
|
2015-02-11 09:47:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
std::string NativeImage::ToDataURL(gin::Arguments* args) {
|
2017-03-07 00:29:39 +00:00
|
|
|
|
float scale_factor = GetScaleFactorFromOptions(args);
|
2017-03-07 00:19:16 +00:00
|
|
|
|
|
|
|
|
|
if (scale_factor == 1.0f) {
|
|
|
|
|
// Use raw 1x PNG bytes when available
|
|
|
|
|
scoped_refptr<base::RefCountedMemory> png = image_.As1xPNGBytes();
|
|
|
|
|
if (png->size() > 0)
|
|
|
|
|
return webui::GetPngDataUrl(png->front(), png->size());
|
|
|
|
|
}
|
2015-02-12 06:32:51 +00:00
|
|
|
|
|
2017-03-07 00:19:16 +00:00
|
|
|
|
return webui::GetBitmapDataUrl(
|
2019-01-12 01:00:43 +00:00
|
|
|
|
image_.AsImageSkia().GetRepresentation(scale_factor).GetBitmap());
|
2017-03-07 00:19:16 +00:00
|
|
|
|
}
|
2017-01-25 21:22:44 +00:00
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
v8::Local<v8::Value> NativeImage::GetBitmap(gin::Arguments* args) {
|
2017-03-07 00:29:39 +00:00
|
|
|
|
float scale_factor = GetScaleFactorFromOptions(args);
|
2017-03-07 00:19:16 +00:00
|
|
|
|
|
|
|
|
|
const SkBitmap bitmap =
|
2019-01-12 01:00:43 +00:00
|
|
|
|
image_.AsImageSkia().GetRepresentation(scale_factor).GetBitmap();
|
2017-03-07 00:19:16 +00:00
|
|
|
|
SkPixelRef* ref = bitmap.pixelRef();
|
|
|
|
|
if (!ref)
|
|
|
|
|
return node::Buffer::New(args->isolate(), 0).ToLocalChecked();
|
|
|
|
|
return node::Buffer::New(args->isolate(),
|
2016-08-05 08:55:57 +00:00
|
|
|
|
reinterpret_cast<char*>(ref->pixels()),
|
2018-04-18 01:55:30 +00:00
|
|
|
|
bitmap.computeByteSize(), &Noop, nullptr)
|
|
|
|
|
.ToLocalChecked();
|
2016-08-05 08:55:57 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
v8::Local<v8::Value> NativeImage::GetNativeHandle(
|
|
|
|
|
gin_helper::ErrorThrower thrower) {
|
2016-03-14 03:08:53 +00:00
|
|
|
|
#if defined(OS_MACOSX)
|
2018-04-18 01:55:30 +00:00
|
|
|
|
if (IsEmpty())
|
2019-10-18 00:31:29 +00:00
|
|
|
|
return node::Buffer::New(thrower.isolate(), 0).ToLocalChecked();
|
2017-01-25 21:22:44 +00:00
|
|
|
|
|
2016-03-17 13:01:31 +00:00
|
|
|
|
NSImage* ptr = image_.AsNSImage();
|
2019-10-18 00:31:29 +00:00
|
|
|
|
return node::Buffer::Copy(thrower.isolate(), reinterpret_cast<char*>(ptr),
|
2018-04-18 01:55:30 +00:00
|
|
|
|
sizeof(void*))
|
|
|
|
|
.ToLocalChecked();
|
2016-03-15 02:48:40 +00:00
|
|
|
|
#else
|
2019-10-18 00:31:29 +00:00
|
|
|
|
thrower.ThrowError("Not implemented");
|
|
|
|
|
return v8::Undefined(thrower.isolate());
|
2016-03-14 03:08:53 +00:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-11 08:49:43 +00:00
|
|
|
|
bool NativeImage::IsEmpty() {
|
|
|
|
|
return image_.IsEmpty();
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-18 16:29:24 +00:00
|
|
|
|
gfx::Size NativeImage::GetSize(const base::Optional<float> scale_factor) {
|
|
|
|
|
float sf = scale_factor.value_or(1.0f);
|
|
|
|
|
gfx::ImageSkiaRep image_rep = image_.AsImageSkia().GetRepresentation(sf);
|
|
|
|
|
|
|
|
|
|
return gfx::Size(image_rep.GetWidth(), image_rep.GetHeight());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::vector<float> NativeImage::GetScaleFactors() {
|
|
|
|
|
gfx::ImageSkia image_skia = image_.AsImageSkia();
|
|
|
|
|
return image_skia.GetSupportedScales();
|
2015-02-11 08:49:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-18 16:29:24 +00:00
|
|
|
|
float NativeImage::GetAspectRatio(const base::Optional<float> scale_factor) {
|
|
|
|
|
float sf = scale_factor.value_or(1.0f);
|
|
|
|
|
gfx::Size size = GetSize(sf);
|
2016-10-05 16:15:06 +00:00
|
|
|
|
if (size.IsEmpty())
|
|
|
|
|
return 1.f;
|
|
|
|
|
else
|
|
|
|
|
return static_cast<float>(size.width()) / static_cast<float>(size.height());
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-18 16:29:24 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::Resize(gin::Arguments* args,
|
2019-10-30 05:30:59 +00:00
|
|
|
|
base::DictionaryValue options) {
|
2020-05-18 16:29:24 +00:00
|
|
|
|
float scale_factor = GetScaleFactorFromOptions(args);
|
|
|
|
|
|
|
|
|
|
gfx::Size size = GetSize(scale_factor);
|
2016-10-04 17:26:01 +00:00
|
|
|
|
int width = size.width();
|
|
|
|
|
int height = size.height();
|
2016-10-05 16:15:06 +00:00
|
|
|
|
bool width_set = options.GetInteger("width", &width);
|
|
|
|
|
bool height_set = options.GetInteger("height", &height);
|
|
|
|
|
size.SetSize(width, height);
|
2017-04-12 18:54:03 +00:00
|
|
|
|
|
2016-10-05 16:15:06 +00:00
|
|
|
|
if (width_set && !height_set) {
|
|
|
|
|
// Scale height to preserve original aspect ratio
|
|
|
|
|
size.set_height(width);
|
2020-05-18 16:29:24 +00:00
|
|
|
|
size =
|
|
|
|
|
gfx::ScaleToRoundedSize(size, 1.f, 1.f / GetAspectRatio(scale_factor));
|
2016-10-05 16:15:06 +00:00
|
|
|
|
} else if (height_set && !width_set) {
|
|
|
|
|
// Scale width to preserve original aspect ratio
|
|
|
|
|
size.set_width(height);
|
2020-05-18 16:29:24 +00:00
|
|
|
|
size = gfx::ScaleToRoundedSize(size, GetAspectRatio(scale_factor), 1.f);
|
2016-10-05 16:15:06 +00:00
|
|
|
|
}
|
2016-10-04 17:26:01 +00:00
|
|
|
|
|
|
|
|
|
skia::ImageOperations::ResizeMethod method =
|
|
|
|
|
skia::ImageOperations::ResizeMethod::RESIZE_BEST;
|
|
|
|
|
std::string quality;
|
|
|
|
|
options.GetString("quality", &quality);
|
|
|
|
|
if (quality == "good")
|
|
|
|
|
method = skia::ImageOperations::ResizeMethod::RESIZE_GOOD;
|
|
|
|
|
else if (quality == "better")
|
|
|
|
|
method = skia::ImageOperations::ResizeMethod::RESIZE_BETTER;
|
|
|
|
|
|
|
|
|
|
gfx::ImageSkia resized = gfx::ImageSkiaOperations::CreateResizedImage(
|
|
|
|
|
image_.AsImageSkia(), method, size);
|
2020-05-18 16:29:24 +00:00
|
|
|
|
return gin::CreateHandle(
|
|
|
|
|
args->isolate(), new NativeImage(args->isolate(), gfx::Image(resized)));
|
2016-10-04 17:26:01 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::Crop(v8::Isolate* isolate,
|
|
|
|
|
const gfx::Rect& rect) {
|
2018-04-18 01:55:30 +00:00
|
|
|
|
gfx::ImageSkia cropped =
|
|
|
|
|
gfx::ImageSkiaOperations::ExtractSubset(image_.AsImageSkia(), rect);
|
2019-10-25 13:03:28 +00:00
|
|
|
|
return gin::CreateHandle(isolate,
|
|
|
|
|
new NativeImage(isolate, gfx::Image(cropped)));
|
2016-10-04 18:23:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
void NativeImage::AddRepresentation(const gin_helper::Dictionary& options) {
|
2017-03-07 21:48:12 +00:00
|
|
|
|
int width = 0;
|
|
|
|
|
int height = 0;
|
|
|
|
|
float scale_factor = 1.0f;
|
|
|
|
|
options.Get("width", &width);
|
|
|
|
|
options.Get("height", &height);
|
|
|
|
|
options.Get("scaleFactor", &scale_factor);
|
|
|
|
|
|
2017-03-07 22:24:37 +00:00
|
|
|
|
bool skia_rep_added = false;
|
|
|
|
|
gfx::ImageSkia image_skia = image_.AsImageSkia();
|
|
|
|
|
|
2017-03-07 21:48:12 +00:00
|
|
|
|
v8::Local<v8::Value> buffer;
|
2017-03-07 22:24:37 +00:00
|
|
|
|
GURL url;
|
2017-03-07 21:48:12 +00:00
|
|
|
|
if (options.Get("buffer", &buffer) && node::Buffer::HasInstance(buffer)) {
|
2019-03-26 01:13:39 +00:00
|
|
|
|
auto* data = reinterpret_cast<unsigned char*>(node::Buffer::Data(buffer));
|
|
|
|
|
auto size = node::Buffer::Length(buffer);
|
2019-07-30 20:49:25 +00:00
|
|
|
|
skia_rep_added = electron::util::AddImageSkiaRepFromBuffer(
|
|
|
|
|
&image_skia, data, size, width, height, scale_factor);
|
2017-03-07 22:24:37 +00:00
|
|
|
|
} else if (options.Get("dataURL", &url)) {
|
|
|
|
|
std::string mime_type, charset, data;
|
|
|
|
|
if (net::DataURL::Parse(url, &mime_type, &charset, &data)) {
|
2019-03-26 01:13:39 +00:00
|
|
|
|
auto* data_ptr = reinterpret_cast<const unsigned char*>(data.c_str());
|
|
|
|
|
if (mime_type == "image/png") {
|
2019-07-30 20:49:25 +00:00
|
|
|
|
skia_rep_added = electron::util::AddImageSkiaRepFromPNG(
|
|
|
|
|
&image_skia, data_ptr, data.size(), scale_factor);
|
2019-03-26 01:13:39 +00:00
|
|
|
|
} else if (mime_type == "image/jpeg") {
|
2019-07-30 20:49:25 +00:00
|
|
|
|
skia_rep_added = electron::util::AddImageSkiaRepFromJPEG(
|
|
|
|
|
&image_skia, data_ptr, data.size(), scale_factor);
|
2017-03-07 22:24:37 +00:00
|
|
|
|
}
|
2017-03-07 21:48:12 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-03-07 22:24:37 +00:00
|
|
|
|
|
|
|
|
|
// Re-initialize image when first representation is added to an empty image
|
|
|
|
|
if (skia_rep_added && IsEmpty()) {
|
|
|
|
|
gfx::Image image(image_skia);
|
2017-11-27 06:50:42 +00:00
|
|
|
|
image_ = std::move(image);
|
2017-03-07 22:24:37 +00:00
|
|
|
|
}
|
2017-03-07 21:48:12 +00:00
|
|
|
|
}
|
2016-10-04 18:23:18 +00:00
|
|
|
|
|
2015-04-13 03:53:24 +00:00
|
|
|
|
#if !defined(OS_MACOSX)
|
2018-04-18 01:55:30 +00:00
|
|
|
|
void NativeImage::SetTemplateImage(bool setAsTemplate) {}
|
2015-07-29 03:48:40 +00:00
|
|
|
|
|
2015-07-27 04:58:48 +00:00
|
|
|
|
bool NativeImage::IsTemplateImage() {
|
2015-07-29 04:36:01 +00:00
|
|
|
|
return false;
|
2015-07-27 04:58:48 +00:00
|
|
|
|
}
|
2015-04-13 03:53:24 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
2015-02-12 05:55:45 +00:00
|
|
|
|
// static
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::CreateEmpty(v8::Isolate* isolate) {
|
|
|
|
|
return gin::CreateHandle(isolate, new NativeImage(isolate, gfx::Image()));
|
2015-02-12 05:55:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-02-11 08:03:19 +00:00
|
|
|
|
// static
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::Create(v8::Isolate* isolate,
|
|
|
|
|
const gfx::Image& image) {
|
|
|
|
|
return gin::CreateHandle(isolate, new NativeImage(isolate, image));
|
2015-02-11 08:03:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-02-11 10:02:59 +00:00
|
|
|
|
// static
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::CreateFromPNG(v8::Isolate* isolate,
|
|
|
|
|
const char* buffer,
|
|
|
|
|
size_t length) {
|
2015-02-11 10:02:59 +00:00
|
|
|
|
gfx::Image image = gfx::Image::CreateFrom1xPNGBytes(
|
2015-02-12 06:27:53 +00:00
|
|
|
|
reinterpret_cast<const unsigned char*>(buffer), length);
|
2015-02-11 10:02:59 +00:00
|
|
|
|
return Create(isolate, image);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// static
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::CreateFromJPEG(v8::Isolate* isolate,
|
|
|
|
|
const char* buffer,
|
|
|
|
|
size_t length) {
|
2015-02-11 10:02:59 +00:00
|
|
|
|
gfx::Image image = gfx::ImageFrom1xJPEGEncodedData(
|
2015-02-12 06:27:53 +00:00
|
|
|
|
reinterpret_cast<const unsigned char*>(buffer), length);
|
2015-02-11 10:02:59 +00:00
|
|
|
|
return Create(isolate, image);
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-11 10:41:06 +00:00
|
|
|
|
// static
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::CreateFromPath(
|
2018-04-18 01:55:30 +00:00
|
|
|
|
v8::Isolate* isolate,
|
|
|
|
|
const base::FilePath& path) {
|
2016-03-07 22:22:16 +00:00
|
|
|
|
base::FilePath image_path = NormalizePath(path);
|
2015-08-08 14:58:05 +00:00
|
|
|
|
#if defined(OS_WIN)
|
2016-05-20 13:22:15 +00:00
|
|
|
|
if (image_path.MatchesExtension(FILE_PATH_LITERAL(".ico"))) {
|
2019-10-25 13:03:28 +00:00
|
|
|
|
return gin::CreateHandle(isolate, new NativeImage(isolate, image_path));
|
2016-05-20 13:22:15 +00:00
|
|
|
|
}
|
2015-08-08 14:58:05 +00:00
|
|
|
|
#endif
|
2016-05-20 13:22:15 +00:00
|
|
|
|
gfx::ImageSkia image_skia;
|
2019-07-30 20:49:25 +00:00
|
|
|
|
electron::util::PopulateImageSkiaRepsFromPath(&image_skia, image_path);
|
2016-05-20 13:22:15 +00:00
|
|
|
|
gfx::Image image(image_skia);
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> handle = Create(isolate, image);
|
2015-02-12 04:24:18 +00:00
|
|
|
|
#if defined(OS_MACOSX)
|
2016-05-20 13:22:15 +00:00
|
|
|
|
if (IsTemplateFilename(image_path))
|
|
|
|
|
handle->SetTemplateImage(true);
|
2015-02-12 04:24:18 +00:00
|
|
|
|
#endif
|
2016-05-20 13:22:15 +00:00
|
|
|
|
return handle;
|
2015-02-11 10:41:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-03-14 18:00:38 +00:00
|
|
|
|
// static
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::CreateFromBitmap(
|
2019-10-18 00:31:29 +00:00
|
|
|
|
gin_helper::ErrorThrower thrower,
|
2019-03-14 18:00:38 +00:00
|
|
|
|
v8::Local<v8::Value> buffer,
|
2019-10-18 00:31:29 +00:00
|
|
|
|
const gin_helper::Dictionary& options) {
|
2019-03-14 18:00:38 +00:00
|
|
|
|
if (!node::Buffer::HasInstance(buffer)) {
|
2019-10-18 00:31:29 +00:00
|
|
|
|
thrower.ThrowError("buffer must be a node Buffer");
|
2019-10-25 13:03:28 +00:00
|
|
|
|
return gin::Handle<NativeImage>();
|
2019-03-14 18:00:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned int width = 0;
|
|
|
|
|
unsigned int height = 0;
|
|
|
|
|
double scale_factor = 1.;
|
|
|
|
|
|
|
|
|
|
if (!options.Get("width", &width)) {
|
2019-10-18 00:31:29 +00:00
|
|
|
|
thrower.ThrowError("width is required");
|
2019-10-25 13:03:28 +00:00
|
|
|
|
return gin::Handle<NativeImage>();
|
2019-03-14 18:00:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!options.Get("height", &height)) {
|
2019-10-18 00:31:29 +00:00
|
|
|
|
thrower.ThrowError("height is required");
|
2019-10-25 13:03:28 +00:00
|
|
|
|
return gin::Handle<NativeImage>();
|
2019-03-14 18:00:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
auto info = SkImageInfo::MakeN32(width, height, kPremul_SkAlphaType);
|
|
|
|
|
auto size_bytes = info.computeMinByteSize();
|
|
|
|
|
|
|
|
|
|
if (size_bytes != node::Buffer::Length(buffer)) {
|
2019-10-18 00:31:29 +00:00
|
|
|
|
thrower.ThrowError("invalid buffer size");
|
2019-10-25 13:03:28 +00:00
|
|
|
|
return gin::Handle<NativeImage>();
|
2019-03-14 18:00:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
options.Get("scaleFactor", &scale_factor);
|
|
|
|
|
|
|
|
|
|
if (width == 0 || height == 0) {
|
2019-10-18 00:31:29 +00:00
|
|
|
|
return CreateEmpty(thrower.isolate());
|
2019-03-14 18:00:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SkBitmap bitmap;
|
|
|
|
|
bitmap.allocN32Pixels(width, height, false);
|
2019-04-18 00:16:06 +00:00
|
|
|
|
bitmap.writePixels({info, node::Buffer::Data(buffer), bitmap.rowBytes()});
|
2019-03-14 18:00:38 +00:00
|
|
|
|
|
|
|
|
|
gfx::ImageSkia image_skia;
|
|
|
|
|
image_skia.AddRepresentation(gfx::ImageSkiaRep(bitmap, scale_factor));
|
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
return Create(thrower.isolate(), gfx::Image(image_skia));
|
2019-03-14 18:00:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 07:19:05 +00:00
|
|
|
|
// static
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::CreateFromBuffer(
|
2019-10-18 00:31:29 +00:00
|
|
|
|
gin_helper::ErrorThrower thrower,
|
|
|
|
|
v8::Local<v8::Value> buffer,
|
|
|
|
|
gin::Arguments* args) {
|
2019-03-13 15:26:11 +00:00
|
|
|
|
if (!node::Buffer::HasInstance(buffer)) {
|
2019-10-18 00:31:29 +00:00
|
|
|
|
thrower.ThrowError("buffer must be a node Buffer");
|
2019-10-25 13:03:28 +00:00
|
|
|
|
return gin::Handle<NativeImage>();
|
2019-03-13 15:26:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-09 23:12:36 +00:00
|
|
|
|
int width = 0;
|
|
|
|
|
int height = 0;
|
2015-02-12 07:19:05 +00:00
|
|
|
|
double scale_factor = 1.;
|
2016-12-13 20:30:45 +00:00
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
gin_helper::Dictionary options;
|
2016-12-13 20:30:45 +00:00
|
|
|
|
if (args->GetNext(&options)) {
|
|
|
|
|
options.Get("width", &width);
|
|
|
|
|
options.Get("height", &height);
|
|
|
|
|
options.Get("scaleFactor", &scale_factor);
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 07:19:05 +00:00
|
|
|
|
gfx::ImageSkia image_skia;
|
2019-07-30 20:49:25 +00:00
|
|
|
|
electron::util::AddImageSkiaRepFromBuffer(
|
2019-03-26 01:13:39 +00:00
|
|
|
|
&image_skia, reinterpret_cast<unsigned char*>(node::Buffer::Data(buffer)),
|
|
|
|
|
node::Buffer::Length(buffer), width, height, scale_factor);
|
2015-02-12 07:19:05 +00:00
|
|
|
|
return Create(args->isolate(), gfx::Image(image_skia));
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 06:27:53 +00:00
|
|
|
|
// static
|
2019-10-25 13:03:28 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::CreateFromDataURL(v8::Isolate* isolate,
|
|
|
|
|
const GURL& url) {
|
2015-02-12 06:27:53 +00:00
|
|
|
|
std::string mime_type, charset, data;
|
|
|
|
|
if (net::DataURL::Parse(url, &mime_type, &charset, &data)) {
|
|
|
|
|
if (mime_type == "image/png")
|
|
|
|
|
return CreateFromPNG(isolate, data.c_str(), data.size());
|
|
|
|
|
else if (mime_type == "image/jpeg")
|
|
|
|
|
return CreateFromJPEG(isolate, data.c_str(), data.size());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return CreateEmpty(isolate);
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-09 16:13:40 +00:00
|
|
|
|
#if !defined(OS_MACOSX)
|
2020-05-11 01:24:45 +00:00
|
|
|
|
gin::Handle<NativeImage> NativeImage::CreateFromNamedImage(gin::Arguments* args,
|
|
|
|
|
std::string name) {
|
2017-10-09 16:13:40 +00:00
|
|
|
|
return CreateEmpty(args->isolate());
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2016-04-25 01:17:54 +00:00
|
|
|
|
// static
|
2018-04-18 01:55:30 +00:00
|
|
|
|
void NativeImage::BuildPrototype(v8::Isolate* isolate,
|
|
|
|
|
v8::Local<v8::FunctionTemplate> prototype) {
|
2019-10-18 00:31:29 +00:00
|
|
|
|
prototype->SetClassName(gin::StringToV8(isolate, "NativeImage"));
|
|
|
|
|
gin_helper::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
|
2016-06-22 22:46:22 +00:00
|
|
|
|
.SetMethod("toPNG", &NativeImage::ToPNG)
|
|
|
|
|
.SetMethod("toJPEG", &NativeImage::ToJPEG)
|
2016-08-01 00:13:31 +00:00
|
|
|
|
.SetMethod("toBitmap", &NativeImage::ToBitmap)
|
2016-08-04 17:36:01 +00:00
|
|
|
|
.SetMethod("getBitmap", &NativeImage::GetBitmap)
|
2020-05-18 16:29:24 +00:00
|
|
|
|
.SetMethod("getScaleFactors", &NativeImage::GetScaleFactors)
|
2016-04-25 01:17:54 +00:00
|
|
|
|
.SetMethod("getNativeHandle", &NativeImage::GetNativeHandle)
|
|
|
|
|
.SetMethod("toDataURL", &NativeImage::ToDataURL)
|
|
|
|
|
.SetMethod("isEmpty", &NativeImage::IsEmpty)
|
|
|
|
|
.SetMethod("getSize", &NativeImage::GetSize)
|
2020-03-18 01:06:52 +00:00
|
|
|
|
.SetMethod("setTemplateImage", &NativeImage::SetTemplateImage)
|
|
|
|
|
.SetMethod("isTemplateImage", &NativeImage::IsTemplateImage)
|
2019-05-07 13:52:07 +00:00
|
|
|
|
.SetProperty("isMacTemplateImage", &NativeImage::IsTemplateImage,
|
|
|
|
|
&NativeImage::SetTemplateImage)
|
2016-10-04 17:26:01 +00:00
|
|
|
|
.SetMethod("resize", &NativeImage::Resize)
|
2016-10-04 18:23:18 +00:00
|
|
|
|
.SetMethod("crop", &NativeImage::Crop)
|
2016-10-05 16:15:06 +00:00
|
|
|
|
.SetMethod("getAspectRatio", &NativeImage::GetAspectRatio)
|
2018-02-20 15:15:27 +00:00
|
|
|
|
.SetMethod("addRepresentation", &NativeImage::AddRepresentation);
|
2016-04-25 01:17:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-02-11 08:03:19 +00:00
|
|
|
|
} // namespace api
|
|
|
|
|
|
2019-06-19 21:23:04 +00:00
|
|
|
|
} // namespace electron
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
namespace gin {
|
2016-05-20 07:14:40 +00:00
|
|
|
|
|
2019-10-25 13:03:28 +00:00
|
|
|
|
v8::Local<v8::Value> Converter<electron::api::NativeImage*>::ToV8(
|
2016-05-20 07:14:40 +00:00
|
|
|
|
v8::Isolate* isolate,
|
2019-10-25 13:03:28 +00:00
|
|
|
|
electron::api::NativeImage* val) {
|
|
|
|
|
if (val)
|
|
|
|
|
return val->GetWrapper();
|
|
|
|
|
else
|
|
|
|
|
return v8::Null(isolate);
|
2016-05-20 07:14:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-25 13:03:28 +00:00
|
|
|
|
bool Converter<electron::api::NativeImage*>::FromV8(
|
2018-04-18 01:55:30 +00:00
|
|
|
|
v8::Isolate* isolate,
|
|
|
|
|
v8::Local<v8::Value> val,
|
2019-10-25 13:03:28 +00:00
|
|
|
|
electron::api::NativeImage** out) {
|
2016-05-20 07:14:40 +00:00
|
|
|
|
// Try converting from file path.
|
|
|
|
|
base::FilePath path;
|
|
|
|
|
if (ConvertFromV8(isolate, val, &path)) {
|
2019-10-25 13:03:28 +00:00
|
|
|
|
*out = electron::api::NativeImage::CreateFromPath(isolate, path).get();
|
2019-11-23 03:16:43 +00:00
|
|
|
|
if ((*out)->image().IsEmpty()) {
|
|
|
|
|
#if defined(OS_WIN)
|
|
|
|
|
const auto img_path = base::UTF16ToUTF8(path.value());
|
|
|
|
|
#else
|
|
|
|
|
const auto img_path = path.value();
|
|
|
|
|
#endif
|
|
|
|
|
isolate->ThrowException(v8::Exception::Error(
|
|
|
|
|
StringToV8(isolate, "Image could not be created from " + img_path)));
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
2016-05-20 07:14:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-25 13:03:28 +00:00
|
|
|
|
*out = static_cast<electron::api::NativeImage*>(
|
2019-12-05 09:46:34 +00:00
|
|
|
|
static_cast<gin_helper::WrappableBase*>(
|
|
|
|
|
gin_helper::internal::FromV8Impl(isolate, val)));
|
2019-10-25 13:03:28 +00:00
|
|
|
|
return *out != nullptr;
|
2016-05-20 07:14:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-18 00:31:29 +00:00
|
|
|
|
} // namespace gin
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
2019-06-19 21:23:04 +00:00
|
|
|
|
using electron::api::NativeImage;
|
2019-03-16 00:32:04 +00:00
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
|
void Initialize(v8::Local<v8::Object> exports,
|
|
|
|
|
v8::Local<v8::Value> unused,
|
|
|
|
|
v8::Local<v8::Context> context,
|
|
|
|
|
void* priv) {
|
2019-03-16 00:32:04 +00:00
|
|
|
|
v8::Isolate* isolate = context->GetIsolate();
|
2019-10-18 00:31:29 +00:00
|
|
|
|
gin_helper::Dictionary dict(isolate, exports);
|
2019-03-16 00:32:04 +00:00
|
|
|
|
dict.Set("NativeImage", NativeImage::GetConstructor(isolate)
|
|
|
|
|
->GetFunction(context)
|
|
|
|
|
.ToLocalChecked());
|
2019-10-18 00:31:29 +00:00
|
|
|
|
gin_helper::Dictionary native_image = gin::Dictionary::CreateEmpty(isolate);
|
2019-03-16 00:32:04 +00:00
|
|
|
|
dict.Set("nativeImage", native_image);
|
|
|
|
|
|
|
|
|
|
native_image.SetMethod("createEmpty", &NativeImage::CreateEmpty);
|
|
|
|
|
native_image.SetMethod("createFromPath", &NativeImage::CreateFromPath);
|
|
|
|
|
native_image.SetMethod("createFromBitmap", &NativeImage::CreateFromBitmap);
|
|
|
|
|
native_image.SetMethod("createFromBuffer", &NativeImage::CreateFromBuffer);
|
|
|
|
|
native_image.SetMethod("createFromDataURL", &NativeImage::CreateFromDataURL);
|
|
|
|
|
native_image.SetMethod("createFromNamedImage",
|
|
|
|
|
&NativeImage::CreateFromNamedImage);
|
2015-02-11 08:03:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
2020-02-14 11:25:39 +00:00
|
|
|
|
NODE_LINKED_MODULE_CONTEXT_AWARE(electron_common_native_image, Initialize)
|