// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "stdafx.h" #include "winrt_utils.h" #include #include "base/files/file_path.h" #include "base/logging.h" #include "base/win/scoped_com_initializer.h" #include "base/win/scoped_comptr.h" #include "chrome/installer/util/browser_distribution.h" #include "chrome/installer/util/install_util.h" void CheckHR(HRESULT hr, const char* message) { if (FAILED(hr)) { if (message) PLOG(DFATAL) << message << ", hr = " << std::hex << hr; else PLOG(DFATAL) << "COM ERROR" << ", hr = " << std::hex << hr; } } HSTRING MakeHString(const base::string16& str) { HSTRING hstr; if (FAILED(::WindowsCreateString(str.c_str(), static_cast(str.size()), &hstr))) { PLOG(DFATAL) << "Hstring creation failed"; } return hstr; } base::string16 MakeStdWString(HSTRING hstring) { const wchar_t* str; UINT32 size = 0; str = ::WindowsGetStringRawBuffer(hstring, &size); if (!size) return base::string16(); return base::string16(str, size); } namespace { #define IMPLEMENT_CREATE_PROPERTY(Name, Type) \ HRESULT Create ## Name ## Property(Type value, \ winfoundtn::IPropertyValue** prop) { \ mswr::ComPtr property_value_statics; \ HRESULT hr = winrt_utils::CreateActivationFactory( \ RuntimeClass_Windows_Foundation_PropertyValue, \ property_value_statics.GetAddressOf()); \ CheckHR(hr, "Can't create IPropertyValueStatics"); \ hr = property_value_statics->Create ## Name ( \ value, \ reinterpret_cast(prop)); \ CheckHR(hr, "Failed to create Property"); \ return hr; \ } #define COMPARE_ATOMIC_PROPERTY_VALUES(Name, Type) \ Type lhs_value; \ hr = lhs->Get ## Name (&lhs_value); \ CheckHR(hr, "Can't get value for lhs"); \ Type rhs_value; \ hr = rhs->Get ## Name (&rhs_value); \ CheckHR(hr, "Can't get value for rhs"); \ if (lhs_value < rhs_value) \ *result = -1; \ else if (lhs_value > rhs_value) \ *result = 1; \ else \ *result = 0; \ hr = S_OK } // namespace namespace winrt_utils { IMPLEMENT_CREATE_PROPERTY(String, HSTRING); IMPLEMENT_CREATE_PROPERTY(Int16, INT16); IMPLEMENT_CREATE_PROPERTY(Int32, INT32); IMPLEMENT_CREATE_PROPERTY(Int64, INT64); IMPLEMENT_CREATE_PROPERTY(UInt8, UINT8); IMPLEMENT_CREATE_PROPERTY(UInt16, UINT16); IMPLEMENT_CREATE_PROPERTY(UInt32, UINT32); IMPLEMENT_CREATE_PROPERTY(UInt64, UINT64); HRESULT CompareProperties(winfoundtn::IPropertyValue* lhs, winfoundtn::IPropertyValue* rhs, INT32* result) { if (result == nullptr) { PLOG(DFATAL) << "Invalid argument to CompareProperties."; return E_INVALIDARG; } if (lhs == rhs) { *result = 0; return S_OK; } winfoundtn::PropertyType lhs_property_type; HRESULT hr = lhs->get_Type(&lhs_property_type); if (FAILED(hr)) { PLOG(DFATAL) << "Can't get property type for lhs, hr=" << std::hex << hr; } winfoundtn::PropertyType rhs_property_type; hr = rhs->get_Type(&rhs_property_type); CheckHR(hr, "Can't get property type for rhs"); if (lhs_property_type != rhs_property_type) return E_INVALIDARG; switch (lhs_property_type) { case winfoundtn::PropertyType::PropertyType_String: { mswrw::HString lhs_string; hr = lhs->GetString(lhs_string.GetAddressOf()); CheckHR(hr, "Can't get string for lhs"); mswrw::HString rhs_string; hr = rhs->GetString(rhs_string.GetAddressOf()); CheckHR(hr, "Can't get string for rhs"); hr = WindowsCompareStringOrdinal( lhs_string.Get(), rhs_string.Get(), result); break; } case winfoundtn::PropertyType::PropertyType_Char16: { COMPARE_ATOMIC_PROPERTY_VALUES(Char16, wchar_t); break; } case winfoundtn::PropertyType::PropertyType_Double: { COMPARE_ATOMIC_PROPERTY_VALUES(Double, double); break; } case winfoundtn::PropertyType::PropertyType_Int16: { COMPARE_ATOMIC_PROPERTY_VALUES(Int16, INT16); break; } case winfoundtn::PropertyType::PropertyType_Int32: { COMPARE_ATOMIC_PROPERTY_VALUES(Int32, INT32); break; } case winfoundtn::PropertyType::PropertyType_Int64: { COMPARE_ATOMIC_PROPERTY_VALUES(Int64, INT64); break; } case winfoundtn::PropertyType::PropertyType_UInt8: { COMPARE_ATOMIC_PROPERTY_VALUES(UInt8, UINT8); break; } case winfoundtn::PropertyType::PropertyType_UInt16: { COMPARE_ATOMIC_PROPERTY_VALUES(UInt16, UINT16); break; } case winfoundtn::PropertyType::PropertyType_UInt32: { COMPARE_ATOMIC_PROPERTY_VALUES(UInt32, UINT32); break; } case winfoundtn::PropertyType::PropertyType_UInt64: { COMPARE_ATOMIC_PROPERTY_VALUES(UInt64, UINT64); break; } default: { hr = E_NOTIMPL; } } return hr; } bool GetArgumentsFromShortcut(const base::FilePath& shortcut, base::string16* arguments) { HRESULT result; base::win::ScopedComPtr i_shell_link; bool is_resolved = false; base::win::ScopedCOMInitializer sta_com_initializer; // Get pointer to the IShellLink interface result = i_shell_link.CreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER); if (SUCCEEDED(result)) { base::win::ScopedComPtr persist; // Query IShellLink for the IPersistFile interface result = persist.QueryFrom(i_shell_link); if (SUCCEEDED(result)) { WCHAR temp_arguments[MAX_PATH]; // Load the shell link result = persist->Load(shortcut.value().c_str(), STGM_READ); if (SUCCEEDED(result)) { result = i_shell_link->GetArguments(temp_arguments, MAX_PATH); *arguments = temp_arguments; is_resolved = true; } } } return is_resolved; } base::string16 ReadArgumentsFromPinnedTaskbarShortcut() { wchar_t path_buffer[MAX_PATH] = {}; if (SUCCEEDED(SHGetFolderPath(NULL, CSIDL_APPDATA, NULL, SHGFP_TYPE_CURRENT, path_buffer))) { base::FilePath shortcut(path_buffer); shortcut = shortcut.Append( L"Microsoft\\Internet Explorer\\Quick Launch\\User Pinned\\TaskBar"); BrowserDistribution* dist = BrowserDistribution::GetDistribution(); base::string16 link_name = dist->GetShortcutName( BrowserDistribution::SHORTCUT_CHROME) + installer::kLnkExt; shortcut = shortcut.Append(link_name); base::string16 arguments; if (GetArgumentsFromShortcut(shortcut, &arguments)) { return arguments; } } return L""; } } // namespace winrt_utils