184 lines
		
	
	
	
		
			6.9 KiB
			
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
	
		
			6.9 KiB
			
		
	
	
	
		
			Text
		
	
	
	
	
	
// 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>
 | 
						|
#import <QuickLookThumbnailing/QuickLookThumbnailing.h>
 | 
						|
 | 
						|
#include "base/mac/foundation_util.h"
 | 
						|
#include "base/mac/scoped_nsobject.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/geometry/size.h"
 | 
						|
#include "ui/gfx/image/image_skia.h"
 | 
						|
#include "ui/gfx/image/image_skia_operations.h"
 | 
						|
 | 
						|
namespace electron::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();
 | 
						|
 | 
						|
  if (@available(macOS 10.15, *)) {
 | 
						|
    NSURL* nsurl = base::mac::FilePathToNSURL(path);
 | 
						|
 | 
						|
    // We need to explicitly check if the user has passed an invalid path
 | 
						|
    // because QLThumbnailGenerationRequest will generate a stock file icon
 | 
						|
    // and pass silently if we do not.
 | 
						|
    if (![[NSFileManager defaultManager] fileExistsAtPath:[nsurl path]]) {
 | 
						|
      promise.RejectWithErrorMessage(
 | 
						|
          "unable to retrieve thumbnail preview image for the given path");
 | 
						|
      return handle;
 | 
						|
    }
 | 
						|
 | 
						|
    NSScreen* screen = [[NSScreen screens] firstObject];
 | 
						|
    base::scoped_nsobject<QLThumbnailGenerationRequest> request(
 | 
						|
        [[QLThumbnailGenerationRequest alloc]
 | 
						|
              initWithFileAtURL:nsurl
 | 
						|
                           size:cg_size
 | 
						|
                          scale:[screen backingScaleFactor]
 | 
						|
            representationTypes:
 | 
						|
                QLThumbnailGenerationRequestRepresentationTypeAll]);
 | 
						|
    __block gin_helper::Promise<gfx::Image> p = std::move(promise);
 | 
						|
    [[QLThumbnailGenerator sharedGenerator]
 | 
						|
        generateBestRepresentationForRequest:request
 | 
						|
                           completionHandler:^(
 | 
						|
                               QLThumbnailRepresentation* thumbnail,
 | 
						|
                               NSError* error) {
 | 
						|
                             if (error || !thumbnail) {
 | 
						|
                               std::string err_msg(
 | 
						|
                                   [error.localizedDescription UTF8String]);
 | 
						|
                               dispatch_async(dispatch_get_main_queue(), ^{
 | 
						|
                                 p.RejectWithErrorMessage(
 | 
						|
                                     "unable to retrieve thumbnail preview "
 | 
						|
                                     "image for the given path: " +
 | 
						|
                                     err_msg);
 | 
						|
                               });
 | 
						|
                             } else {
 | 
						|
                               NSImage* result = [[[NSImage alloc]
 | 
						|
                                   initWithCGImage:[thumbnail CGImage]
 | 
						|
                                              size:cg_size] autorelease];
 | 
						|
                               gfx::Image image(result);
 | 
						|
                               dispatch_async(dispatch_get_main_queue(), ^{
 | 
						|
                                 p.Resolve(image);
 | 
						|
                               });
 | 
						|
                             }
 | 
						|
                           }];
 | 
						|
  } else {
 | 
						|
    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);
 | 
						|
    // Do not block the main thread 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 electron::api
 |