blob: ce442d0bb338049275a79be234843a7f564a38af [file] [log] [blame]
// Copyright (c) 2011 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 "chrome/browser/extensions/extension_warning_set.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/extensions/extension_global_error_badge.h"
#include "chrome/browser/extensions/extension_service.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/profiles/profile_manager.h"
#include "chrome/browser/ui/global_error_service.h"
#include "chrome/browser/ui/global_error_service_factory.h"
#include "chrome/common/chrome_notification_types.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/notification_service.h"
#include "grit/chromium_strings.h"
#include "grit/generated_resources.h"
#include "ui/base/l10n/l10n_util.h"
using content::BrowserThread;
// This class is used to represent warnings if extensions misbehave.
class ExtensionWarning {
public:
// Default constructor for storing ExtensionServiceWarning in STL containers
// do not use.
ExtensionWarning();
// Constructs a warning of type |type| for extension |extension_id|. This
// could be for example the fact that an extension conflicted with others.
ExtensionWarning(ExtensionWarningSet::WarningType type,
const std::string& extension_id);
~ExtensionWarning();
// Returns the specific warning type.
ExtensionWarningSet::WarningType warning_type() const { return type_; }
// Returns the id of the extension for which this warning is valid.
const std::string& extension_id() const { return extension_id_; }
private:
ExtensionWarningSet::WarningType type_;
std::string extension_id_;
// Allow implicit copy and assign operator.
};
ExtensionWarning::ExtensionWarning() : type_(ExtensionWarningSet::kInvalid) {
}
ExtensionWarning::ExtensionWarning(
ExtensionWarningSet::WarningType type,
const std::string& extension_id)
: type_(type), extension_id_(extension_id) {
// These are invalid here because they do not have corresponding warning
// messages in the UI.
CHECK(type != ExtensionWarningSet::kInvalid);
CHECK(type != ExtensionWarningSet::kMaxWarningType);
}
ExtensionWarning::~ExtensionWarning() {
}
bool operator<(const ExtensionWarning& a, const ExtensionWarning& b) {
if (a.warning_type() == b.warning_type())
return a.extension_id() < b.extension_id();
return a.warning_type() < b.warning_type();
}
// Static
string16 ExtensionWarningSet::GetLocalizedWarning(
ExtensionWarningSet::WarningType warning_type) {
switch (warning_type) {
case kInvalid:
case kMaxWarningType:
NOTREACHED();
return string16();
case kNetworkDelay:
return l10n_util::GetStringFUTF16(
IDS_EXTENSION_WARNINGS_NETWORK_DELAY,
l10n_util::GetStringUTF16(IDS_PRODUCT_NAME));
case kNetworkConflict:
return l10n_util::GetStringUTF16(IDS_EXTENSION_WARNINGS_NETWORK_CONFLICT);
case kRepeatedCacheFlushes:
return l10n_util::GetStringFUTF16(
IDS_EXTENSION_WARNINGS_NETWORK_DELAY,
l10n_util::GetStringUTF16(IDS_PRODUCT_NAME));
}
NOTREACHED(); // Switch statement has no default branch.
return string16();
}
ExtensionWarningSet::ExtensionWarningSet(Profile* profile)
: extension_global_error_badge_(NULL), profile_(profile) {
}
ExtensionWarningSet::~ExtensionWarningSet() {
}
void ExtensionWarningSet::SetWarning(ExtensionWarningSet::WarningType type,
const std::string& extension_id) {
ExtensionWarning warning(type, extension_id);
bool inserted = warnings_.insert(warning).second;
if (inserted) {
NotifyWarningsChanged();
UpdateWarningBadge();
}
}
void ExtensionWarningSet::ClearWarnings(
const std::set<ExtensionWarningSet::WarningType>& types) {
bool deleted_anything = false;
for (const_iterator i = warnings_.begin(); i != warnings_.end();) {
if (types.find(i->warning_type()) != types.end()) {
deleted_anything = true;
warnings_.erase(i++);
} else {
++i;
}
}
if (deleted_anything) {
NotifyWarningsChanged();
UpdateWarningBadge();
}
}
void ExtensionWarningSet::GetWarningsAffectingExtension(
const std::string& extension_id,
std::set<ExtensionWarningSet::WarningType>* result) const {
result->clear();
for (const_iterator i = warnings_.begin(); i != warnings_.end(); ++i) {
if (i->extension_id() == extension_id)
result->insert(i->warning_type());
}
}
// static
void ExtensionWarningSet::NotifyWarningsOnUI(
void* profile_id,
std::set<std::string> extension_ids,
WarningType warning_type) {
CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
Profile* profile = reinterpret_cast<Profile*>(profile_id);
if (!profile ||
!g_browser_process->profile_manager()->IsValidProfile(profile)) {
return;
}
ExtensionWarningSet* warnings =
profile->GetExtensionService()->extension_warnings();
for (std::set<std::string>::const_iterator i = extension_ids.begin();
i != extension_ids.end(); ++i) {
warnings->SetWarning(warning_type, *i);
}
}
void ExtensionWarningSet::SuppressBadgeForCurrentWarnings() {
badge_suppressions_.insert(warnings_.begin(), warnings_.end());
UpdateWarningBadge();
}
void ExtensionWarningSet::NotifyWarningsChanged() {
content::NotificationService::current()->Notify(
chrome::NOTIFICATION_EXTENSION_WARNING_CHANGED,
content::Source<Profile>(profile_),
content::NotificationService::NoDetails());
}
void ExtensionWarningSet::ActivateBadge() {
DCHECK(!extension_global_error_badge_);
DCHECK(profile_);
extension_global_error_badge_ = new ExtensionGlobalErrorBadge;
GlobalErrorServiceFactory::GetForProfile(profile_)->AddGlobalError(
extension_global_error_badge_);
}
void ExtensionWarningSet::DeactivateBadge() {
DCHECK(extension_global_error_badge_);
DCHECK(profile_);
GlobalErrorServiceFactory::GetForProfile(profile_)->RemoveGlobalError(
extension_global_error_badge_);
delete extension_global_error_badge_;
extension_global_error_badge_ = NULL;
}
void ExtensionWarningSet::UpdateWarningBadge() {
// We need a badge if a warning exists that has not been suppressed.
bool need_warning_badge = false;
for (const_iterator i = warnings_.begin(); i != warnings_.end(); ++i) {
if (badge_suppressions_.find(*i) == badge_suppressions_.end()) {
need_warning_badge = true;
break;
}
}
// Activate or hide the warning badge in case the current state is incorrect.
if (extension_global_error_badge_ && !need_warning_badge)
DeactivateBadge();
else if (!extension_global_error_badge_ && need_warning_badge)
ActivateBadge();
}