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.
|
|
|
|
|
|
|
|
#include "atom/common/api/atom_api_native_image.h"
|
|
|
|
|
2015-02-11 13:59:08 +00:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
2015-02-12 11:34:21 +00:00
|
|
|
#include "atom/common/asar/asar_util.h"
|
2015-02-11 10:41:06 +00:00
|
|
|
#include "atom/common/native_mate_converters/file_path_converter.h"
|
2015-02-11 08:49:43 +00:00
|
|
|
#include "atom/common/native_mate_converters/gfx_converter.h"
|
2015-02-12 06:27:53 +00:00
|
|
|
#include "atom/common/native_mate_converters/gurl_converter.h"
|
2015-02-12 06:32:51 +00:00
|
|
|
#include "base/base64.h"
|
2015-02-11 10:41:06 +00:00
|
|
|
#include "base/strings/string_util.h"
|
2015-02-11 08:03:19 +00:00
|
|
|
#include "native_mate/dictionary.h"
|
|
|
|
#include "native_mate/object_template_builder.h"
|
2015-02-12 06:27:53 +00:00
|
|
|
#include "net/base/data_url.h"
|
2015-02-11 10:41:06 +00:00
|
|
|
#include "ui/base/layout.h"
|
|
|
|
#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"
|
2015-02-11 09:47:54 +00:00
|
|
|
#include "ui/gfx/image/image_util.h"
|
2015-02-11 08:03:19 +00:00
|
|
|
|
|
|
|
#include "atom/common/node_includes.h"
|
|
|
|
|
|
|
|
namespace atom {
|
|
|
|
|
|
|
|
namespace api {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2015-02-11 10:41:06 +00:00
|
|
|
struct ScaleFactorPair {
|
|
|
|
const char* name;
|
|
|
|
float scale;
|
|
|
|
};
|
|
|
|
|
|
|
|
ScaleFactorPair kScaleFactorPairs[] = {
|
|
|
|
// The "@2x" is put as first one to make scale matching faster.
|
|
|
|
{ "@2x" , 2.0f },
|
|
|
|
{ "@3x" , 3.0f },
|
|
|
|
{ "@1x" , 1.0f },
|
|
|
|
{ "@4x" , 4.0f },
|
|
|
|
{ "@5x" , 5.0f },
|
|
|
|
{ "@1.25x" , 1.25f },
|
|
|
|
{ "@1.33x" , 1.33f },
|
|
|
|
{ "@1.4x" , 1.4f },
|
|
|
|
{ "@1.5x" , 1.5f },
|
|
|
|
{ "@1.8x" , 1.8f },
|
|
|
|
{ "@2.5x" , 2.5f },
|
|
|
|
};
|
|
|
|
|
|
|
|
float GetScaleFactorFromPath(const base::FilePath& path) {
|
|
|
|
std::string filename(path.BaseName().RemoveExtension().AsUTF8Unsafe());
|
|
|
|
|
|
|
|
// We don't try to convert string to float here because it is very very
|
|
|
|
// expensive.
|
|
|
|
for (unsigned i = 0; i < arraysize(kScaleFactorPairs); ++i) {
|
|
|
|
if (EndsWith(filename, kScaleFactorPairs[i].name, true))
|
|
|
|
return kScaleFactorPairs[i].scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AddImageSkiaRep(gfx::ImageSkia* image,
|
2015-02-12 07:19:05 +00:00
|
|
|
const unsigned char* data,
|
|
|
|
size_t size,
|
2015-02-11 10:41:06 +00:00
|
|
|
double scale_factor) {
|
|
|
|
scoped_ptr<SkBitmap> decoded(new SkBitmap());
|
|
|
|
|
|
|
|
// Try PNG first.
|
|
|
|
if (!gfx::PNGCodec::Decode(data, size, decoded.get()))
|
|
|
|
// Try JPEG.
|
|
|
|
decoded.reset(gfx::JPEGCodec::Decode(data, size));
|
|
|
|
|
|
|
|
if (!decoded)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
image->AddRepresentation(gfx::ImageSkiaRep(*decoded.release(), scale_factor));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-02-12 07:19:05 +00:00
|
|
|
bool AddImageSkiaRep(gfx::ImageSkia* image,
|
|
|
|
const base::FilePath& path,
|
|
|
|
double scale_factor) {
|
|
|
|
std::string file_contents;
|
2015-02-12 11:34:21 +00:00
|
|
|
if (!asar::ReadFileToString(path, &file_contents))
|
2015-02-12 07:19:05 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
const unsigned char* data =
|
|
|
|
reinterpret_cast<const unsigned char*>(file_contents.data());
|
|
|
|
size_t size = file_contents.size();
|
|
|
|
return AddImageSkiaRep(image, data, size, scale_factor);
|
|
|
|
}
|
|
|
|
|
2015-02-11 10:41:06 +00:00
|
|
|
bool PopulateImageSkiaRepsFromPath(gfx::ImageSkia* image,
|
|
|
|
const base::FilePath& path) {
|
|
|
|
bool succeed = false;
|
|
|
|
std::string filename(path.BaseName().RemoveExtension().AsUTF8Unsafe());
|
|
|
|
if (MatchPattern(filename, "*@*x"))
|
|
|
|
// Don't search for other representations if the DPI has been specified.
|
|
|
|
return AddImageSkiaRep(image, path, GetScaleFactorFromPath(path));
|
|
|
|
else
|
|
|
|
succeed |= AddImageSkiaRep(image, path, 1.0f);
|
|
|
|
|
|
|
|
for (const ScaleFactorPair& pair : kScaleFactorPairs)
|
|
|
|
succeed |= AddImageSkiaRep(image,
|
|
|
|
path.InsertBeforeExtensionASCII(pair.name),
|
|
|
|
pair.scale);
|
|
|
|
return succeed;
|
|
|
|
}
|
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-02-12 04:24:18 +00:00
|
|
|
return (MatchPattern(path.value(), "*Template.*") ||
|
|
|
|
MatchPattern(path.value(), "*Template@*x.*"));
|
|
|
|
}
|
|
|
|
#endif
|
2015-02-11 10:41:06 +00:00
|
|
|
|
2015-02-11 08:03:19 +00:00
|
|
|
v8::Persistent<v8::ObjectTemplate> template_;
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2015-02-11 11:22:41 +00:00
|
|
|
NativeImage::NativeImage() {}
|
|
|
|
|
2015-02-11 08:03:19 +00:00
|
|
|
NativeImage::NativeImage(const gfx::Image& image) : image_(image) {}
|
|
|
|
|
|
|
|
NativeImage::~NativeImage() {}
|
|
|
|
|
|
|
|
mate::ObjectTemplateBuilder NativeImage::GetObjectTemplateBuilder(
|
|
|
|
v8::Isolate* isolate) {
|
|
|
|
if (template_.IsEmpty())
|
|
|
|
template_.Reset(isolate, mate::ObjectTemplateBuilder(isolate)
|
|
|
|
.SetMethod("toPng", &NativeImage::ToPNG)
|
2015-02-11 09:47:54 +00:00
|
|
|
.SetMethod("toJpeg", &NativeImage::ToJPEG)
|
2015-02-12 06:32:51 +00:00
|
|
|
.SetMethod("toDataUrl", &NativeImage::ToDataURL)
|
2015-02-11 08:49:43 +00:00
|
|
|
.SetMethod("isEmpty", &NativeImage::IsEmpty)
|
|
|
|
.SetMethod("getSize", &NativeImage::GetSize)
|
2015-04-08 08:20:03 +00:00
|
|
|
.SetMethod("setTemplateImage", &NativeImage::SetTemplateImage)
|
2015-07-27 04:58:48 +00:00
|
|
|
.SetMethod("isTemplateImage", &NativeImage::IsTemplateImage)
|
2015-02-11 08:03:19 +00:00
|
|
|
.Build());
|
|
|
|
|
|
|
|
return mate::ObjectTemplateBuilder(
|
|
|
|
isolate, v8::Local<v8::ObjectTemplate>::New(isolate, template_));
|
|
|
|
}
|
|
|
|
|
2015-05-22 11:11:22 +00:00
|
|
|
v8::Local<v8::Value> NativeImage::ToPNG(v8::Isolate* isolate) {
|
2015-02-11 08:03:19 +00:00
|
|
|
scoped_refptr<base::RefCountedMemory> png = image_.As1xPNGBytes();
|
|
|
|
return node::Buffer::New(isolate,
|
|
|
|
reinterpret_cast<const char*>(png->front()),
|
|
|
|
png->size());
|
|
|
|
}
|
|
|
|
|
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);
|
|
|
|
return node::Buffer::New(isolate,
|
|
|
|
reinterpret_cast<const char*>(&output.front()),
|
|
|
|
output.size());
|
|
|
|
}
|
|
|
|
|
2015-02-12 06:32:51 +00:00
|
|
|
std::string NativeImage::ToDataURL() {
|
|
|
|
scoped_refptr<base::RefCountedMemory> png = image_.As1xPNGBytes();
|
|
|
|
std::string data_url;
|
|
|
|
data_url.insert(data_url.end(), png->front(), png->front() + png->size());
|
|
|
|
base::Base64Encode(data_url, &data_url);
|
|
|
|
data_url.insert(0, "data:image/png;base64,");
|
|
|
|
return data_url;
|
|
|
|
}
|
|
|
|
|
2015-02-11 08:49:43 +00:00
|
|
|
bool NativeImage::IsEmpty() {
|
|
|
|
return image_.IsEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
gfx::Size NativeImage::GetSize() {
|
|
|
|
return image_.Size();
|
|
|
|
}
|
|
|
|
|
2015-04-13 03:53:24 +00:00
|
|
|
#if !defined(OS_MACOSX)
|
|
|
|
void NativeImage::SetTemplateImage(bool setAsTemplate) {
|
|
|
|
}
|
2015-07-27 04:58:48 +00:00
|
|
|
bool NativeImage::IsTemplateImage() {
|
|
|
|
}
|
2015-04-13 03:53:24 +00:00
|
|
|
#endif
|
|
|
|
|
2015-02-12 05:55:45 +00:00
|
|
|
// static
|
|
|
|
mate::Handle<NativeImage> NativeImage::CreateEmpty(v8::Isolate* isolate) {
|
|
|
|
return mate::CreateHandle(isolate, new NativeImage);
|
|
|
|
}
|
|
|
|
|
2015-02-11 08:03:19 +00:00
|
|
|
// static
|
2015-02-11 10:02:59 +00:00
|
|
|
mate::Handle<NativeImage> NativeImage::Create(
|
|
|
|
v8::Isolate* isolate, const gfx::Image& image) {
|
2015-02-11 08:03:19 +00:00
|
|
|
return mate::CreateHandle(isolate, new NativeImage(image));
|
|
|
|
}
|
|
|
|
|
2015-02-11 10:02:59 +00:00
|
|
|
// static
|
|
|
|
mate::Handle<NativeImage> NativeImage::CreateFromPNG(
|
2015-02-12 06:27:53 +00:00
|
|
|
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
|
|
|
|
mate::Handle<NativeImage> NativeImage::CreateFromJPEG(
|
2015-02-12 06:27:53 +00:00
|
|
|
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
|
|
|
|
mate::Handle<NativeImage> NativeImage::CreateFromPath(
|
|
|
|
v8::Isolate* isolate, const base::FilePath& path) {
|
|
|
|
gfx::ImageSkia image_skia;
|
|
|
|
PopulateImageSkiaRepsFromPath(&image_skia, path);
|
2015-02-12 04:24:18 +00:00
|
|
|
gfx::Image image(image_skia);
|
2015-04-13 03:53:24 +00:00
|
|
|
mate::Handle<NativeImage> handle = Create(isolate, image);
|
2015-02-12 04:24:18 +00:00
|
|
|
#if defined(OS_MACOSX)
|
2015-07-27 04:58:48 +00:00
|
|
|
if (IsTemplateFilename(path))
|
2015-04-13 03:53:24 +00:00
|
|
|
handle->SetTemplateImage(true);
|
2015-02-12 04:24:18 +00:00
|
|
|
#endif
|
2015-04-13 03:53:24 +00:00
|
|
|
return handle;
|
2015-02-11 10:41:06 +00:00
|
|
|
}
|
|
|
|
|
2015-02-12 07:19:05 +00:00
|
|
|
// static
|
|
|
|
mate::Handle<NativeImage> NativeImage::CreateFromBuffer(
|
2015-05-22 11:11:22 +00:00
|
|
|
mate::Arguments* args, v8::Local<v8::Value> buffer) {
|
2015-02-12 07:19:05 +00:00
|
|
|
double scale_factor = 1.;
|
|
|
|
args->GetNext(&scale_factor);
|
|
|
|
|
|
|
|
gfx::ImageSkia image_skia;
|
|
|
|
AddImageSkiaRep(&image_skia,
|
|
|
|
reinterpret_cast<unsigned char*>(node::Buffer::Data(buffer)),
|
|
|
|
node::Buffer::Length(buffer),
|
|
|
|
scale_factor);
|
|
|
|
return Create(args->isolate(), gfx::Image(image_skia));
|
|
|
|
}
|
|
|
|
|
2015-02-12 06:27:53 +00:00
|
|
|
// static
|
|
|
|
mate::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);
|
|
|
|
}
|
|
|
|
|
2015-02-11 08:03:19 +00:00
|
|
|
} // namespace api
|
|
|
|
|
|
|
|
} // namespace atom
|
|
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2015-05-22 11:11:22 +00:00
|
|
|
void Initialize(v8::Local<v8::Object> exports, v8::Local<v8::Value> unused,
|
|
|
|
v8::Local<v8::Context> context, void* priv) {
|
2015-02-11 10:09:51 +00:00
|
|
|
mate::Dictionary dict(context->GetIsolate(), exports);
|
2015-02-12 05:55:45 +00:00
|
|
|
dict.SetMethod("createEmpty", &atom::api::NativeImage::CreateEmpty);
|
2015-02-11 10:41:06 +00:00
|
|
|
dict.SetMethod("createFromPath", &atom::api::NativeImage::CreateFromPath);
|
2015-02-12 07:19:05 +00:00
|
|
|
dict.SetMethod("createFromBuffer", &atom::api::NativeImage::CreateFromBuffer);
|
2015-02-12 06:27:53 +00:00
|
|
|
dict.SetMethod("createFromDataUrl",
|
|
|
|
&atom::api::NativeImage::CreateFromDataURL);
|
2015-02-11 08:03:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
NODE_MODULE_CONTEXT_AWARE_BUILTIN(atom_common_native_image, Initialize)
|