// 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 #include #include #import #import #import #include "base/apple/foundation_util.h" #include "base/strings/sys_string_conversions.h" #include "base/task/bind_post_task.h" #include "gin/arguments.h" #include "gin/handle.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 { namespace { base::span as_byte_span(NSData* data) { // SAFETY: There is no NSData API that passes the UNSAFE_BUFFER_USAGE // test, so let's isolate the unsafe API use into this function. Instead of // calling '[data bytes]' and '[data length]' directly, the rest of our // code should prefer to use spans returned by this function. return UNSAFE_BUFFERS(base::span{ reinterpret_cast([data bytes]), [data length]}); } } // namespace 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:NSBitmapImageFileTypePNG properties:[[NSDictionary alloc] init]]; } double safeShift(double in, double def) { if (in >= 0 || in <= 1 || in == def) return in; return def; } void ReceivedThumbnailResult(CGSize size, gin_helper::Promise p, QLThumbnailRepresentation* thumbnail, NSError* error) { if (error || !thumbnail) { std::string err_msg([error.localizedDescription UTF8String]); p.RejectWithErrorMessage("unable to retrieve thumbnail preview " "image for the given path: " + err_msg); } else { NSImage* result = [[NSImage alloc] initWithCGImage:[thumbnail CGImage] size:size]; gfx::Image image(result); p.Resolve(image); } } // static v8::Local NativeImage::CreateThumbnailFromPath( v8::Isolate* isolate, const base::FilePath& path, const gfx::Size& size) { gin_helper::Promise promise(isolate); v8::Local handle = promise.GetHandle(); if (size.IsEmpty()) { promise.RejectWithErrorMessage("size must not be empty"); return handle; } CGSize cg_size = size.ToCGSize(); NSURL* nsurl = base::apple::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]; QLThumbnailGenerationRequest* request([[QLThumbnailGenerationRequest alloc] initWithFileAtURL:nsurl size:cg_size scale:[screen backingScaleFactor] representationTypes:QLThumbnailGenerationRequestRepresentationTypeAll]); __block auto block_callback = base::BindPostTaskToCurrentDefault( base::BindOnce(&ReceivedThumbnailResult, cg_size, std::move(promise))); auto completionHandler = ^(QLThumbnailRepresentation* thumbnail, NSError* error) { std::move(block_callback).Run(thumbnail, error); }; [[QLThumbnailGenerator sharedGenerator] generateBestRepresentationForRequest:request completionHandler:completionHandler]; return handle; } gin::Handle NativeImage::CreateFromNamedImage(gin::Arguments* args, std::string name) { @autoreleasepool { std::vector 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) { auto gfx_image = gfx::Image::CreateFrom1xPNGBytes(as_byte_span(png_data)); 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(), as_byte_span(png_data)); } } void NativeImage::SetTemplateImage(bool setAsTemplate) { [image_.AsNSImage() setTemplate:setAsTemplate]; } bool NativeImage::IsTemplateImage() { return [image_.AsNSImage() isTemplate]; } } // namespace electron::api