shell_integration.cc revision a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7
1// Copyright (c) 2012 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "chrome/browser/shell_integration.h"
6
7#include "base/bind.h"
8#include "base/command_line.h"
9#include "base/file_util.h"
10#include "base/path_service.h"
11#include "base/prefs/pref_service.h"
12#include "base/strings/string_util.h"
13#include "base/strings/utf_string_conversions.h"
14#include "base/threading/thread_restrictions.h"
15#include "chrome/browser/policy/policy_path_parser.h"
16#include "chrome/common/chrome_paths.h"
17#include "chrome/common/chrome_switches.h"
18#include "chrome/common/pref_names.h"
19#include "content/public/browser/browser_thread.h"
20
21#if defined(OS_CHROMEOS)
22#include "chromeos/chromeos_switches.h"
23#endif
24
25#if !defined(OS_WIN)
26#include "chrome/common/chrome_version_info.h"
27#include "grit/chromium_strings.h"
28#include "ui/base/l10n/l10n_util.h"
29#endif
30
31using content::BrowserThread;
32
33ShellIntegration::DefaultWebClientSetPermission
34    ShellIntegration::CanSetAsDefaultProtocolClient() {
35  // Allowed as long as the browser can become the operating system default
36  // browser.
37  return CanSetAsDefaultBrowser();
38}
39
40ShellIntegration::ShortcutInfo::ShortcutInfo()
41    : is_platform_app(false) {
42}
43
44ShellIntegration::ShortcutInfo::~ShortcutInfo() {}
45
46ShellIntegration::ShortcutLocations::ShortcutLocations()
47    : on_desktop(false),
48      applications_menu_location(APP_MENU_LOCATION_NONE),
49      in_quick_launch_bar(false)
50#if defined(OS_POSIX)
51      , hidden(false)
52#endif
53      {
54}
55
56static const struct ShellIntegration::AppModeInfo* gAppModeInfo = NULL;
57
58// static
59void ShellIntegration::SetAppModeInfo(const struct AppModeInfo* info) {
60  gAppModeInfo = info;
61}
62
63// static
64const struct ShellIntegration::AppModeInfo* ShellIntegration::AppModeInfo() {
65  return gAppModeInfo;
66}
67
68// static
69bool ShellIntegration::IsRunningInAppMode() {
70  return gAppModeInfo != NULL;
71}
72
73// static
74CommandLine ShellIntegration::CommandLineArgsForLauncher(
75    const GURL& url,
76    const std::string& extension_app_id,
77    const base::FilePath& profile_path) {
78  base::ThreadRestrictions::AssertIOAllowed();
79  const CommandLine& cmd_line = *CommandLine::ForCurrentProcess();
80  CommandLine new_cmd_line(CommandLine::NO_PROGRAM);
81
82  // Use the same UserDataDir for new launches that we currently have set.
83  base::FilePath user_data_dir =
84      cmd_line.GetSwitchValuePath(switches::kUserDataDir);
85#if defined(OS_MACOSX) || defined(OS_WIN)
86  policy::path_parser::CheckUserDataDirPolicy(&user_data_dir);
87#endif
88  if (!user_data_dir.empty()) {
89    // Make sure user_data_dir is an absolute path.
90    user_data_dir = base::MakeAbsoluteFilePath(user_data_dir);
91    if (!user_data_dir.empty() && base::PathExists(user_data_dir))
92      new_cmd_line.AppendSwitchPath(switches::kUserDataDir, user_data_dir);
93  }
94
95#if defined(OS_CHROMEOS)
96  base::FilePath profile = cmd_line.GetSwitchValuePath(
97      chromeos::switches::kLoginProfile);
98  if (!profile.empty())
99    new_cmd_line.AppendSwitchPath(chromeos::switches::kLoginProfile, profile);
100#else
101  if (!profile_path.empty() && !extension_app_id.empty())
102    new_cmd_line.AppendSwitchPath(switches::kProfileDirectory,
103                                  profile_path.BaseName());
104#endif
105
106  // If |extension_app_id| is present, we use the kAppId switch rather than
107  // the kApp switch (the launch url will be read from the extension app
108  // during launch.
109  if (!extension_app_id.empty()) {
110    new_cmd_line.AppendSwitchASCII(switches::kAppId, extension_app_id);
111  } else {
112    // Use '--app=url' instead of just 'url' to launch the browser with minimal
113    // chrome.
114    // Note: Do not change this flag!  Old Gears shortcuts will break if you do!
115    new_cmd_line.AppendSwitchASCII(switches::kApp, url.spec());
116  }
117  return new_cmd_line;
118}
119
120#if !defined(OS_WIN)
121
122base::string16 ShellIntegration::GetAppShortcutsSubdirName() {
123  if (chrome::VersionInfo::GetChannel() == chrome::VersionInfo::CHANNEL_CANARY)
124    return l10n_util::GetStringUTF16(IDS_APP_SHORTCUTS_SUBDIR_NAME_CANARY);
125  return l10n_util::GetStringUTF16(IDS_APP_SHORTCUTS_SUBDIR_NAME);
126}
127
128// static
129bool ShellIntegration::SetAsDefaultBrowserInteractive() {
130  return false;
131}
132
133// static
134bool ShellIntegration::SetAsDefaultProtocolClientInteractive(
135    const std::string& protocol) {
136  return false;
137}
138#endif
139
140bool ShellIntegration::DefaultWebClientObserver::IsOwnedByWorker() {
141  return false;
142}
143
144bool ShellIntegration::DefaultWebClientObserver::
145    IsInteractiveSetDefaultPermitted() {
146  return false;
147}
148
149///////////////////////////////////////////////////////////////////////////////
150// ShellIntegration::DefaultWebClientWorker
151//
152
153ShellIntegration::DefaultWebClientWorker::DefaultWebClientWorker(
154    DefaultWebClientObserver* observer)
155    : observer_(observer) {
156}
157
158void ShellIntegration::DefaultWebClientWorker::StartCheckIsDefault() {
159  if (observer_) {
160    observer_->SetDefaultWebClientUIState(STATE_PROCESSING);
161    BrowserThread::PostTask(
162        BrowserThread::FILE, FROM_HERE,
163        base::Bind(
164            &DefaultWebClientWorker::ExecuteCheckIsDefault, this));
165  }
166}
167
168void ShellIntegration::DefaultWebClientWorker::StartSetAsDefault() {
169  bool interactive_permitted = false;
170  if (observer_) {
171    observer_->SetDefaultWebClientUIState(STATE_PROCESSING);
172    interactive_permitted = observer_->IsInteractiveSetDefaultPermitted();
173  }
174  BrowserThread::PostTask(
175      BrowserThread::FILE, FROM_HERE,
176      base::Bind(&DefaultWebClientWorker::ExecuteSetAsDefault, this,
177                 interactive_permitted));
178}
179
180void ShellIntegration::DefaultWebClientWorker::ObserverDestroyed() {
181  // Our associated view has gone away, so we shouldn't call back to it if
182  // our worker thread returns after the view is dead.
183  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
184  observer_ = NULL;
185}
186
187///////////////////////////////////////////////////////////////////////////////
188// DefaultWebClientWorker, private:
189
190void ShellIntegration::DefaultWebClientWorker::ExecuteCheckIsDefault() {
191  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
192  DefaultWebClientState state = CheckIsDefault();
193  BrowserThread::PostTask(
194      BrowserThread::UI, FROM_HERE,
195      base::Bind(
196          &DefaultWebClientWorker::CompleteCheckIsDefault, this, state));
197}
198
199void ShellIntegration::DefaultWebClientWorker::CompleteCheckIsDefault(
200    DefaultWebClientState state) {
201  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
202  UpdateUI(state);
203  // The worker has finished everything it needs to do, so free the observer
204  // if we own it.
205  if (observer_ && observer_->IsOwnedByWorker()) {
206    delete observer_;
207    observer_ = NULL;
208  }
209}
210
211void ShellIntegration::DefaultWebClientWorker::ExecuteSetAsDefault(
212    bool interactive_permitted) {
213  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
214
215  bool result = SetAsDefault(interactive_permitted);
216  BrowserThread::PostTask(
217      BrowserThread::UI, FROM_HERE,
218      base::Bind(&DefaultWebClientWorker::CompleteSetAsDefault, this, result));
219}
220
221void ShellIntegration::DefaultWebClientWorker::CompleteSetAsDefault(
222    bool succeeded) {
223  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
224  // First tell the observer what the SetAsDefault call has returned.
225  if (observer_)
226    observer_->OnSetAsDefaultConcluded(succeeded);
227  // Set as default completed, check again to make sure it stuck...
228  StartCheckIsDefault();
229}
230
231void ShellIntegration::DefaultWebClientWorker::UpdateUI(
232    DefaultWebClientState state) {
233  if (observer_) {
234    switch (state) {
235      case NOT_DEFAULT:
236        observer_->SetDefaultWebClientUIState(STATE_NOT_DEFAULT);
237        break;
238      case IS_DEFAULT:
239        observer_->SetDefaultWebClientUIState(STATE_IS_DEFAULT);
240        break;
241      case UNKNOWN_DEFAULT:
242        observer_->SetDefaultWebClientUIState(STATE_UNKNOWN);
243        break;
244      default:
245        break;
246    }
247  }
248}
249
250///////////////////////////////////////////////////////////////////////////////
251// ShellIntegration::DefaultBrowserWorker
252//
253
254ShellIntegration::DefaultBrowserWorker::DefaultBrowserWorker(
255    DefaultWebClientObserver* observer)
256    : DefaultWebClientWorker(observer) {
257}
258
259///////////////////////////////////////////////////////////////////////////////
260// DefaultBrowserWorker, private:
261
262ShellIntegration::DefaultWebClientState
263ShellIntegration::DefaultBrowserWorker::CheckIsDefault() {
264  return ShellIntegration::GetDefaultBrowser();
265}
266
267bool ShellIntegration::DefaultBrowserWorker::SetAsDefault(
268    bool interactive_permitted) {
269  bool result = false;
270  switch (ShellIntegration::CanSetAsDefaultBrowser()) {
271    case ShellIntegration::SET_DEFAULT_UNATTENDED:
272      result = ShellIntegration::SetAsDefaultBrowser();
273      break;
274    case ShellIntegration::SET_DEFAULT_INTERACTIVE:
275      if (interactive_permitted)
276        result = ShellIntegration::SetAsDefaultBrowserInteractive();
277      break;
278    default:
279      NOTREACHED();
280  }
281
282  return result;
283}
284
285///////////////////////////////////////////////////////////////////////////////
286// ShellIntegration::DefaultProtocolClientWorker
287//
288
289ShellIntegration::DefaultProtocolClientWorker::DefaultProtocolClientWorker(
290    DefaultWebClientObserver* observer, const std::string& protocol)
291    : DefaultWebClientWorker(observer),
292      protocol_(protocol) {
293}
294
295///////////////////////////////////////////////////////////////////////////////
296// DefaultProtocolClientWorker, private:
297
298ShellIntegration::DefaultWebClientState
299ShellIntegration::DefaultProtocolClientWorker::CheckIsDefault() {
300  return ShellIntegration::IsDefaultProtocolClient(protocol_);
301}
302
303bool ShellIntegration::DefaultProtocolClientWorker::SetAsDefault(
304    bool interactive_permitted) {
305  bool result = false;
306  switch (ShellIntegration::CanSetAsDefaultProtocolClient()) {
307    case ShellIntegration::SET_DEFAULT_NOT_ALLOWED:
308      result = false;
309      break;
310    case ShellIntegration::SET_DEFAULT_UNATTENDED:
311      result = ShellIntegration::SetAsDefaultProtocolClient(protocol_);
312      break;
313    case ShellIntegration::SET_DEFAULT_INTERACTIVE:
314      if (interactive_permitted) {
315        result = ShellIntegration::SetAsDefaultProtocolClientInteractive(
316            protocol_);
317      }
318      break;
319  }
320
321  return result;
322}
323