#include "base/win/default_apps_util.h"
#include <shobjidl.h>
#include <shellapi.h>
#include <wrl/client.h>
#include <optional>
#include <string_view>
#include "base/metrics/histogram_functions.h"
#include "base/strings/cstring_view.h"
#include "base/strings/escape.h"
#include "base/strings/strcat.h"
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "base/win/com_init_util.h"
#include "base/win/registry.h"
#include "base/win/windows_version.h"
namespace {
class __declspec(uuid("6A283FE2-ECFA-4599-91C4-E80957137B26")) IOpenWithLauncher
: public IUnknown {
public:
virtual HRESULT STDMETHODCALLTYPE Launch(HWND hWndParent,
const wchar_t* lpszPath,
int flags) = 0;
};
std::optional<CLSID> GetOpenWithLauncherCLSID() {
std::wstring value;
base::win::RegKey(HKEY_LOCAL_MACHINE,
L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\OpenWith",
KEY_QUERY_VALUE)
.ReadValue(L"OpenWithLauncher", &value);
if (value.empty()) {
return std::nullopt;
}
CLSID clsid;
const auto hr = ::CLSIDFromString(value.c_str(), &clsid);
return SUCCEEDED(hr) ? std::make_optional(clsid) : std::nullopt;
}
enum class OpenWithLauncherResult {
kSuccess = 0,
kSuccessNoChange = 1,
kClsidNotFound = 2,
kComError = 3,
kLaunchError = 4,
kMaxValue = kLaunchError
};
void RecordOpenWithLauncherResult(OpenWithLauncherResult result) {
base::UmaHistogramEnumeration("Windows.OpenWithLauncherResult", result);
}
std::wstring GetTargetForDefaultAppsSettings(std::wstring_view protocol) {
static constexpr std::wstring_view kSystemSettingsDefaultAppsPrefix(
L"SystemSettings_DefaultApps_");
if (base::EqualsCaseInsensitiveASCII(protocol, L"http")) {
return base::StrCat({kSystemSettingsDefaultAppsPrefix, L"Browser"});
}
if (base::EqualsCaseInsensitiveASCII(protocol, L"mailto")) {
return base::StrCat({kSystemSettingsDefaultAppsPrefix, L"Email"});
}
return L"SettingsPageAppsDefaultsProtocolView";
}
}
namespace base::win {
bool LaunchDefaultAppsSettingsModernDialog(std::wstring_view protocol) {
static constexpr wchar_t kControlPanelAppModelId[] =
L"windows.immersivecontrolpanel_cw5n1h2txyewy"
L"!microsoft.windows.immersivecontrolpanel";
Microsoft::WRL::ComPtr<IApplicationActivationManager> activator;
HRESULT hr =
::CoCreateInstance(CLSID_ApplicationActivationManager, nullptr,
CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&activator));
if (FAILED(hr)) {
return false;
}
DWORD pid = 0;
hr = activator->ActivateApplication(
kControlPanelAppModelId, L"page=SettingsPageAppsDefaults", AO_NONE, &pid);
if (FAILED(hr)) {
return false;
}
if (protocol.empty() || GetVersion() >= Version::WIN11) {
return true;
}
hr = activator->ActivateApplication(
kControlPanelAppModelId,
base::StrCat({L"page=SettingsPageAppsDefaults&target=",
GetTargetForDefaultAppsSettings(protocol)})
.c_str(),
AO_NONE, &pid);
return SUCCEEDED(hr);
}
bool LaunchDefaultAppForFileExtensionSettings(
base::wcstring_view file_extension,
HWND parent_hwnd) {
AssertComInitialized();
const auto open_with_launcher_clsid = GetOpenWithLauncherCLSID();
if (!open_with_launcher_clsid) {
RecordOpenWithLauncherResult(OpenWithLauncherResult::kClsidNotFound);
return false;
}
Microsoft::WRL::ComPtr<IOpenWithLauncher> open_with_launcher;
if (FAILED(::CoCreateInstance(*open_with_launcher_clsid, nullptr,
CLSCTX_LOCAL_SERVER,
IID_PPV_ARGS(&open_with_launcher)))) {
RecordOpenWithLauncherResult(OpenWithLauncherResult::kComError);
return false;
}
static constexpr int kOpenWithFlags = 0x2004;
const HRESULT hr = open_with_launcher->Launch(
parent_hwnd, file_extension.data(), kOpenWithFlags);
if (SUCCEEDED(hr)) {
RecordOpenWithLauncherResult(OpenWithLauncherResult::kSuccess);
return true;
}
if (hr == HRESULT_FROM_WIN32(ERROR_CANCELLED)) {
RecordOpenWithLauncherResult(OpenWithLauncherResult::kSuccessNoChange);
return true;
}
RecordOpenWithLauncherResult(OpenWithLauncherResult::kLaunchError);
return false;
}
bool LaunchSettingsDefaultApps(std::wstring_view app_name,
bool is_per_user_install) {
AssertComInitialized();
const std::wstring settings_url = StrCat(
{L"ms-settings:defaultapps?",
is_per_user_install ? L"registeredAppUser=" : L"registeredAppMachine=",
ASCIIToWide(EscapeQueryParamValue(WideToUTF8(app_name),
false))});
return reinterpret_cast<intptr_t>(::ShellExecute(
nullptr, L"open", settings_url.c_str(),
nullptr,
nullptr, SW_SHOWNORMAL)) > 32;
}
bool LaunchSettingsUri(base::wcstring_view uri) {
AssertComInitialized();
Microsoft::WRL::ComPtr<IApplicationActivationManager> activator;
HRESULT hr = ::CoCreateInstance(CLSID_ApplicationActivationManager, nullptr,
CLSCTX_ALL, IID_PPV_ARGS(&activator));
if (FAILED(hr)) {
return false;
}
::CoAllowSetForegroundWindow(activator.Get(), nullptr);
static constexpr wchar_t kControlPanelAppModelId[] =
L"windows.immersivecontrolpanel_cw5n1h2txyewy"
L"!microsoft.windows.immersivecontrolpanel";
DWORD pid = 0;
hr = activator->ActivateApplication(kControlPanelAppModelId, uri.c_str(),
AO_NONE, &pid);
return SUCCEEDED(hr);
}
}