e86d1cba75
* chore: bump chromium in DEPS to 104.0.5096.0 * 3651284: Use the entry settings object for window.open navigation https://chromium-review.googlesource.com/c/chromium/src/+/3651284 * 3644598: Make RenderFrameHost used for notification permission decision https://chromium-review.googlesource.com/c/chromium/src/+/3644598 * 3642842: Window Placement: Prototype Fullscreen Capability Delegation - Part 2 https://chromium-review.googlesource.com/c/chromium/src/+/3642842 * 3652785: [sandbox] Enable sandboxed pointers on sanitizer builds https://chromium-review.googlesource.com/c/v8/v8/+/3652785 * 3611967: webhid: Migrate HidDelegate to use BrowserContext and Origin https://chromium-review.googlesource.com/c/chromium/src/+/3611967 * 3665762: Remove RenderFrameHost::IsRenderFrameCreated from //content/. https://chromium-review.googlesource.com/c/chromium/src/+/3665762 * 3659375: Fold x509_util_ios and most of x509_util_mac into x509_util_apple https://chromium-review.googlesource.com/c/chromium/src/+/3659375 * 3656234: [CodeHealth] Remove uses of base::ListValue::Append() (Final, take 2) https://chromium-review.googlesource.com/c/chromium/src/+/3656234 * chore: update patch indices * chore: fix lint * 3644598: Make RenderFrameHost used for notification permission decision https://chromium-review.googlesource.com/c/chromium/src/+/3644598 * webhid: Migrate HidDelegate to use BrowserContext and Origin This is a temporary fix for https://chromium-review.googlesource.com/c/chromium/src/+/3611967 to get the build compiling, but we need to either patch around https://chromium-review.googlesource.com/c/chromium/src/+/3611967 or move our device permissioning to BrowserContext * chore: fix lint * build: run electron/script/gen-libc++-filenames.js fixes gn check * chore: bump chromium in DEPS to 104.0.5098.0 * chore: disable flaking tests * 3682394: Change pipes surrounding code references in comments to backticks https://chromium-review.googlesource.com/c/chromium/src/+/3682394 * 3652749: Delete GLRenderer and related classes https://chromium-review.googlesource.com/c/chromium/src/+/3652749 * chore: fixup patch indices * 3671199: Remove ContentMainDelegate::PostFieldTrialInitialization https://chromium-review.googlesource.com/c/chromium/src/+/3671199 * 3607963: hid: Do not exclude HID device with no collections https://chromium-review.googlesource.com/c/chromium/src/+/3607963 * refactor: use ElectronBrowserContext instead of WebContents to persist devices due to changes like https://chromium-review.googlesource.com/c/chromium/src/+/3611967, we can no longer use WebContents to store device permissions so this commit moves device permission storage to live in memory in ElectronBrowserContext instead. * 3557253: Deprecate some signature checks https://chromium-review.googlesource.com/c/v8/v8/+/3557253 * chore: bump chromium in DEPS to 104.0.5100.0 * 3680781: Add policy for Renderer App Container. https://chromium-review.googlesource.com/c/chromium/src/+/3680781 * chore: update patch indices * 3675465: Update NetLog network service API to use mojom::DictionaryValue. https://chromium-review.googlesource.com/c/chromium/src/+/3675465 * chore: bump chromium in DEPS to 104.0.5102.0 * chore: update patches * chore: bump chromium in DEPS to 104.0.5103.0 * chore: update patches * chore: bump chromium in DEPS to 104.0.5104.0 * chore: update patches * fix: add patch for DCHECK in fullscreen test * build: fix nan build * build: make the nan spec runner work on macOS * chore: bump chromium in DEPS to 104.0.5106.0 * chore: update patches * chore: bump chromium in DEPS to 104.0.5108.0 * chore: update patches * chore: bump chromium in DEPS to 104.0.5110.0 * chore: update patches * chore: bump chromium in DEPS to 104.0.5112.0 * chore: bump chromium in DEPS to 105.0.5113.0 * chore: bump chromium in DEPS to 105.0.5115.0 * chore: bump chromium in DEPS to 105.0.5117.0 * chore: update patches * chore: update libcpp patch * 3693745: Delete base::LowerCaseEqualsASCII() Ref: https://chromium-review.googlesource.com/c/chromium/src/+/3693745 * 3669226: Remove printing PostTask usage of IO thread Ref: https://chromium-review.googlesource.com/c/chromium/src/+/3669226 * 3687395: Remove DictionaryValue::HasKey(). Ref: https://chromium-review.googlesource.com/c/chromium/src/+/3687395 * 3691014: Prevent unsafe narrowing: ui/accessibility, part 2 Ref: https://chromium-review.googlesource.com/c/chromium/src/+/3691014 * 3560567: [MSC] Porting GenerateStreams clients to handle stream vectors. Ref: https://chromium-review.googlesource.com/c/chromium/src/+/3560567 * 3684873: [Bluetooth][Win/Linux] Add bluetooth pair confirmation prompt https://chromium-review.googlesource.com/c/chromium/src/+/3684873 * chore: bump chromium in DEPS to 105.0.5119.0 * chore: missing includes in desktop_notification_controller * chore: update patches * 3685951: Reland "Make sure screen object is created once in tests." https://chromium-review.googlesource.com/c/chromium/src/+/3685951 * fixup: Reland "Make sure screen object is created once in tests." * 3646014: [API] Deprecate LegacyOOMErrorCallback Ref: https://chromium-review.googlesource.com/c/v8/v8/+/3646014 * chore: bump chromium in DEPS to 105.0.5121.0 * chore: update patches * 3699085: [cleanup] update PrintBackend::EnumeratePrinters to use reference Ref: https://chromium-review.googlesource.com/c/chromium/src/+/3699085 * chore: bump chromium in DEPS to 105.0.5123.0 * chore: update patches * chore: bump chromium in DEPS to 105.0.5125.0 * chore: update patches * 3630082: [sandbox] Also enable the sandbox outside of Chromium builds Ref: https://chromium-review.googlesource.com/c/v8/v8/+/3630082 * chore: bump chromium in DEPS to 105.0.5127.0 * chore: update patches * chore: bump chromium in DEPS to 105.0.5129.0 * chore: update patches * 3703741: Remove WebContents::GetMainFrame. https://chromium-review.googlesource.com/c/chromium/src/+/3703741 * chore: update patches * fixup! 3703741: Remove WebContents::GetMainFrame. * fix lint * more lint * chore: document breaking change * 3687671: [v8] Freeze flags after initialization https://chromium-review.googlesource.com/c/chromium/src/+/3687671 * fixup! 3560567: [MSC] Porting GenerateStreams clients to handle stream vectors. * use the v8 allocator for node serdes * chore: update patches * remove faulty non-v8-sandbox-compatible code * make NodeArrayBufferAllocator use the v8 allocator under the hood * fixup! 3560567: [MSC] Porting GenerateStreams clients to handle stream vectors. * fix build on windows * 3691954: [Reland][Extensions Bindings] Validate arguments sent to API events https://chromium-review.googlesource.com/c/chromium/src/+/3691954 * chore: remove deprecated AccessorSignatures https://github.com/nodejs/nan/pull/941 * Update patches/chromium/notification_provenance.patch Co-authored-by: John Kleinschmidt <jkleinsc@electronjs.org> * remove chore_expose_v8_initialization_isolate_callbacks.patch * add link to nodejs/nan#941 * 52026: Do not allow md4 or md5 based signatures in X.509 certificates. https://boringssl-review.googlesource.com/c/boringssl/+/52026 * chore: update patches * disable nan buffer-test * disable sandboxed pointers for now * force sandboxed pointers off * improve node allocation patch * restore accidentally dropped node crypto test patch Co-authored-by: electron-roller[bot] <84116207+electron-roller[bot]@users.noreply.github.com> Co-authored-by: Shelley Vohr <shelley.vohr@gmail.com> Co-authored-by: John Kleinschmidt <jkleinsc@electronjs.org> Co-authored-by: Charles Kerr <charles@charleskerr.com> Co-authored-by: PatchUp <73610968+patchup[bot]@users.noreply.github.com> Co-authored-by: Samuel Attard <sattard@salesforce.com> Co-authored-by: Keeley Hammond <khammond@slack-corp.com> Co-authored-by: VerteDinde <keeleymhammond@gmail.com> Co-authored-by: VerteDinde <vertedinde@electronjs.org> Co-authored-by: Jeremy Rose <jeremya@chromium.org> Co-authored-by: Jeremy Rose <japthorp@slack-corp.com>
648 lines
22 KiB
C++
648 lines
22 KiB
C++
// Copyright (c) 2015 GitHub, Inc.
|
||
// Use of this source code is governed by the MIT license that can be
|
||
// found in the LICENSE file.
|
||
|
||
#include "shell/common/api/electron_api_native_image.h"
|
||
|
||
#include <memory>
|
||
#include <string>
|
||
#include <utility>
|
||
#include <vector>
|
||
|
||
#include "base/files/file_util.h"
|
||
#include "base/logging.h"
|
||
#include "base/strings/pattern.h"
|
||
#include "base/strings/string_util.h"
|
||
#include "base/strings/utf_string_conversions.h"
|
||
#include "base/threading/thread_restrictions.h"
|
||
#include "gin/arguments.h"
|
||
#include "gin/object_template_builder.h"
|
||
#include "gin/per_isolate_data.h"
|
||
#include "gin/wrappable.h"
|
||
#include "net/base/data_url.h"
|
||
#include "shell/common/asar/asar_util.h"
|
||
#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"
|
||
#include "shell/common/gin_converters/value_converter.h"
|
||
#include "shell/common/gin_helper/dictionary.h"
|
||
#include "shell/common/gin_helper/function_template_extensions.h"
|
||
#include "shell/common/gin_helper/object_template_builder.h"
|
||
#include "shell/common/node_includes.h"
|
||
#include "shell/common/skia_util.h"
|
||
#include "third_party/skia/include/core/SkBitmap.h"
|
||
#include "third_party/skia/include/core/SkImageInfo.h"
|
||
#include "third_party/skia/include/core/SkPixelRef.h"
|
||
#include "ui/base/layout.h"
|
||
#include "ui/base/webui/web_ui_util.h"
|
||
#include "ui/gfx/codec/jpeg_codec.h"
|
||
#include "ui/gfx/codec/png_codec.h"
|
||
#include "ui/gfx/geometry/size.h"
|
||
#include "ui/gfx/image/image_skia.h"
|
||
#include "ui/gfx/image/image_skia_operations.h"
|
||
#include "ui/gfx/image/image_util.h"
|
||
|
||
#if BUILDFLAG(IS_WIN)
|
||
#include "base/win/scoped_gdi_object.h"
|
||
#include "shell/common/asar/archive.h"
|
||
#include "ui/gfx/icon_util.h"
|
||
#endif
|
||
|
||
namespace electron {
|
||
|
||
namespace api {
|
||
|
||
namespace {
|
||
|
||
// Get the scale factor from options object at the first argument
|
||
float GetScaleFactorFromOptions(gin::Arguments* args) {
|
||
float scale_factor = 1.0f;
|
||
gin_helper::Dictionary options;
|
||
if (args->GetNext(&options))
|
||
options.Get("scaleFactor", &scale_factor);
|
||
return scale_factor;
|
||
}
|
||
|
||
base::FilePath NormalizePath(const base::FilePath& path) {
|
||
if (!path.ReferencesParent()) {
|
||
return path;
|
||
}
|
||
|
||
base::ThreadRestrictions::ScopedAllowIO allow_blocking;
|
||
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;
|
||
}
|
||
}
|
||
|
||
#if BUILDFLAG(IS_MAC)
|
||
bool IsTemplateFilename(const base::FilePath& path) {
|
||
return (base::MatchPattern(path.value(), "*Template.*") ||
|
||
base::MatchPattern(path.value(), "*Template@*x.*"));
|
||
}
|
||
#endif
|
||
|
||
#if BUILDFLAG(IS_WIN)
|
||
base::win::ScopedHICON ReadICOFromPath(int size, const base::FilePath& path) {
|
||
// 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.
|
||
return base::win::ScopedHICON(
|
||
static_cast<HICON>(LoadImage(NULL, image_path.value().c_str(), IMAGE_ICON,
|
||
size, size, LR_LOADFROMFILE)));
|
||
}
|
||
#endif
|
||
|
||
} // namespace
|
||
|
||
NativeImage::NativeImage(v8::Isolate* isolate, const gfx::Image& image)
|
||
: image_(image), isolate_(isolate) {
|
||
UpdateExternalAllocatedMemoryUsage();
|
||
}
|
||
|
||
#if BUILDFLAG(IS_WIN)
|
||
NativeImage::NativeImage(v8::Isolate* isolate, const base::FilePath& hicon_path)
|
||
: hicon_path_(hicon_path), isolate_(isolate) {
|
||
// Use the 256x256 icon as fallback icon.
|
||
gfx::ImageSkia image_skia;
|
||
electron::util::ReadImageSkiaFromICO(&image_skia, GetHICON(256));
|
||
image_ = gfx::Image(image_skia);
|
||
|
||
UpdateExternalAllocatedMemoryUsage();
|
||
}
|
||
#endif
|
||
|
||
NativeImage::~NativeImage() {
|
||
isolate_->AdjustAmountOfExternalAllocatedMemory(-memory_usage_);
|
||
}
|
||
|
||
void NativeImage::UpdateExternalAllocatedMemoryUsage() {
|
||
int32_t new_memory_usage = 0;
|
||
|
||
if (image_.HasRepresentation(gfx::Image::kImageRepSkia)) {
|
||
auto* const image_skia = image_.ToImageSkia();
|
||
if (!image_skia->isNull()) {
|
||
new_memory_usage = image_skia->bitmap()->computeByteSize();
|
||
}
|
||
}
|
||
|
||
isolate_->AdjustAmountOfExternalAllocatedMemory(new_memory_usage -
|
||
memory_usage_);
|
||
memory_usage_ = new_memory_usage;
|
||
}
|
||
|
||
// static
|
||
bool NativeImage::TryConvertNativeImage(v8::Isolate* isolate,
|
||
v8::Local<v8::Value> image,
|
||
NativeImage** native_image,
|
||
OnConvertError on_error) {
|
||
std::string error_message;
|
||
|
||
base::FilePath icon_path;
|
||
if (gin::ConvertFromV8(isolate, image, &icon_path)) {
|
||
*native_image = NativeImage::CreateFromPath(isolate, icon_path).get();
|
||
if ((*native_image)->image().IsEmpty()) {
|
||
#if BUILDFLAG(IS_WIN)
|
||
const auto img_path = base::WideToUTF8(icon_path.value());
|
||
#else
|
||
const auto img_path = icon_path.value();
|
||
#endif
|
||
error_message = "Failed to load image from path '" + img_path + "'";
|
||
}
|
||
} else {
|
||
if (!gin::ConvertFromV8(isolate, image, native_image)) {
|
||
error_message = "Argument must be a file path or a NativeImage";
|
||
}
|
||
}
|
||
|
||
if (!error_message.empty()) {
|
||
switch (on_error) {
|
||
case OnConvertError::kThrow:
|
||
isolate->ThrowException(
|
||
v8::Exception::Error(gin::StringToV8(isolate, error_message)));
|
||
break;
|
||
case OnConvertError::kWarn:
|
||
LOG(WARNING) << error_message;
|
||
break;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
#if BUILDFLAG(IS_WIN)
|
||
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()) {
|
||
hicons_[size] = ReadICOFromPath(size, hicon_path_);
|
||
return hicons_[size].get();
|
||
}
|
||
|
||
// Then convert the image to ICO.
|
||
if (image_.IsEmpty())
|
||
return NULL;
|
||
hicons_[size] = IconUtil::CreateHICONFromSkBitmap(image_.AsBitmap());
|
||
return hicons_[size].get();
|
||
}
|
||
#endif
|
||
|
||
v8::Local<v8::Value> NativeImage::ToPNG(gin::Arguments* args) {
|
||
float scale_factor = GetScaleFactorFromOptions(args);
|
||
|
||
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();
|
||
}
|
||
}
|
||
|
||
const SkBitmap bitmap =
|
||
image_.AsImageSkia().GetRepresentation(scale_factor).GetBitmap();
|
||
std::vector<unsigned char> encoded;
|
||
gfx::PNGCodec::EncodeBGRASkBitmap(bitmap, false, &encoded);
|
||
const char* data = reinterpret_cast<char*>(encoded.data());
|
||
size_t size = encoded.size();
|
||
return node::Buffer::Copy(args->isolate(), data, size).ToLocalChecked();
|
||
}
|
||
|
||
v8::Local<v8::Value> NativeImage::ToBitmap(gin::Arguments* args) {
|
||
float scale_factor = GetScaleFactorFromOptions(args);
|
||
|
||
const SkBitmap bitmap =
|
||
image_.AsImageSkia().GetRepresentation(scale_factor).GetBitmap();
|
||
|
||
SkImageInfo info =
|
||
SkImageInfo::MakeN32Premul(bitmap.width(), bitmap.height());
|
||
|
||
auto array_buffer =
|
||
v8::ArrayBuffer::New(args->isolate(), info.computeMinByteSize());
|
||
auto backing_store = array_buffer->GetBackingStore();
|
||
if (bitmap.readPixels(info, backing_store->Data(), info.minRowBytes(), 0,
|
||
0)) {
|
||
return node::Buffer::New(args->isolate(), array_buffer, 0,
|
||
info.computeMinByteSize())
|
||
.ToLocalChecked();
|
||
}
|
||
return node::Buffer::New(args->isolate(), 0).ToLocalChecked();
|
||
}
|
||
|
||
v8::Local<v8::Value> NativeImage::ToJPEG(v8::Isolate* isolate, int quality) {
|
||
std::vector<unsigned char> output;
|
||
gfx::JPEG1xEncodedDataFromImage(image_, quality, &output);
|
||
if (output.empty())
|
||
return node::Buffer::New(isolate, 0).ToLocalChecked();
|
||
return node::Buffer::Copy(isolate,
|
||
reinterpret_cast<const char*>(&output.front()),
|
||
output.size())
|
||
.ToLocalChecked();
|
||
}
|
||
|
||
std::string NativeImage::ToDataURL(gin::Arguments* args) {
|
||
float scale_factor = GetScaleFactorFromOptions(args);
|
||
|
||
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());
|
||
}
|
||
|
||
return webui::GetBitmapDataUrl(
|
||
image_.AsImageSkia().GetRepresentation(scale_factor).GetBitmap());
|
||
}
|
||
|
||
v8::Local<v8::Value> NativeImage::GetBitmap(gin::Arguments* args) {
|
||
float scale_factor = GetScaleFactorFromOptions(args);
|
||
|
||
const SkBitmap bitmap =
|
||
image_.AsImageSkia().GetRepresentation(scale_factor).GetBitmap();
|
||
SkPixelRef* ref = bitmap.pixelRef();
|
||
if (!ref)
|
||
return node::Buffer::New(args->isolate(), 0).ToLocalChecked();
|
||
return node::Buffer::Copy(args->isolate(),
|
||
reinterpret_cast<char*>(ref->pixels()),
|
||
bitmap.computeByteSize())
|
||
.ToLocalChecked();
|
||
}
|
||
|
||
v8::Local<v8::Value> NativeImage::GetNativeHandle(
|
||
gin_helper::ErrorThrower thrower) {
|
||
#if BUILDFLAG(IS_MAC)
|
||
if (IsEmpty())
|
||
return node::Buffer::New(thrower.isolate(), 0).ToLocalChecked();
|
||
|
||
NSImage* ptr = image_.AsNSImage();
|
||
return node::Buffer::Copy(thrower.isolate(), reinterpret_cast<char*>(ptr),
|
||
sizeof(void*))
|
||
.ToLocalChecked();
|
||
#else
|
||
thrower.ThrowError("Not implemented");
|
||
return v8::Undefined(thrower.isolate());
|
||
#endif
|
||
}
|
||
|
||
bool NativeImage::IsEmpty() {
|
||
return image_.IsEmpty();
|
||
}
|
||
|
||
gfx::Size NativeImage::GetSize(const absl::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();
|
||
std::vector<float> scale_factors;
|
||
for (const auto& rep : image_skia.image_reps()) {
|
||
scale_factors.push_back(rep.scale());
|
||
}
|
||
return scale_factors;
|
||
}
|
||
|
||
float NativeImage::GetAspectRatio(const absl::optional<float> scale_factor) {
|
||
float sf = scale_factor.value_or(1.0f);
|
||
gfx::Size size = GetSize(sf);
|
||
if (size.IsEmpty())
|
||
return 1.f;
|
||
else
|
||
return static_cast<float>(size.width()) / static_cast<float>(size.height());
|
||
}
|
||
|
||
gin::Handle<NativeImage> NativeImage::Resize(gin::Arguments* args,
|
||
base::DictionaryValue options) {
|
||
float scale_factor = GetScaleFactorFromOptions(args);
|
||
|
||
gfx::Size size = GetSize(scale_factor);
|
||
int width = size.width();
|
||
int height = size.height();
|
||
bool width_set = options.GetInteger("width", &width);
|
||
bool height_set = options.GetInteger("height", &height);
|
||
size.SetSize(width, height);
|
||
|
||
if (width <= 0 && height <= 0) {
|
||
return CreateEmpty(args->isolate());
|
||
} else if (width_set && !height_set) {
|
||
// Scale height to preserve original aspect ratio
|
||
size.set_height(width);
|
||
size =
|
||
gfx::ScaleToRoundedSize(size, 1.f, 1.f / GetAspectRatio(scale_factor));
|
||
} else if (height_set && !width_set) {
|
||
// Scale width to preserve original aspect ratio
|
||
size.set_width(height);
|
||
size = gfx::ScaleToRoundedSize(size, GetAspectRatio(scale_factor), 1.f);
|
||
}
|
||
|
||
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);
|
||
return gin::CreateHandle(
|
||
args->isolate(), new NativeImage(args->isolate(), gfx::Image(resized)));
|
||
}
|
||
|
||
gin::Handle<NativeImage> NativeImage::Crop(v8::Isolate* isolate,
|
||
const gfx::Rect& rect) {
|
||
gfx::ImageSkia cropped =
|
||
gfx::ImageSkiaOperations::ExtractSubset(image_.AsImageSkia(), rect);
|
||
return gin::CreateHandle(isolate,
|
||
new NativeImage(isolate, gfx::Image(cropped)));
|
||
}
|
||
|
||
void NativeImage::AddRepresentation(const gin_helper::Dictionary& options) {
|
||
int width = 0;
|
||
int height = 0;
|
||
float scale_factor = 1.0f;
|
||
options.Get("width", &width);
|
||
options.Get("height", &height);
|
||
options.Get("scaleFactor", &scale_factor);
|
||
|
||
bool skia_rep_added = false;
|
||
gfx::ImageSkia image_skia = image_.AsImageSkia();
|
||
|
||
v8::Local<v8::Value> buffer;
|
||
GURL url;
|
||
if (options.Get("buffer", &buffer) && node::Buffer::HasInstance(buffer)) {
|
||
auto* data = reinterpret_cast<unsigned char*>(node::Buffer::Data(buffer));
|
||
auto size = node::Buffer::Length(buffer);
|
||
skia_rep_added = electron::util::AddImageSkiaRepFromBuffer(
|
||
&image_skia, data, size, width, height, scale_factor);
|
||
} else if (options.Get("dataURL", &url)) {
|
||
std::string mime_type, charset, data;
|
||
if (net::DataURL::Parse(url, &mime_type, &charset, &data)) {
|
||
auto* data_ptr = reinterpret_cast<const unsigned char*>(data.c_str());
|
||
if (mime_type == "image/png") {
|
||
skia_rep_added = electron::util::AddImageSkiaRepFromPNG(
|
||
&image_skia, data_ptr, data.size(), scale_factor);
|
||
} else if (mime_type == "image/jpeg") {
|
||
skia_rep_added = electron::util::AddImageSkiaRepFromJPEG(
|
||
&image_skia, data_ptr, data.size(), scale_factor);
|
||
}
|
||
}
|
||
}
|
||
|
||
// Re-initialize image when first representation is added to an empty image
|
||
if (skia_rep_added && IsEmpty()) {
|
||
gfx::Image image(image_skia);
|
||
image_ = std::move(image);
|
||
}
|
||
}
|
||
|
||
#if !BUILDFLAG(IS_MAC)
|
||
void NativeImage::SetTemplateImage(bool setAsTemplate) {}
|
||
|
||
bool NativeImage::IsTemplateImage() {
|
||
return false;
|
||
}
|
||
#endif
|
||
|
||
// static
|
||
gin::Handle<NativeImage> NativeImage::CreateEmpty(v8::Isolate* isolate) {
|
||
return gin::CreateHandle(isolate, new NativeImage(isolate, gfx::Image()));
|
||
}
|
||
|
||
// static
|
||
gin::Handle<NativeImage> NativeImage::Create(v8::Isolate* isolate,
|
||
const gfx::Image& image) {
|
||
return gin::CreateHandle(isolate, new NativeImage(isolate, image));
|
||
}
|
||
|
||
// static
|
||
gin::Handle<NativeImage> NativeImage::CreateFromPNG(v8::Isolate* isolate,
|
||
const char* buffer,
|
||
size_t length) {
|
||
gfx::ImageSkia image_skia;
|
||
electron::util::AddImageSkiaRepFromPNG(
|
||
&image_skia, reinterpret_cast<const unsigned char*>(buffer), length, 1.0);
|
||
return Create(isolate, gfx::Image(image_skia));
|
||
}
|
||
|
||
// static
|
||
gin::Handle<NativeImage> NativeImage::CreateFromJPEG(v8::Isolate* isolate,
|
||
const char* buffer,
|
||
size_t length) {
|
||
gfx::ImageSkia image_skia;
|
||
electron::util::AddImageSkiaRepFromJPEG(
|
||
&image_skia, reinterpret_cast<const unsigned char*>(buffer), length, 1.0);
|
||
return Create(isolate, gfx::Image(image_skia));
|
||
}
|
||
|
||
// static
|
||
gin::Handle<NativeImage> NativeImage::CreateFromPath(
|
||
v8::Isolate* isolate,
|
||
const base::FilePath& path) {
|
||
base::FilePath image_path = NormalizePath(path);
|
||
#if BUILDFLAG(IS_WIN)
|
||
if (image_path.MatchesExtension(FILE_PATH_LITERAL(".ico"))) {
|
||
return gin::CreateHandle(isolate, new NativeImage(isolate, image_path));
|
||
}
|
||
#endif
|
||
gfx::ImageSkia image_skia;
|
||
electron::util::PopulateImageSkiaRepsFromPath(&image_skia, image_path);
|
||
gfx::Image image(image_skia);
|
||
gin::Handle<NativeImage> handle = Create(isolate, image);
|
||
#if BUILDFLAG(IS_MAC)
|
||
if (IsTemplateFilename(image_path))
|
||
handle->SetTemplateImage(true);
|
||
#endif
|
||
return handle;
|
||
}
|
||
|
||
// static
|
||
gin::Handle<NativeImage> NativeImage::CreateFromBitmap(
|
||
gin_helper::ErrorThrower thrower,
|
||
v8::Local<v8::Value> buffer,
|
||
const gin_helper::Dictionary& options) {
|
||
if (!node::Buffer::HasInstance(buffer)) {
|
||
thrower.ThrowError("buffer must be a node Buffer");
|
||
return gin::Handle<NativeImage>();
|
||
}
|
||
|
||
unsigned int width = 0;
|
||
unsigned int height = 0;
|
||
double scale_factor = 1.;
|
||
|
||
if (!options.Get("width", &width)) {
|
||
thrower.ThrowError("width is required");
|
||
return gin::Handle<NativeImage>();
|
||
}
|
||
|
||
if (!options.Get("height", &height)) {
|
||
thrower.ThrowError("height is required");
|
||
return gin::Handle<NativeImage>();
|
||
}
|
||
|
||
auto info = SkImageInfo::MakeN32(width, height, kPremul_SkAlphaType);
|
||
auto size_bytes = info.computeMinByteSize();
|
||
|
||
if (size_bytes != node::Buffer::Length(buffer)) {
|
||
thrower.ThrowError("invalid buffer size");
|
||
return gin::Handle<NativeImage>();
|
||
}
|
||
|
||
options.Get("scaleFactor", &scale_factor);
|
||
|
||
if (width == 0 || height == 0) {
|
||
return CreateEmpty(thrower.isolate());
|
||
}
|
||
|
||
SkBitmap bitmap;
|
||
bitmap.allocN32Pixels(width, height, false);
|
||
bitmap.writePixels({info, node::Buffer::Data(buffer), bitmap.rowBytes()});
|
||
|
||
gfx::ImageSkia image_skia =
|
||
gfx::ImageSkia::CreateFromBitmap(bitmap, scale_factor);
|
||
|
||
return Create(thrower.isolate(), gfx::Image(image_skia));
|
||
}
|
||
|
||
// static
|
||
gin::Handle<NativeImage> NativeImage::CreateFromBuffer(
|
||
gin_helper::ErrorThrower thrower,
|
||
v8::Local<v8::Value> buffer,
|
||
gin::Arguments* args) {
|
||
if (!node::Buffer::HasInstance(buffer)) {
|
||
thrower.ThrowError("buffer must be a node Buffer");
|
||
return gin::Handle<NativeImage>();
|
||
}
|
||
|
||
int width = 0;
|
||
int height = 0;
|
||
double scale_factor = 1.;
|
||
|
||
gin_helper::Dictionary options;
|
||
if (args->GetNext(&options)) {
|
||
options.Get("width", &width);
|
||
options.Get("height", &height);
|
||
options.Get("scaleFactor", &scale_factor);
|
||
}
|
||
|
||
gfx::ImageSkia image_skia;
|
||
electron::util::AddImageSkiaRepFromBuffer(
|
||
&image_skia, reinterpret_cast<unsigned char*>(node::Buffer::Data(buffer)),
|
||
node::Buffer::Length(buffer), width, height, scale_factor);
|
||
return Create(args->isolate(), gfx::Image(image_skia));
|
||
}
|
||
|
||
// static
|
||
gin::Handle<NativeImage> NativeImage::CreateFromDataURL(v8::Isolate* isolate,
|
||
const GURL& url) {
|
||
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);
|
||
}
|
||
|
||
#if !BUILDFLAG(IS_MAC)
|
||
gin::Handle<NativeImage> NativeImage::CreateFromNamedImage(gin::Arguments* args,
|
||
std::string name) {
|
||
return CreateEmpty(args->isolate());
|
||
}
|
||
#endif
|
||
|
||
// static
|
||
gin::ObjectTemplateBuilder NativeImage::GetObjectTemplateBuilder(
|
||
v8::Isolate* isolate) {
|
||
gin::PerIsolateData* data = gin::PerIsolateData::From(isolate);
|
||
auto* wrapper_info = &kWrapperInfo;
|
||
v8::Local<v8::FunctionTemplate> constructor =
|
||
data->GetFunctionTemplate(wrapper_info);
|
||
if (constructor.IsEmpty()) {
|
||
constructor = v8::FunctionTemplate::New(isolate);
|
||
constructor->SetClassName(gin::StringToV8(isolate, GetTypeName()));
|
||
data->SetFunctionTemplate(wrapper_info, constructor);
|
||
}
|
||
return gin::ObjectTemplateBuilder(isolate, GetTypeName(),
|
||
constructor->InstanceTemplate())
|
||
.SetMethod("toPNG", &NativeImage::ToPNG)
|
||
.SetMethod("toJPEG", &NativeImage::ToJPEG)
|
||
.SetMethod("toBitmap", &NativeImage::ToBitmap)
|
||
.SetMethod("getBitmap", &NativeImage::GetBitmap)
|
||
.SetMethod("getScaleFactors", &NativeImage::GetScaleFactors)
|
||
.SetMethod("getNativeHandle", &NativeImage::GetNativeHandle)
|
||
.SetMethod("toDataURL", &NativeImage::ToDataURL)
|
||
.SetMethod("isEmpty", &NativeImage::IsEmpty)
|
||
.SetMethod("getSize", &NativeImage::GetSize)
|
||
.SetMethod("setTemplateImage", &NativeImage::SetTemplateImage)
|
||
.SetMethod("isTemplateImage", &NativeImage::IsTemplateImage)
|
||
.SetProperty("isMacTemplateImage", &NativeImage::IsTemplateImage,
|
||
&NativeImage::SetTemplateImage)
|
||
.SetMethod("resize", &NativeImage::Resize)
|
||
.SetMethod("crop", &NativeImage::Crop)
|
||
.SetMethod("getAspectRatio", &NativeImage::GetAspectRatio)
|
||
.SetMethod("addRepresentation", &NativeImage::AddRepresentation);
|
||
}
|
||
|
||
const char* NativeImage::GetTypeName() {
|
||
return "NativeImage";
|
||
}
|
||
|
||
// static
|
||
gin::WrapperInfo NativeImage::kWrapperInfo = {gin::kEmbedderNativeGin};
|
||
|
||
} // namespace api
|
||
|
||
} // namespace electron
|
||
|
||
namespace {
|
||
|
||
using electron::api::NativeImage;
|
||
|
||
void Initialize(v8::Local<v8::Object> exports,
|
||
v8::Local<v8::Value> unused,
|
||
v8::Local<v8::Context> context,
|
||
void* priv) {
|
||
v8::Isolate* isolate = context->GetIsolate();
|
||
gin_helper::Dictionary dict(isolate, exports);
|
||
gin_helper::Dictionary native_image = gin::Dictionary::CreateEmpty(isolate);
|
||
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);
|
||
#if !BUILDFLAG(IS_LINUX)
|
||
native_image.SetMethod("createThumbnailFromPath",
|
||
&NativeImage::CreateThumbnailFromPath);
|
||
#endif
|
||
}
|
||
|
||
} // namespace
|
||
|
||
NODE_LINKED_MODULE_CONTEXT_AWARE(electron_common_native_image, Initialize)
|