blob: 508f13604c97f39db40d98a6e4d6ccae74107423 [file] [log] [blame]
// Copyright 2020 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/apps/link_capturing/mac_intent_picker_helpers.h"
#import <Cocoa/Cocoa.h>
#import <SafariServices/SafariServices.h>
#include "base/feature_list.h"
#include "base/functional/callback_helpers.h"
#include "base/mac/launch_application.h"
#include "base/no_destructor.h"
#include "base/strings/sys_string_conversions.h"
#include "net/base/apple/url_conversions.h"
#include "ui/base/models/image_model.h"
#include "ui/gfx/image/image_skia_util_mac.h"
namespace apps {
namespace {
bool& UseFakeAppForTesting() {
static bool value = false;
return value;
}
std::string& FakeAppForTesting() {
static base::NoDestructor<std::string> value;
return *value;
}
NSImage* CreateRedIconForTesting() {
return [NSImage imageWithSize:NSMakeSize(16, 16)
flipped:NO
drawingHandler:^(NSRect rect) {
[NSColor.redColor set];
NSRectFill(rect);
return YES;
}];
}
MacAppInfo AppInfoForAppUrl(NSURL* app_url, base::span<int> icon_sizes) {
CHECK(!icon_sizes.empty());
NSString* app_name = nil;
if (![app_url getResourceValue:&app_name
forKey:NSURLLocalizedNameKey
error:nil]) {
// This shouldn't happen but just in case.
app_name = app_url.lastPathComponent;
}
NSImage* app_icon = nil;
if (![app_url getResourceValue:&app_icon
forKey:NSURLEffectiveIconKey
error:nil]) {
// This shouldn't happen, but just in case. (Note that, despite its name,
// NSImageNameApplicationIcon is the icon of "this app". There is no
// constant for "generic app icon", only this string value. This value has
// been verified to exist from macOS 10.15 through macOS 14; see -[NSImage
// _systemImageNamed:].)
app_icon = [NSImage imageNamed:@"NSDefaultApplicationIcon"];
}
if (UseFakeAppForTesting()) { // IN-TEST
app_icon = CreateRedIconForTesting(); // IN-TEST
}
gfx::ImageFamily image_family;
if (app_icon) {
for (int icon_size : icon_sizes) {
CHECK_GT(icon_size, 0);
auto image = gfx::ImageSkiaFromResizedNSImage(
app_icon, NSMakeSize(icon_size, icon_size));
image.SetReadOnly();
image_family.Add(std::move(image));
}
}
return MacAppInfo{{PickerEntryType::kMacOs, ui::ImageModel(),
base::SysNSStringToUTF8(app_url.path),
base::SysNSStringToUTF8(app_name)},
std::move(image_family)};
}
} // namespace
std::optional<MacAppInfo> FindMacAppForUrl(const GURL& url,
base::span<int> icon_sizes) {
if (UseFakeAppForTesting()) {
std::string fake_app = FakeAppForTesting(); // IN-TEST
if (fake_app.empty()) {
return std::nullopt;
}
return AppInfoForAppUrl(
[NSURL fileURLWithPath:base::SysUTF8ToNSString(fake_app)], icon_sizes);
}
NSURL* nsurl = net::NSURLWithGURL(url);
if (!nsurl) {
return std::nullopt;
}
SFUniversalLink* link = [[SFUniversalLink alloc] initWithWebpageURL:nsurl];
if (link) {
return AppInfoForAppUrl(link.applicationURL, icon_sizes);
}
return std::nullopt;
}
void LaunchMacApp(const GURL& url,
const std::string& launch_name,
base::OnceClosure callback) {
base::mac::LaunchApplication(
base::FilePath(launch_name),
/*command_line_args=*/{}, {url.spec()},
/*options=*/{},
base::IgnoreArgs<NSRunningApplication*, NSError*>(std::move(callback)));
}
void OverrideMacAppForUrlForTesting(bool fake, const std::string& app_path) {
UseFakeAppForTesting() = fake; // IN-TEST
FakeAppForTesting() = app_path; // IN-TEST
}
} // namespace apps