blob: dd9639c883dafcefecf8019ea7770d7fcf7034e7 [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "apps/app_restore_service.h"
#include "apps/app_lifetime_monitor_factory.h"
#include "apps/saved_files_service.h"
#include "chrome/browser/extensions/api/app_runtime/app_runtime_api.h"
#include "chrome/browser/extensions/extension_host.h"
#include "chrome/browser/extensions/extension_prefs.h"
#include "chrome/browser/extensions/extension_service.h"
#include "chrome/browser/extensions/extension_system.h"
#include "chrome/browser/extensions/platform_app_launcher.h"
#include "chrome/common/chrome_notification_types.h"
#include "chrome/common/extensions/extension.h"
#include "chrome/common/extensions/extension_set.h"
#include "content/public/browser/notification_details.h"
#include "content/public/browser/notification_service.h"
#if defined(OS_WIN)
#include "win8/util/win8_util.h"
#endif
using extensions::Extension;
using extensions::ExtensionHost;
using extensions::ExtensionPrefs;
using extensions::ExtensionSystem;
namespace apps {
// static
bool AppRestoreService::ShouldRestoreApps(bool is_browser_restart) {
bool should_restore_apps = is_browser_restart;
#if defined(OS_CHROMEOS)
// Chromeos always restarts apps, even if it was a regular shutdown.
should_restore_apps = true;
#elif defined(OS_WIN)
// Packaged apps are not supported in Metro mode, so don't try to start them.
if (win8::IsSingleWindowMetroMode())
should_restore_apps = false;
#endif
return should_restore_apps;
}
AppRestoreService::AppRestoreService(Profile* profile)
: profile_(profile) {
StartObservingAppLifetime();
registrar_.Add(
this, chrome::NOTIFICATION_APP_TERMINATING,
content::NotificationService::AllSources());
}
void AppRestoreService::HandleStartup(bool should_restore_apps) {
ExtensionService* extension_service =
ExtensionSystem::Get(profile_)->extension_service();
const ExtensionSet* extensions = extension_service->extensions();
ExtensionPrefs* extension_prefs = extension_service->extension_prefs();
for (ExtensionSet::const_iterator it = extensions->begin();
it != extensions->end(); ++it) {
const Extension* extension = *it;
if (extension_prefs->IsExtensionRunning(extension->id())) {
RecordAppStop(extension->id());
// If we are not restoring apps (e.g., because it is a clean restart), and
// the app does not have retain permission, explicitly clear the retained
// entries queue.
if (should_restore_apps) {
RestoreApp(*it);
} else {
SavedFilesService::Get(profile_)->ClearQueueIfNoRetainPermission(
extension);
}
}
}
}
void AppRestoreService::Observe(int type,
const content::NotificationSource& source,
const content::NotificationDetails& details) {
switch (type) {
case chrome::NOTIFICATION_APP_TERMINATING: {
// Stop listening to NOTIFICATION_EXTENSION_HOST_DESTROYED in particular
// as all extension hosts will be destroyed as a result of shutdown.
registrar_.RemoveAll();
// We want to preserve the state when the app begins terminating, so stop
// listening to app lifetime events.
StopObservingAppLifetime();
break;
}
}
}
void AppRestoreService::OnAppStart(Profile* profile,
const std::string& app_id) {
RecordAppStart(app_id);
}
void AppRestoreService::OnAppActivated(Profile* profile,
const std::string& app_id) {
RecordAppActiveState(app_id, true);
}
void AppRestoreService::OnAppDeactivated(Profile* profile,
const std::string& app_id) {
RecordAppActiveState(app_id, false);
}
void AppRestoreService::OnAppStop(Profile* profile, const std::string& app_id) {
RecordAppStop(app_id);
}
void AppRestoreService::Shutdown() {
StopObservingAppLifetime();
}
void AppRestoreService::RecordAppStart(const std::string& extension_id) {
ExtensionPrefs* extension_prefs =
ExtensionSystem::Get(profile_)->extension_service()->extension_prefs();
extension_prefs->SetExtensionRunning(extension_id, true);
}
void AppRestoreService::RecordAppStop(const std::string& extension_id) {
ExtensionPrefs* extension_prefs =
ExtensionSystem::Get(profile_)->extension_service()->extension_prefs();
extension_prefs->SetExtensionRunning(extension_id, false);
}
void AppRestoreService::RecordAppActiveState(const std::string& id,
bool is_active) {
ExtensionService* extension_service =
ExtensionSystem::Get(profile_)->extension_service();
ExtensionPrefs* extension_prefs = extension_service->extension_prefs();
// If the extension isn't running then we will already have recorded whether
// it is active or not.
if (!extension_prefs->IsExtensionRunning(id))
return;
extension_prefs->SetIsActive(id, is_active);
}
void AppRestoreService::RestoreApp(const Extension* extension) {
extensions::RestartPlatformApp(profile_, extension);
}
void AppRestoreService::StartObservingAppLifetime() {
AppLifetimeMonitorFactory::GetForProfile(profile_)->AddObserver(this);
}
void AppRestoreService::StopObservingAppLifetime() {
AppLifetimeMonitorFactory::GetForProfile(profile_)->RemoveObserver(this);
}
} // namespace apps