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)                          &reg_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