15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved. 25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be 35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file. 45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Implementation of the CommandExecuteImpl class which implements the 55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// IExecuteCommand and related interfaces for handling ShellExecute based 65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// launches of the Chrome browser. 75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "win8/delegate_execute/command_execute_impl.h" 95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <shlguid.h> 115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/file_util.h" 135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/path_service.h" 14bbcdd45c55eb7c4641ab97aef9889b0fc828e7d3Ben Murdoch#include "base/process/launch.h" 15bbcdd45c55eb7c4641ab97aef9889b0fc828e7d3Ben Murdoch#include "base/process/process_handle.h" 16868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "base/strings/utf_string_conversions.h" 177dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch#include "base/win/message_window.h" 185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/win/registry.h" 195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/win/scoped_co_mem.h" 205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/win/scoped_handle.h" 215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/win/scoped_process_information.h" 225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/win/win_util.h" 235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/chrome_constants.h" 245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/chrome_paths.h" 255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/chrome_switches.h" 26d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles)#include "chrome/installer/util/browser_distribution.h" 27d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles)#include "chrome/installer/util/install_util.h" 28d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles)#include "chrome/installer/util/shell_util.h" 295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/installer/util/util_constants.h" 305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ui/base/clipboard/clipboard_util_win.h" 315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "win8/delegate_execute/chrome_util.h" 325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "win8/delegate_execute/delegate_execute_util.h" 33a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)#include "win8/viewer/metro_viewer_constants.h" 345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace { 365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Helper function to retrieve the url from IShellItem interface passed in. 385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Returns S_OK on success. 395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)HRESULT GetUrlFromShellItem(IShellItem* shell_item, string16* url) { 405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(shell_item); 415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(url); 425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // First attempt to get the url from the underlying IDataObject if any. This 435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // ensures that we get the full url, i.e. including the anchor. 445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If we fail to get the underlying IDataObject we retrieve the url via the 455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // IShellItem::GetDisplayName function. 465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CComPtr<IDataObject> object; 475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) HRESULT hr = shell_item->BindToHandler(NULL, 485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) BHID_DataObject, 495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IID_IDataObject, 505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) reinterpret_cast<void**>(&object)); 515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (SUCCEEDED(hr)) { 525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(object); 535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (ui::ClipboardUtil::GetPlainText(object, url)) 545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::win::ScopedCoMem<wchar_t> name; 585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) hr = shell_item->GetDisplayName(SIGDN_URL, &name); 595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (hr != S_OK) { 605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to get display name\n"); 615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return hr; 625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *url = static_cast<const wchar_t*>(name); 655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Retrieved url from display name %ls\n", url->c_str()); 665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 69eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch#if defined(USE_AURA) 702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool LaunchChromeBrowserProcess() { 712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::FilePath delegate_exe_path; 722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!PathService::Get(base::FILE_EXE, &delegate_exe_path)) 732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return false; 742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // First try and go up a level to find chrome.exe. 762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::FilePath chrome_exe_path = 772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) delegate_exe_path.DirName() 782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) .DirName() 792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) .Append(chrome::kBrowserProcessExecutableName); 807dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch if (!base::PathExists(chrome_exe_path)) { 812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Try looking in the current directory if we couldn't find it one up in 822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // order to support developer installs. 832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) chrome_exe_path = 842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) delegate_exe_path.DirName() 852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) .Append(chrome::kBrowserProcessExecutableName); 862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 887dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch if (!base::PathExists(chrome_exe_path)) { 892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AtlTrace("Could not locate chrome.exe at: %ls\n", 902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) chrome_exe_path.value().c_str()); 912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return false; 922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) CommandLine cl(chrome_exe_path); 952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Prevent a Chrome window from showing up on the desktop. 972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) cl.AppendSwitch(switches::kSilentLaunch); 982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 99eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch // Tell Chrome to connect to the Metro viewer process. 100eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch cl.AppendSwitch(switches::kViewerConnect); 1012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::LaunchOptions launch_options; 1032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) launch_options.start_hidden = true; 1042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return base::LaunchProcess(cl, launch_options, NULL); 1062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 107eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch#endif // defined(USE_AURA) 1082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} // namespace 1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool CommandExecuteImpl::path_provider_initialized_ = false; 1122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl is resposible for activating chrome in Windows 8. The 1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// flow is complicated and this tries to highlight the important events. 1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// The current approach is to have a single instance of chrome either 1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// running in desktop or metro mode. If there is no current instance then 1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// the desktop shortcut launches desktop chrome and the metro tile or search 1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// charm launches metro chrome. 1195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// If chrome is running then focus/activation is given to the existing one 1205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// regarless of what launch point the user used. 1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// The general flow when chrome is the default browser is as follows: 1235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 1- User interacts with launch point (icon, tile, search, shellexec, etc) 1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 2- Windows finds the appid for launch item and resolves it to chrome 1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 3- Windows activates CommandExecuteImpl inside a surrogate process 1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 4- Windows calls the following sequence of entry points: 1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl::SetShowWindow 1295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl::SetPosition 1305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl::SetDirectory 1315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl::SetParameter 1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl::SetNoShowUI 1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl::SetSelection 1345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl::Initialize 1355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Up to this point the code basically just gathers values passed in, like 1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// the launch scheme (or url) and the activation verb. 1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 5- Windows calls CommandExecuteImpl::Getvalue() 1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Here we need to return AHE_IMMERSIVE or AHE_DESKTOP. That depends on: 1395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// a) if run in high-integrity return AHE_DESKTOP 1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// b) if chrome is running return the AHE_ mode of chrome 1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// c) else we return what GetLaunchMode() tells us, which is: 1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// i) if the command line --force-xxx is present return that 1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// ii) if the registry 'launch_mode' exists return that 144c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)// iii) if IsTouchEnabledDevice() is true return AHE_IMMERSIVE 1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// iv) else return AHE_DESKTOP 1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 6- If we returned AHE_IMMERSIVE in step 5 windows might not call us back 1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// and simply activate chrome in metro by itself, however in some cases 1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// it might proceed at step 7. 1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// As far as we know if we return AHE_DESKTOP then step 7 always happens. 1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 7- Windows calls CommandExecuteImpl::Execute() 1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Here we call GetLaunchMode() which returns the cached answer 1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// computed at step 5c. which can be: 1532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// a) ECHUIM_DESKTOP then we call LaunchDesktopChrome() that calls 1545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// ::CreateProcess and we exit at this point even on failure. 1555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// b) else we call one of the IApplicationActivationManager activation 1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// functions depending on the parameters passed in step 4. 1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// c) If the activation returns E_APPLICATION_NOT_REGISTERED, then we fall 1585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// back to launching chrome on the desktop via LaunchDestopChrome(). 1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 1605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Note that if a command line --force-xxx is present we write that launch mode 1615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// in the registry so next time the logic reaches 5c-ii it will use the same 1625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// mode again. 1635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 164c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)// Also note that if we are not the default browser and IsTouchEnabledDevice() 1655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// returns true, launching chrome can go all the way to 7c, which might be 1665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// a slow way to start chrome. 1675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// 1685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)CommandExecuteImpl::CommandExecuteImpl() 1695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) : parameters_(CommandLine::NO_PROGRAM), 1705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_scheme_(INTERNET_SCHEME_DEFAULT), 1715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) integrity_level_(base::INTEGRITY_UNKNOWN), 1725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome_mode_(ECHUIM_SYSTEM_LAUNCHER) { 1735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) memset(&start_info_, 0, sizeof(start_info_)); 1745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) start_info_.cb = sizeof(start_info_); 1752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We need to query the user data dir of chrome so we need chrome's 1772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // path provider. We can be created multiplie times in a single instance 1782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // however so make sure we do this only once. 1792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!path_provider_initialized_) { 1802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) chrome::RegisterPathProvider(); 1812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) path_provider_initialized_ = true; 1822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 1835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// CommandExecuteImpl 1865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::SetKeyState(DWORD key_state) { 1875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 1885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 1895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::SetParameters(LPCWSTR params) { 1925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs [%S]\n", __FUNCTION__, params); 1935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) parameters_ = delegate_execute::CommandLineFromParameters(params); 1945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 1955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::SetPosition(POINT pt) { 1985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 1995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 2005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::SetShowWindow(int show) { 2035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs show=%d\n", __FUNCTION__, show); 2045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) start_info_.wShowWindow = show; 2055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) start_info_.dwFlags |= STARTF_USESHOWWINDOW; 2065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 2075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::SetNoShowUI(BOOL no_show_ui) { 2105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs no_show=%d\n", __FUNCTION__, no_show_ui); 2115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 2125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::SetDirectory(LPCWSTR directory) { 2155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 2165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 2175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::GetValue(enum AHE_TYPE* pahe) { 2205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 2215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!GetLaunchScheme(&display_name_, &launch_scheme_)) { 2235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to get scheme, E_FAIL\n"); 2245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return E_FAIL; 2255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (integrity_level_ == base::HIGH_INTEGRITY) { 2285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Metro mode apps don't work in high integrity mode. 2295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("High integrity, AHE_DESKTOP\n"); 2305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *pahe = AHE_DESKTOP; 2315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 2325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (GetAsyncKeyState(VK_SHIFT) && GetAsyncKeyState(VK_F11)) { 2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Using Shift-F11 debug hook, returning AHE_IMMERSIVE\n"); 2365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *pahe = AHE_IMMERSIVE; 2372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 2382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#if defined(USE_AURA) 2392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Launch the chrome browser process that metro chrome will connect to. 2402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) LaunchChromeBrowserProcess(); 2412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#endif 2422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 2432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return S_OK; 2442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 2452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 2462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (GetAsyncKeyState(VK_SHIFT) && GetAsyncKeyState(VK_F12)) { 2472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AtlTrace("Using Shift-F12 debug hook, returning AHE_DESKTOP\n"); 2482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) *pahe = AHE_DESKTOP; 2495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 2505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::FilePath user_data_dir; 2535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!PathService::Get(chrome::DIR_USER_DATA, &user_data_dir)) { 2545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to get chrome's data dir path, E_FAIL\n"); 2555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return E_FAIL; 2565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) bool decision_made = false; 259c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 260c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // New Aura/Ash world we don't want to go throgh FindWindow path 261c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // and instead take decision based on launch mode. 262c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#if !defined(USE_AURA) 2637dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch HWND chrome_window = base::win::MessageWindow::FindWindow( 2647dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch user_data_dir.value()); 2655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (chrome_window) { 2665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Found chrome window %p\n", chrome_window); 2675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // The failure cases below are deemed to happen due to the inherently racy 2685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // procedure of going from chrome's window to process handle during which 2695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // chrome might have exited. Failing here would probably just cause the 2705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // user to retry at which point we would do the right thing. 2715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DWORD chrome_pid = 0; 2725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ::GetWindowThreadProcessId(chrome_window, &chrome_pid); 2735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!chrome_pid) { 2745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to get chrome's PID, E_FAIL\n"); 2755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return E_FAIL; 2765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::win::ScopedHandle process( 2785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ::OpenProcess(PROCESS_QUERY_INFORMATION, FALSE, chrome_pid)); 2795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!process.IsValid()) { 2805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to open chrome's process [%d], E_FAIL\n", chrome_pid); 2815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return E_FAIL; 2825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 2845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (IsImmersiveProcess(process.Get())) { 2855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Chrome [%d] is inmmersive, AHE_IMMERSIVE\n", chrome_pid); 2865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome_mode_ = ECHUIM_IMMERSIVE; 2875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *pahe = AHE_IMMERSIVE; 2885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 2895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Chrome [%d] is Desktop, AHE_DESKTOP\n"); 2905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome_mode_ = ECHUIM_DESKTOP; 2915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *pahe = AHE_DESKTOP; 2925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 2942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) decision_made = true; 2955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 296c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#endif 2975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!decision_made) { 2992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) EC_HOST_UI_MODE mode = GetLaunchMode(); 3002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) *pahe = (mode == ECHUIM_DESKTOP) ? AHE_DESKTOP : AHE_IMMERSIVE; 3012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 3022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 3032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#if defined(USE_AURA) 304a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles) if (*pahe == AHE_IMMERSIVE && verb_ != win8::kMetroViewerConnectVerb) 3052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) LaunchChromeBrowserProcess(); 3062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#endif 3072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 3085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 3095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::Execute() { 3125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 3135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (integrity_level_ == base::HIGH_INTEGRITY) 3155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return LaunchDesktopChrome(); 3165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EC_HOST_UI_MODE mode = GetLaunchMode(); 3185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (mode == ECHUIM_DESKTOP) 3195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return LaunchDesktopChrome(); 3205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) HRESULT hr = E_FAIL; 3225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CComPtr<IApplicationActivationManager> activation_manager; 3235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) hr = activation_manager.CoCreateInstance(CLSID_ApplicationActivationManager); 3245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!activation_manager) { 3255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to get the activation manager, error 0x%x\n", hr); 3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 3275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 329d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles) BrowserDistribution* distribution = BrowserDistribution::GetDistribution(); 330d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles) bool is_per_user_install = InstallUtil::IsPerUserInstall( 331d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles) chrome_exe_.value().c_str()); 332d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles) string16 app_id = ShellUtil::GetBrowserModelId( 333d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles) distribution, is_per_user_install); 3345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DWORD pid = 0; 3365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (launch_scheme_ == INTERNET_SCHEME_FILE && 3375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) display_name_.find(installer::kChromeExe) != string16::npos) { 3385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Activating for file\n"); 3395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) hr = activation_manager->ActivateApplication(app_id.c_str(), 3405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) verb_.c_str(), 3415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AO_NONE, 3425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &pid); 3435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 3445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Activating for protocol\n"); 3455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) hr = activation_manager->ActivateForProtocol(app_id.c_str(), 3465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) item_array_, 3475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &pid); 3485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (hr == E_APPLICATION_NOT_REGISTERED) { 3505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Metro chrome is not registered, launching in desktop\n"); 3515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return LaunchDesktopChrome(); 3525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Metro Chrome launch, pid=%d, returned 0x%x\n", pid, hr); 3545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 3555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::Initialize(LPCWSTR name, 3585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IPropertyBag* bag) { 3595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 3605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!FindChromeExe(&chrome_exe_)) 3615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return E_FAIL; 3625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) delegate_execute::UpdateChromeIfNeeded(chrome_exe_); 3635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (name) { 3645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Verb is %S\n", name); 3655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) verb_ = name; 3665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::GetProcessIntegrityLevel(base::GetCurrentProcessHandle(), 3695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &integrity_level_); 3705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (integrity_level_ == base::HIGH_INTEGRITY) { 3715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Delegate execute launched in high integrity level\n"); 3725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 3745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::SetSelection(IShellItemArray* item_array) { 3775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 3785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) item_array_ = item_array; 3795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 3805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::GetSelection(REFIID riid, void** selection) { 3835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 3845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 3855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)STDMETHODIMP CommandExecuteImpl::AllowForegroundTransfer(void* reserved) { 3885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 3895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 3905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Returns false if chrome.exe cannot be found. 3935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static 3942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool CommandExecuteImpl::FindChromeExe(base::FilePath* chrome_exe) { 3955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 3965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Look for chrome.exe one folder above delegate_execute.exe (as expected in 3975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Chrome installs). Failing that, look for it alonside delegate_execute.exe. 3982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::FilePath dir_exe; 3995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!PathService::Get(base::DIR_EXE, &dir_exe)) { 4005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to get current exe path\n"); 4015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 4025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *chrome_exe = dir_exe.DirName().Append(chrome::kBrowserProcessExecutableName); 4057dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch if (!base::PathExists(*chrome_exe)) { 4065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *chrome_exe = dir_exe.Append(chrome::kBrowserProcessExecutableName); 4077dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch if (!base::PathExists(*chrome_exe)) { 4085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to find chrome exe file\n"); 4095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 4105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Got chrome exe path as %ls\n", chrome_exe->value().c_str()); 4145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 4155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool CommandExecuteImpl::GetLaunchScheme( 4185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) string16* display_name, INTERNET_SCHEME* scheme) { 4195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!item_array_) 4205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 4215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ATLASSERT(display_name); 4235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ATLASSERT(scheme); 4245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DWORD count = 0; 4265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) item_array_->GetCount(&count); 4275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (count != 1) { 4295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Cannot handle %d elements in the IShellItemArray\n", count); 4305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 4315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CComPtr<IEnumShellItems> items; 4345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) item_array_->EnumItems(&items); 4355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CComPtr<IShellItem> shell_item; 4365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) HRESULT hr = items->Next(1, &shell_item, &count); 4375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (hr != S_OK) { 4385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to read element from the IShellItemsArray\n"); 4395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 4405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) hr = GetUrlFromShellItem(shell_item, display_name); 4435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (FAILED(hr)) { 4445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to get url. Error 0x%x\n", hr); 4455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 4465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("url [%ls]\n", display_name->c_str()); 4495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) wchar_t scheme_name[16]; 4515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) URL_COMPONENTS components = {0}; 4525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) components.lpszScheme = scheme_name; 4535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) components.dwSchemeLength = sizeof(scheme_name)/sizeof(scheme_name[0]); 4545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) components.dwStructSize = sizeof(components); 4565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!InternetCrackUrlW(display_name->c_str(), 0, 0, &components)) { 4575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to crack url %ls\n", display_name->c_str()); 4585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 4595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Launch scheme is [%ls] (%d)\n", scheme_name, components.nScheme); 4625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *scheme = components.nScheme; 4635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 4645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)HRESULT CommandExecuteImpl::LaunchDesktopChrome() { 4675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("In %hs\n", __FUNCTION__); 4685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) string16 display_name = display_name_; 4695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) switch (launch_scheme_) { 4715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case INTERNET_SCHEME_FILE: 4725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If anything other than chrome.exe is passed in the display name we 4735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // should honor it. For e.g. If the user clicks on a html file when 4745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // chrome is the default we should treat it as a parameter to be passed 4755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // to chrome. 4765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (display_name.find(installer::kChromeExe) != string16::npos) 4775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) display_name.clear(); 4785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 4795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) default: 4815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 4825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CommandLine chrome( 4855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) delegate_execute::MakeChromeCommandLine(chrome_exe_, parameters_, 4865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) display_name)); 4875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) string16 command_line(chrome.GetCommandLineString()); 4885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Formatted command line is %ls\n", command_line.c_str()); 4905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 491f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles) PROCESS_INFORMATION temp_process_info = {}; 4925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) BOOL ret = CreateProcess(chrome_exe_.value().c_str(), 4935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const_cast<LPWSTR>(command_line.c_str()), 4945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NULL, NULL, FALSE, 0, NULL, NULL, &start_info_, 495f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles) &temp_process_info); 4965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (ret) { 497f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles) base::win::ScopedProcessInformation proc_info(temp_process_info); 4985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Process id is %d\n", proc_info.process_id()); 4995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AllowSetForegroundWindow(proc_info.process_id()); 5005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 5015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Process launch failed, error %d\n", ::GetLastError()); 5025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 5035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return S_OK; 5055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 5065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)EC_HOST_UI_MODE CommandExecuteImpl::GetLaunchMode() { 5085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // See the header file for an explanation of the mode selection logic. 5095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) static bool launch_mode_determined = false; 5105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) static EC_HOST_UI_MODE launch_mode = ECHUIM_DESKTOP; 5115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const char* modes[] = { "Desktop", "Immersive", "SysLauncher", "??" }; 5135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (launch_mode_determined) 5155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return launch_mode; 5165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (chrome_mode_ != ECHUIM_SYSTEM_LAUNCHER) { 5185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode = chrome_mode_; 5195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Launch mode is that of chrome, %s\n", modes[launch_mode]); 5205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode_determined = true; 5215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return launch_mode; 5225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 5235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (parameters_.HasSwitch(switches::kForceImmersive)) { 5255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode = ECHUIM_IMMERSIVE; 5265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode_determined = true; 5275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) parameters_ = CommandLine(CommandLine::NO_PROGRAM); 5285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (parameters_.HasSwitch(switches::kForceDesktop)) { 5295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode = ECHUIM_DESKTOP; 5305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode_determined = true; 5315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) parameters_ = CommandLine(CommandLine::NO_PROGRAM); 5325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 5335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::win::RegKey reg_key; 5355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LONG key_result = reg_key.Create(HKEY_CURRENT_USER, 5365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::kMetroRegistryPath, 5375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) KEY_ALL_ACCESS); 5385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (key_result != ERROR_SUCCESS) { 5395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Failed to open HKCU %ls key, error 0x%x\n", 5405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::kMetroRegistryPath, 5415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) key_result); 5425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!launch_mode_determined) { 5435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode = ECHUIM_DESKTOP; 5445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode_determined = true; 5455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 5465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return launch_mode; 5475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 5485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (launch_mode_determined) { 5505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Launch mode forced by cmdline to %s\n", modes[launch_mode]); 5515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) reg_key.WriteValue(chrome::kLaunchModeValue, 5525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) static_cast<DWORD>(launch_mode)); 5535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return launch_mode; 5545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 5555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DWORD reg_value; 5575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (reg_key.ReadValueDW(chrome::kLaunchModeValue, 5585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ®_value) != ERROR_SUCCESS) { 5590f1bc08d4cfcc34181b0b5cbf065c40f687bf740Torne (Richard Coles) launch_mode = ECHUIM_DESKTOP; 5605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Launch mode forced by heuristics to %s\n", modes[launch_mode]); 5615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (reg_value >= ECHUIM_SYSTEM_LAUNCHER) { 5625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AtlTrace("Invalid registry launch mode value %u\n", reg_value); 5635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode = ECHUIM_DESKTOP; 5645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 5655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode = static_cast<EC_HOST_UI_MODE>(reg_value); 566c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) AtlTrace("Launch mode forced by registry to %s\n", modes[launch_mode]); 5675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 5685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_mode_determined = true; 5705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return launch_mode; 5715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 572