| // Copyright 2013 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "apps/app_load_service.h" |
| |
| #include "apps/app_load_service_factory.h" |
| #include "apps/app_restore_service.h" |
| #include "apps/launcher.h" |
| #include "chrome/browser/extensions/extension_service.h" |
| #include "chrome/browser/extensions/unpacked_installer.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "content/public/browser/notification_details.h" |
| #include "content/public/browser/notification_service.h" |
| #include "content/public/browser/notification_types.h" |
| #include "extensions/browser/app_window/app_window_registry.h" |
| #include "extensions/browser/extension_host.h" |
| #include "extensions/browser/extension_prefs.h" |
| #include "extensions/browser/extension_registry.h" |
| #include "extensions/browser/extension_system.h" |
| #include "extensions/browser/notification_types.h" |
| #include "extensions/common/extension.h" |
| |
| using extensions::Extension; |
| using extensions::ExtensionPrefs; |
| using extensions::ExtensionSystem; |
| |
| namespace apps { |
| |
| AppLoadService::PostReloadAction::PostReloadAction() |
| : action_type(LAUNCH_FOR_RELOAD), |
| command_line(base::CommandLine::NO_PROGRAM) { |
| } |
| |
| AppLoadService::AppLoadService(Profile* profile) |
| : profile_(profile) { |
| registrar_.Add(this, |
| extensions::NOTIFICATION_EXTENSION_HOST_DID_STOP_FIRST_LOAD, |
| content::NotificationService::AllSources()); |
| extensions::ExtensionRegistry::Get(profile_)->AddObserver(this); |
| } |
| |
| AppLoadService::~AppLoadService() { |
| extensions::ExtensionRegistry::Get(profile_)->RemoveObserver(this); |
| } |
| |
| void AppLoadService::RestartApplication(const std::string& extension_id) { |
| post_reload_actions_[extension_id].action_type = RESTART; |
| ExtensionService* service = extensions::ExtensionSystem::Get(profile_)-> |
| extension_service(); |
| DCHECK(service); |
| service->ReloadExtension(extension_id); |
| } |
| |
| void AppLoadService::RestartApplicationIfRunning( |
| const std::string& extension_id) { |
| if (apps::AppRestoreService::Get(profile_)->IsAppRestorable(extension_id)) |
| RestartApplication(extension_id); |
| } |
| |
| bool AppLoadService::LoadAndLaunch(const base::FilePath& extension_path, |
| const base::CommandLine& command_line, |
| const base::FilePath& current_dir) { |
| ExtensionService* extension_service = |
| ExtensionSystem::Get(profile_)->extension_service(); |
| std::string extension_id; |
| if (!extensions::UnpackedInstaller::Create(extension_service)-> |
| LoadFromCommandLine(base::FilePath(extension_path), &extension_id)) { |
| return false; |
| } |
| |
| // Schedule the app to be launched once loaded. |
| PostReloadAction& action = post_reload_actions_[extension_id]; |
| action.action_type = LAUNCH_FOR_LOAD_AND_LAUNCH; |
| action.command_line = command_line; |
| action.current_dir = current_dir; |
| return true; |
| } |
| |
| bool AppLoadService::Load(const base::FilePath& extension_path) { |
| ExtensionService* extension_service = |
| ExtensionSystem::Get(profile_)->extension_service(); |
| std::string extension_id; |
| return extensions::UnpackedInstaller::Create(extension_service)-> |
| LoadFromCommandLine(base::FilePath(extension_path), &extension_id); |
| } |
| |
| // static |
| AppLoadService* AppLoadService::Get(Profile* profile) { |
| return apps::AppLoadServiceFactory::GetForProfile(profile); |
| } |
| |
| void AppLoadService::Observe(int type, |
| const content::NotificationSource& source, |
| const content::NotificationDetails& details) { |
| DCHECK_EQ(type, extensions::NOTIFICATION_EXTENSION_HOST_DID_STOP_FIRST_LOAD); |
| extensions::ExtensionHost* host = |
| content::Details<extensions::ExtensionHost>(details).ptr(); |
| const Extension* extension = host->extension(); |
| // It is possible for an extension to be unloaded before it stops loading. |
| if (!extension) |
| return; |
| std::map<std::string, PostReloadAction>::iterator it = |
| post_reload_actions_.find(extension->id()); |
| if (it == post_reload_actions_.end()) |
| return; |
| |
| switch (it->second.action_type) { |
| case LAUNCH_FOR_RELOAD: |
| LaunchPlatformApp(profile_, extension, extensions::SOURCE_RELOAD); |
| break; |
| case RESTART: |
| RestartPlatformApp(profile_, extension); |
| break; |
| case LAUNCH_FOR_LOAD_AND_LAUNCH: |
| LaunchPlatformAppWithCommandLine(profile_, |
| extension, |
| it->second.command_line, |
| it->second.current_dir, |
| extensions::SOURCE_LOAD_AND_LAUNCH); |
| break; |
| default: |
| NOTREACHED(); |
| } |
| |
| post_reload_actions_.erase(it); |
| } |
| |
| void AppLoadService::OnExtensionUnloaded( |
| content::BrowserContext* browser_context, |
| const Extension* extension, |
| extensions::UnloadedExtensionInfo::Reason reason) { |
| if (!extension->is_platform_app()) |
| return; |
| |
| extensions::ExtensionPrefs* extension_prefs = |
| extensions::ExtensionPrefs::Get(browser_context); |
| if (WasUnloadedForReload(extension->id(), reason) && |
| extension_prefs->IsActive(extension->id()) && |
| !HasPostReloadAction(extension->id())) { |
| post_reload_actions_[extension->id()].action_type = LAUNCH_FOR_RELOAD; |
| } |
| } |
| |
| bool AppLoadService::WasUnloadedForReload( |
| const extensions::ExtensionId& extension_id, |
| const extensions::UnloadedExtensionInfo::Reason reason) { |
| if (reason == extensions::UnloadedExtensionInfo::REASON_DISABLE) { |
| ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_); |
| return (prefs->GetDisableReasons(extension_id) & |
| Extension::DISABLE_RELOAD) != 0; |
| } |
| return false; |
| } |
| |
| bool AppLoadService::HasPostReloadAction(const std::string& extension_id) { |
| return post_reload_actions_.find(extension_id) != post_reload_actions_.end(); |
| } |
| |
| } // namespace apps |