// 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 <string> #include <utility> #include <vector> #import <Cocoa/Cocoa.h> #import <QuickLook/QuickLook.h> #include "base/mac/foundation_util.h" #include "base/strings/sys_string_conversions.h" #include "gin/arguments.h" #include "shell/common/gin_converters/image_converter.h" #include "shell/common/gin_helper/promise.h" #include "ui/gfx/color_utils.h" #include "ui/gfx/image/image.h" #include "ui/gfx/image/image_skia.h" #include "ui/gfx/image/image_skia_operations.h" namespace electron { namespace api { NSData* bufferFromNSImage(NSImage* image) { CGImageRef ref = [image CGImageForProposedRect:nil context:nil hints:nil]; NSBitmapImageRep* rep = [[NSBitmapImageRep alloc] initWithCGImage:ref]; [rep setSize:[image size]]; return [rep representationUsingType:NSPNGFileType properties:[[NSDictionary alloc] init]]; } double safeShift(double in, double def) { if (in >= 0 || in <= 1 || in == def) return in; return def; } // static v8::Local<v8::Promise> NativeImage::CreateThumbnailFromPath( v8::Isolate* isolate, const base::FilePath& path, const gfx::Size& size) { gin_helper::Promise<gfx::Image> promise(isolate); v8::Local<v8::Promise> handle = promise.GetHandle(); if (size.IsEmpty()) { promise.RejectWithErrorMessage("size must not be empty"); return handle; } CGSize cg_size = size.ToCGSize(); base::ScopedCFTypeRef<CFURLRef> cfurl = base::mac::FilePathToCFURL(path); base::ScopedCFTypeRef<QLThumbnailRef> ql_thumbnail( QLThumbnailCreate(kCFAllocatorDefault, cfurl, cg_size, NULL)); __block gin_helper::Promise<gfx::Image> p = std::move(promise); // we do not want to blocking the main thread while waiting for quicklook to // generate the thumbnail QLThumbnailDispatchAsync( ql_thumbnail, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, /*flags*/ 0), ^{ base::ScopedCFTypeRef<CGImageRef> cg_thumbnail( QLThumbnailCopyImage(ql_thumbnail)); if (cg_thumbnail) { NSImage* result = [[[NSImage alloc] initWithCGImage:cg_thumbnail size:cg_size] autorelease]; gfx::Image thumbnail(result); dispatch_async(dispatch_get_main_queue(), ^{ p.Resolve(thumbnail); }); } else { dispatch_async(dispatch_get_main_queue(), ^{ p.RejectWithErrorMessage("unable to retrieve thumbnail preview " "image for the given path"); }); } }); return handle; } gin::Handle<NativeImage> NativeImage::CreateFromNamedImage(gin::Arguments* args, std::string name) { @autoreleasepool { std::vector<double> hsl_shift; // The string representations of NSImageNames don't match the strings // themselves; they instead follow the following pattern: // * NSImageNameActionTemplate -> "NSActionTemplate" // * NSImageNameMultipleDocuments -> "NSMultipleDocuments" // To account for this, we strip out "ImageName" from the passed string. std::string to_remove("ImageName"); size_t pos = name.find(to_remove); if (pos != std::string::npos) { name.erase(pos, to_remove.length()); } NSImage* image = [NSImage imageNamed:base::SysUTF8ToNSString(name)]; if (!image.valid) { return CreateEmpty(args->isolate()); } NSData* png_data = bufferFromNSImage(image); if (args->GetNext(&hsl_shift) && hsl_shift.size() == 3) { gfx::Image gfx_image = gfx::Image::CreateFrom1xPNGBytes( reinterpret_cast<const unsigned char*>((char*)[png_data bytes]), [png_data length]); color_utils::HSL shift = {safeShift(hsl_shift[0], -1), safeShift(hsl_shift[1], 0.5), safeShift(hsl_shift[2], 0.5)}; png_data = bufferFromNSImage( gfx::Image(gfx::ImageSkiaOperations::CreateHSLShiftedImage( gfx_image.AsImageSkia(), shift)) .AsNSImage()); } return CreateFromPNG(args->isolate(), (char*)[png_data bytes], [png_data length]); } } void NativeImage::SetTemplateImage(bool setAsTemplate) { [image_.AsNSImage() setTemplate:setAsTemplate]; } bool NativeImage::IsTemplateImage() { return [image_.AsNSImage() isTemplate]; } } // namespace api } // namespace electron