| // 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/themes/theme_service.h" |
| |
| #include "base/string_split.h" |
| #include "base/string_util.h" |
| #include "base/utf_string_conversions.h" |
| #include "chrome/browser/extensions/extension_service.h" |
| #include "chrome/browser/metrics/user_metrics.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/themes/browser_theme_pack.h" |
| #include "chrome/common/chrome_constants.h" |
| #include "chrome/common/pref_names.h" |
| #include "content/common/notification_service.h" |
| #include "content/common/notification_type.h" |
| #include "grit/app_resources.h" |
| #include "grit/theme_resources.h" |
| #include "ui/base/resource/resource_bundle.h" |
| |
| #if defined(OS_WIN) |
| #include "views/widget/widget_win.h" |
| #endif |
| |
| // Strings used in alignment properties. |
| const char* ThemeService::kAlignmentTop = "top"; |
| const char* ThemeService::kAlignmentBottom = "bottom"; |
| const char* ThemeService::kAlignmentLeft = "left"; |
| const char* ThemeService::kAlignmentRight = "right"; |
| |
| // Strings used in background tiling repetition properties. |
| const char* ThemeService::kTilingNoRepeat = "no-repeat"; |
| const char* ThemeService::kTilingRepeatX = "repeat-x"; |
| const char* ThemeService::kTilingRepeatY = "repeat-y"; |
| const char* ThemeService::kTilingRepeat = "repeat"; |
| |
| // The default theme if we haven't installed a theme yet or if we've clicked |
| // the "Use Classic" button. |
| const char* ThemeService::kDefaultThemeID = ""; |
| |
| namespace { |
| |
| // The default theme if we've gone to the theme gallery and installed the |
| // "Default" theme. We have to detect this case specifically. (By the time we |
| // realize we've installed the default theme, we already have an extension |
| // unpacked on the filesystem.) |
| const char* kDefaultThemeGalleryID = "hkacjpbfdknhflllbcmjibkdeoafencn"; |
| |
| SkColor TintForUnderline(SkColor input) { |
| return SkColorSetA(input, SkColorGetA(input) / 3); |
| } |
| |
| SkColor IncreaseLightness(SkColor color, double percent) { |
| color_utils::HSL result; |
| color_utils::SkColorToHSL(color, &result); |
| result.l += (1 - result.l) * percent; |
| return color_utils::HSLToSkColor(result, SkColorGetA(color)); |
| } |
| |
| // Default colors. |
| const SkColor kDefaultColorFrame = SkColorSetRGB(66, 116, 201); |
| const SkColor kDefaultColorFrameInactive = SkColorSetRGB(161, 182, 228); |
| const SkColor kDefaultColorFrameIncognito = SkColorSetRGB(83, 106, 139); |
| const SkColor kDefaultColorFrameIncognitoInactive = |
| SkColorSetRGB(126, 139, 156); |
| #if defined(OS_MACOSX) |
| const SkColor kDefaultColorToolbar = SkColorSetRGB(230, 230, 230); |
| #else |
| const SkColor kDefaultColorToolbar = SkColorSetRGB(223, 223, 223); |
| #endif |
| const SkColor kDefaultColorTabText = SK_ColorBLACK; |
| #if defined(OS_MACOSX) |
| const SkColor kDefaultColorBackgroundTabText = SK_ColorBLACK; |
| const SkColor kDefaultColorBookmarkText = SK_ColorBLACK; |
| #else |
| const SkColor kDefaultColorBackgroundTabText = SkColorSetRGB(64, 64, 64); |
| const SkColor kDefaultColorBookmarkText = SkColorSetRGB(18, 50, 114); |
| #endif |
| #if defined(OS_WIN) |
| const SkColor kDefaultColorNTPBackground = |
| color_utils::GetSysSkColor(COLOR_WINDOW); |
| const SkColor kDefaultColorNTPText = |
| color_utils::GetSysSkColor(COLOR_WINDOWTEXT); |
| const SkColor kDefaultColorNTPLink = |
| color_utils::GetSysSkColor(COLOR_HOTLIGHT); |
| #else |
| // TODO(beng): source from theme provider. |
| const SkColor kDefaultColorNTPBackground = SK_ColorWHITE; |
| const SkColor kDefaultColorNTPText = SK_ColorBLACK; |
| const SkColor kDefaultColorNTPLink = SkColorSetRGB(6, 55, 116); |
| #endif |
| const SkColor kDefaultColorNTPHeader = SkColorSetRGB(150, 150, 150); |
| const SkColor kDefaultColorNTPSection = SkColorSetRGB(229, 229, 229); |
| const SkColor kDefaultColorNTPSectionText = SK_ColorBLACK; |
| const SkColor kDefaultColorNTPSectionLink = SkColorSetRGB(6, 55, 116); |
| const SkColor kDefaultColorControlBackground = SkColorSetARGB(0, 0, 0, 0); |
| const SkColor kDefaultColorButtonBackground = SkColorSetARGB(0, 0, 0, 0); |
| #if defined(OS_MACOSX) |
| const SkColor kDefaultColorToolbarButtonStroke = SkColorSetARGB(75, 81, 81, 81); |
| const SkColor kDefaultColorToolbarButtonStrokeInactive = |
| SkColorSetARGB(75, 99, 99, 99); |
| const SkColor kDefaultColorToolbarBezel = SkColorSetRGB(247, 247, 247); |
| const SkColor kDefaultColorToolbarStroke = SkColorSetRGB(103, 103, 103); |
| const SkColor kDefaultColorToolbarStrokeInactive = SkColorSetRGB(123, 123, 123); |
| #endif |
| |
| // Default tints. |
| const color_utils::HSL kDefaultTintButtons = { -1, -1, -1 }; |
| const color_utils::HSL kDefaultTintFrame = { -1, -1, -1 }; |
| const color_utils::HSL kDefaultTintFrameInactive = { -1, -1, 0.75f }; |
| const color_utils::HSL kDefaultTintFrameIncognito = { -1, 0.2f, 0.35f }; |
| const color_utils::HSL kDefaultTintFrameIncognitoInactive = { -1, 0.3f, 0.6f }; |
| const color_utils::HSL kDefaultTintBackgroundTab = { -1, 0.5, 0.75 }; |
| |
| // Default display properties. |
| const int kDefaultDisplayPropertyNTPAlignment = |
| ThemeService::ALIGN_BOTTOM; |
| const int kDefaultDisplayPropertyNTPTiling = |
| ThemeService::NO_REPEAT; |
| const int kDefaultDisplayPropertyNTPInverseLogo = 0; |
| |
| // The sum of kFrameBorderThickness and kNonClientRestoredExtraThickness from |
| // OpaqueBrowserFrameView. |
| const int kRestoredTabVerticalOffset = 15; |
| |
| // The image resources we will allow people to theme. |
| const int kThemeableImages[] = { |
| IDR_THEME_FRAME, |
| IDR_THEME_FRAME_INACTIVE, |
| IDR_THEME_FRAME_INCOGNITO, |
| IDR_THEME_FRAME_INCOGNITO_INACTIVE, |
| IDR_THEME_TOOLBAR, |
| IDR_THEME_TAB_BACKGROUND, |
| IDR_THEME_TAB_BACKGROUND_INCOGNITO, |
| IDR_THEME_TAB_BACKGROUND_V, |
| IDR_THEME_NTP_BACKGROUND, |
| IDR_THEME_FRAME_OVERLAY, |
| IDR_THEME_FRAME_OVERLAY_INACTIVE, |
| IDR_THEME_BUTTON_BACKGROUND, |
| IDR_THEME_NTP_ATTRIBUTION, |
| IDR_THEME_WINDOW_CONTROL_BACKGROUND |
| }; |
| |
| bool HasThemeableImage(int themeable_image_id) { |
| static std::set<int> themeable_images; |
| if (themeable_images.empty()) { |
| themeable_images.insert( |
| kThemeableImages, kThemeableImages + arraysize(kThemeableImages)); |
| } |
| return themeable_images.count(themeable_image_id) > 0; |
| } |
| |
| // The image resources that will be tinted by the 'button' tint value. |
| // If you change this list, you must increment the version number in |
| // browser_theme_pack.cc, and you should assign persistent IDs to the |
| // data table at the start of said file or else tinted versions of |
| // these resources will not be created. |
| const int kToolbarButtonIDs[] = { |
| IDR_BACK, IDR_BACK_D, IDR_BACK_H, IDR_BACK_P, |
| IDR_FORWARD, IDR_FORWARD_D, IDR_FORWARD_H, IDR_FORWARD_P, |
| IDR_HOME, IDR_HOME_H, IDR_HOME_P, |
| IDR_RELOAD, IDR_RELOAD_H, IDR_RELOAD_P, |
| IDR_STOP, IDR_STOP_D, IDR_STOP_H, IDR_STOP_P, |
| IDR_LOCATIONBG_C, IDR_LOCATIONBG_L, IDR_LOCATIONBG_R, |
| IDR_BROWSER_ACTIONS_OVERFLOW, IDR_BROWSER_ACTIONS_OVERFLOW_H, |
| IDR_BROWSER_ACTIONS_OVERFLOW_P, |
| IDR_TOOLS, IDR_TOOLS_H, IDR_TOOLS_P, |
| IDR_MENU_DROPARROW, |
| IDR_THROBBER, IDR_THROBBER_WAITING, IDR_THROBBER_LIGHT, |
| }; |
| |
| // Writes the theme pack to disk on a separate thread. |
| class WritePackToDiskTask : public Task { |
| public: |
| WritePackToDiskTask(BrowserThemePack* pack, const FilePath& path) |
| : theme_pack_(pack), pack_path_(path) {} |
| |
| virtual void Run() { |
| if (!theme_pack_->WriteToDisk(pack_path_)) { |
| NOTREACHED() << "Could not write theme pack to disk"; |
| } |
| } |
| |
| private: |
| scoped_refptr<BrowserThemePack> theme_pack_; |
| FilePath pack_path_; |
| }; |
| |
| } // namespace |
| |
| bool ThemeService::IsThemeableImage(int resource_id) { |
| return HasThemeableImage(resource_id); |
| } |
| |
| ThemeService::ThemeService() |
| : rb_(ResourceBundle::GetSharedInstance()), |
| profile_(NULL), |
| number_of_infobars_(0) { |
| // Initialize the themeable image map so we can use it on other threads. |
| HasThemeableImage(0); |
| } |
| |
| ThemeService::~ThemeService() { |
| FreePlatformCaches(); |
| } |
| |
| void ThemeService::Init(Profile* profile) { |
| DCHECK(CalledOnValidThread()); |
| profile_ = profile; |
| |
| // Listen to EXTENSION_LOADED instead of EXTENSION_INSTALLED because |
| // the extension cannot yet be found via GetExtensionById() if it is |
| // installed but not loaded (which may confuse listeners to |
| // BROWSER_THEME_CHANGED). |
| registrar_.Add(this, |
| NotificationType::EXTENSION_LOADED, |
| Source<Profile>(profile_)); |
| |
| LoadThemePrefs(); |
| } |
| |
| SkBitmap* ThemeService::GetBitmapNamed(int id) const { |
| DCHECK(CalledOnValidThread()); |
| |
| SkBitmap* bitmap = NULL; |
| |
| if (theme_pack_.get()) |
| bitmap = theme_pack_->GetBitmapNamed(id); |
| |
| if (!bitmap) |
| bitmap = rb_.GetBitmapNamed(id); |
| |
| return bitmap; |
| } |
| |
| SkColor ThemeService::GetColor(int id) const { |
| DCHECK(CalledOnValidThread()); |
| |
| SkColor color; |
| if (theme_pack_.get() && theme_pack_->GetColor(id, &color)) |
| return color; |
| |
| // For backward compat with older themes, some newer colors are generated from |
| // older ones if they are missing. |
| switch (id) { |
| case COLOR_NTP_SECTION_HEADER_TEXT: |
| return IncreaseLightness(GetColor(COLOR_NTP_TEXT), 0.30); |
| case COLOR_NTP_SECTION_HEADER_TEXT_HOVER: |
| return GetColor(COLOR_NTP_TEXT); |
| case COLOR_NTP_SECTION_HEADER_RULE: |
| return IncreaseLightness(GetColor(COLOR_NTP_TEXT), 0.70); |
| case COLOR_NTP_SECTION_HEADER_RULE_LIGHT: |
| return IncreaseLightness(GetColor(COLOR_NTP_TEXT), 0.86); |
| case COLOR_NTP_TEXT_LIGHT: |
| return IncreaseLightness(GetColor(COLOR_NTP_TEXT), 0.40); |
| } |
| |
| return GetDefaultColor(id); |
| } |
| |
| bool ThemeService::GetDisplayProperty(int id, int* result) const { |
| if (theme_pack_.get()) |
| return theme_pack_->GetDisplayProperty(id, result); |
| |
| return GetDefaultDisplayProperty(id, result); |
| } |
| |
| bool ThemeService::ShouldUseNativeFrame() const { |
| if (HasCustomImage(IDR_THEME_FRAME)) |
| return false; |
| #if defined(OS_WIN) |
| return views::WidgetWin::IsAeroGlassEnabled(); |
| #else |
| return false; |
| #endif |
| } |
| |
| bool ThemeService::HasCustomImage(int id) const { |
| if (!HasThemeableImage(id)) |
| return false; |
| |
| if (theme_pack_) |
| return theme_pack_->HasCustomImage(id); |
| |
| return false; |
| } |
| |
| RefCountedMemory* ThemeService::GetRawData(int id) const { |
| // Check to see whether we should substitute some images. |
| int ntp_alternate; |
| GetDisplayProperty(NTP_LOGO_ALTERNATE, &ntp_alternate); |
| if (id == IDR_PRODUCT_LOGO && ntp_alternate != 0) |
| id = IDR_PRODUCT_LOGO_WHITE; |
| |
| RefCountedMemory* data = NULL; |
| if (theme_pack_.get()) |
| data = theme_pack_->GetRawData(id); |
| if (!data) |
| data = rb_.LoadDataResourceBytes(id); |
| |
| return data; |
| } |
| |
| void ThemeService::SetTheme(const Extension* extension) { |
| // Clear our image cache. |
| FreePlatformCaches(); |
| |
| DCHECK(extension); |
| DCHECK(extension->is_theme()); |
| |
| BuildFromExtension(extension); |
| SaveThemeID(extension->id()); |
| |
| NotifyThemeChanged(); |
| UserMetrics::RecordAction(UserMetricsAction("Themes_Installed"), profile_); |
| } |
| |
| void ThemeService::RemoveUnusedThemes() { |
| if (!profile_) |
| return; |
| ExtensionService* service = profile_->GetExtensionService(); |
| if (!service) |
| return; |
| std::string current_theme = GetThemeID(); |
| std::vector<std::string> remove_list; |
| const ExtensionList* extensions = service->extensions(); |
| for (ExtensionList::const_iterator it = extensions->begin(); |
| it != extensions->end(); ++it) { |
| if ((*it)->is_theme() && (*it)->id() != current_theme) { |
| remove_list.push_back((*it)->id()); |
| } |
| } |
| for (size_t i = 0; i < remove_list.size(); ++i) |
| service->UninstallExtension(remove_list[i], false, NULL); |
| } |
| |
| void ThemeService::UseDefaultTheme() { |
| ClearAllThemeData(); |
| NotifyThemeChanged(); |
| UserMetrics::RecordAction(UserMetricsAction("Themes_Reset"), profile_); |
| } |
| |
| void ThemeService::SetNativeTheme() { |
| UseDefaultTheme(); |
| } |
| |
| bool ThemeService::UsingDefaultTheme() { |
| std::string id = GetThemeID(); |
| return id == ThemeService::kDefaultThemeID || |
| id == kDefaultThemeGalleryID; |
| } |
| |
| std::string ThemeService::GetThemeID() const { |
| return profile_->GetPrefs()->GetString(prefs::kCurrentThemeID); |
| } |
| |
| // static |
| std::string ThemeService::AlignmentToString(int alignment) { |
| // Convert from an AlignmentProperty back into a string. |
| std::string vertical_string; |
| std::string horizontal_string; |
| |
| if (alignment & ThemeService::ALIGN_TOP) |
| vertical_string = kAlignmentTop; |
| else if (alignment & ThemeService::ALIGN_BOTTOM) |
| vertical_string = kAlignmentBottom; |
| |
| if (alignment & ThemeService::ALIGN_LEFT) |
| horizontal_string = kAlignmentLeft; |
| else if (alignment & ThemeService::ALIGN_RIGHT) |
| horizontal_string = kAlignmentRight; |
| |
| if (vertical_string.empty()) |
| return horizontal_string; |
| if (horizontal_string.empty()) |
| return vertical_string; |
| return vertical_string + " " + horizontal_string; |
| } |
| |
| // static |
| int ThemeService::StringToAlignment(const std::string& alignment) { |
| std::vector<std::wstring> split; |
| base::SplitStringAlongWhitespace(UTF8ToWide(alignment), &split); |
| |
| int alignment_mask = 0; |
| for (std::vector<std::wstring>::iterator alignments(split.begin()); |
| alignments != split.end(); ++alignments) { |
| std::string comp = WideToUTF8(*alignments); |
| const char* component = comp.c_str(); |
| |
| if (base::strcasecmp(component, kAlignmentTop) == 0) |
| alignment_mask |= ThemeService::ALIGN_TOP; |
| else if (base::strcasecmp(component, kAlignmentBottom) == 0) |
| alignment_mask |= ThemeService::ALIGN_BOTTOM; |
| |
| if (base::strcasecmp(component, kAlignmentLeft) == 0) |
| alignment_mask |= ThemeService::ALIGN_LEFT; |
| else if (base::strcasecmp(component, kAlignmentRight) == 0) |
| alignment_mask |= ThemeService::ALIGN_RIGHT; |
| } |
| return alignment_mask; |
| } |
| |
| // static |
| std::string ThemeService::TilingToString(int tiling) { |
| // Convert from a TilingProperty back into a string. |
| if (tiling == ThemeService::REPEAT_X) |
| return kTilingRepeatX; |
| if (tiling == ThemeService::REPEAT_Y) |
| return kTilingRepeatY; |
| if (tiling == ThemeService::REPEAT) |
| return kTilingRepeat; |
| return kTilingNoRepeat; |
| } |
| |
| // static |
| int ThemeService::StringToTiling(const std::string& tiling) { |
| const char* component = tiling.c_str(); |
| |
| if (base::strcasecmp(component, kTilingRepeatX) == 0) |
| return ThemeService::REPEAT_X; |
| if (base::strcasecmp(component, kTilingRepeatY) == 0) |
| return ThemeService::REPEAT_Y; |
| if (base::strcasecmp(component, kTilingRepeat) == 0) |
| return ThemeService::REPEAT; |
| // NO_REPEAT is the default choice. |
| return ThemeService::NO_REPEAT; |
| } |
| |
| // static |
| color_utils::HSL ThemeService::GetDefaultTint(int id) { |
| switch (id) { |
| case TINT_FRAME: |
| return kDefaultTintFrame; |
| case TINT_FRAME_INACTIVE: |
| return kDefaultTintFrameInactive; |
| case TINT_FRAME_INCOGNITO: |
| return kDefaultTintFrameIncognito; |
| case TINT_FRAME_INCOGNITO_INACTIVE: |
| return kDefaultTintFrameIncognitoInactive; |
| case TINT_BUTTONS: |
| return kDefaultTintButtons; |
| case TINT_BACKGROUND_TAB: |
| return kDefaultTintBackgroundTab; |
| default: |
| color_utils::HSL result = {-1, -1, -1}; |
| return result; |
| } |
| } |
| |
| // static |
| SkColor ThemeService::GetDefaultColor(int id) { |
| switch (id) { |
| case COLOR_FRAME: |
| return kDefaultColorFrame; |
| case COLOR_FRAME_INACTIVE: |
| return kDefaultColorFrameInactive; |
| case COLOR_FRAME_INCOGNITO: |
| return kDefaultColorFrameIncognito; |
| case COLOR_FRAME_INCOGNITO_INACTIVE: |
| return kDefaultColorFrameIncognitoInactive; |
| case COLOR_TOOLBAR: |
| return kDefaultColorToolbar; |
| case COLOR_TAB_TEXT: |
| return kDefaultColorTabText; |
| case COLOR_BACKGROUND_TAB_TEXT: |
| return kDefaultColorBackgroundTabText; |
| case COLOR_BOOKMARK_TEXT: |
| return kDefaultColorBookmarkText; |
| case COLOR_NTP_BACKGROUND: |
| return kDefaultColorNTPBackground; |
| case COLOR_NTP_TEXT: |
| return kDefaultColorNTPText; |
| case COLOR_NTP_LINK: |
| return kDefaultColorNTPLink; |
| case COLOR_NTP_LINK_UNDERLINE: |
| return TintForUnderline(kDefaultColorNTPLink); |
| case COLOR_NTP_HEADER: |
| return kDefaultColorNTPHeader; |
| case COLOR_NTP_SECTION: |
| return kDefaultColorNTPSection; |
| case COLOR_NTP_SECTION_TEXT: |
| return kDefaultColorNTPSectionText; |
| case COLOR_NTP_SECTION_LINK: |
| return kDefaultColorNTPSectionLink; |
| case COLOR_NTP_SECTION_LINK_UNDERLINE: |
| return TintForUnderline(kDefaultColorNTPSectionLink); |
| case COLOR_CONTROL_BACKGROUND: |
| return kDefaultColorControlBackground; |
| case COLOR_BUTTON_BACKGROUND: |
| return kDefaultColorButtonBackground; |
| #if defined(OS_MACOSX) |
| case COLOR_TOOLBAR_BUTTON_STROKE: |
| return kDefaultColorToolbarButtonStroke; |
| case COLOR_TOOLBAR_BUTTON_STROKE_INACTIVE: |
| return kDefaultColorToolbarButtonStrokeInactive; |
| case COLOR_TOOLBAR_BEZEL: |
| return kDefaultColorToolbarBezel; |
| case COLOR_TOOLBAR_STROKE: |
| return kDefaultColorToolbarStroke; |
| case COLOR_TOOLBAR_STROKE_INACTIVE: |
| return kDefaultColorToolbarStrokeInactive; |
| #endif |
| default: |
| // Return a debugging red color. |
| return 0xffff0000; |
| } |
| } |
| |
| // static |
| bool ThemeService::GetDefaultDisplayProperty(int id, int* result) { |
| switch (id) { |
| case NTP_BACKGROUND_ALIGNMENT: |
| *result = kDefaultDisplayPropertyNTPAlignment; |
| return true; |
| case NTP_BACKGROUND_TILING: |
| *result = kDefaultDisplayPropertyNTPTiling; |
| return true; |
| case NTP_LOGO_ALTERNATE: |
| *result = kDefaultDisplayPropertyNTPInverseLogo; |
| return true; |
| } |
| |
| return false; |
| } |
| |
| // static |
| const std::set<int>& ThemeService::GetTintableToolbarButtons() { |
| static std::set<int> button_set; |
| if (button_set.empty()) { |
| button_set = std::set<int>( |
| kToolbarButtonIDs, |
| kToolbarButtonIDs + arraysize(kToolbarButtonIDs)); |
| } |
| |
| return button_set; |
| } |
| |
| color_utils::HSL ThemeService::GetTint(int id) const { |
| DCHECK(CalledOnValidThread()); |
| |
| color_utils::HSL hsl; |
| if (theme_pack_.get() && theme_pack_->GetTint(id, &hsl)) |
| return hsl; |
| |
| return GetDefaultTint(id); |
| } |
| |
| void ThemeService::ClearAllThemeData() { |
| // Clear our image cache. |
| FreePlatformCaches(); |
| theme_pack_ = NULL; |
| |
| profile_->GetPrefs()->ClearPref(prefs::kCurrentThemePackFilename); |
| SaveThemeID(kDefaultThemeID); |
| } |
| |
| void ThemeService::LoadThemePrefs() { |
| PrefService* prefs = profile_->GetPrefs(); |
| |
| std::string current_id = GetThemeID(); |
| if (current_id != kDefaultThemeID) { |
| bool loaded_pack = false; |
| |
| // If we don't have a file pack, we're updating from an old version. |
| FilePath path = prefs->GetFilePath(prefs::kCurrentThemePackFilename); |
| if (path != FilePath()) { |
| theme_pack_ = BrowserThemePack::BuildFromDataPack(path, current_id); |
| loaded_pack = theme_pack_.get() != NULL; |
| } |
| |
| if (loaded_pack) { |
| UserMetrics::RecordAction(UserMetricsAction("Themes.Loaded"), profile_); |
| } else { |
| // TODO(erg): We need to pop up a dialog informing the user that their |
| // theme is being migrated. |
| ExtensionService* service = profile_->GetExtensionService(); |
| if (service) { |
| const Extension* extension = |
| service->GetExtensionById(current_id, false); |
| if (extension) { |
| DLOG(ERROR) << "Migrating theme"; |
| BuildFromExtension(extension); |
| UserMetrics::RecordAction(UserMetricsAction("Themes.Migrated"), |
| profile_); |
| } else { |
| DLOG(ERROR) << "Theme is mysteriously gone."; |
| ClearAllThemeData(); |
| UserMetrics::RecordAction(UserMetricsAction("Themes.Gone"), profile_); |
| } |
| } |
| } |
| } |
| } |
| |
| void ThemeService::NotifyThemeChanged() { |
| VLOG(1) << "Sending BROWSER_THEME_CHANGED"; |
| // Redraw! |
| NotificationService* service = NotificationService::current(); |
| service->Notify(NotificationType::BROWSER_THEME_CHANGED, |
| Source<ThemeService>(this), |
| NotificationService::NoDetails()); |
| #if defined(OS_MACOSX) |
| NotifyPlatformThemeChanged(); |
| #endif // OS_MACOSX |
| } |
| |
| #if defined(OS_WIN) |
| void ThemeService::FreePlatformCaches() { |
| // Views (Skia) has no platform image cache to clear. |
| } |
| #endif |
| |
| void ThemeService::Observe(NotificationType type, |
| const NotificationSource& source, |
| const NotificationDetails& details) { |
| DCHECK(type == NotificationType::EXTENSION_LOADED); |
| const Extension* extension = Details<const Extension>(details).ptr(); |
| if (!extension->is_theme()) { |
| return; |
| } |
| SetTheme(extension); |
| } |
| |
| void ThemeService::SavePackName(const FilePath& pack_path) { |
| profile_->GetPrefs()->SetFilePath( |
| prefs::kCurrentThemePackFilename, pack_path); |
| } |
| |
| void ThemeService::SaveThemeID(const std::string& id) { |
| profile_->GetPrefs()->SetString(prefs::kCurrentThemeID, id); |
| } |
| |
| void ThemeService::BuildFromExtension(const Extension* extension) { |
| scoped_refptr<BrowserThemePack> pack( |
| BrowserThemePack::BuildFromExtension(extension)); |
| if (!pack.get()) { |
| // TODO(erg): We've failed to install the theme; perhaps we should tell the |
| // user? http://crbug.com/34780 |
| LOG(ERROR) << "Could not load theme."; |
| return; |
| } |
| |
| // Write the packed file to disk. |
| FilePath pack_path = extension->path().Append(chrome::kThemePackFilename); |
| BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE, |
| new WritePackToDiskTask(pack, pack_path)); |
| |
| SavePackName(pack_path); |
| theme_pack_ = pack; |
| } |
| |
| void ThemeService::OnInfobarDisplayed() { |
| number_of_infobars_++; |
| } |
| |
| void ThemeService::OnInfobarDestroyed() { |
| number_of_infobars_--; |
| |
| if (number_of_infobars_ == 0) |
| RemoveUnusedThemes(); |
| } |