summaryrefslogtreecommitdiffstats
path: root/chrome/browser
diff options
context:
space:
mode:
authorglen@chromium.org <glen@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-05-09 01:07:42 +0000
committerglen@chromium.org <glen@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-05-09 01:07:42 +0000
commit4a19063d1459a4c9c8c4c50ed86eb9048f69ea3f (patch)
tree69a1e9f78b3a5fa8b909cfab336826b09c44235f /chrome/browser
parent5085ee0b4bfbe4625e63ee6975bb95702e13e0aa (diff)
downloadchromium_src-4a19063d1459a4c9c8c4c50ed86eb9048f69ea3f.zip
chromium_src-4a19063d1459a4c9c8c4c50ed86eb9048f69ea3f.tar.gz
chromium_src-4a19063d1459a4c9c8c4c50ed86eb9048f69ea3f.tar.bz2
This is the first pass at themes.
This CL is paired with http://codereview.chromium.org/67284 This CL (for commit purposes) includes http://codereview.chromium.org/67284 BUG=4463,11232,11233,11234,11235 Review URL: http://codereview.chromium.org/99030 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@15704 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome/browser')
-rw-r--r--chrome/browser/browser.vcproj16
-rw-r--r--chrome/browser/browser_init.cc1
-rw-r--r--chrome/browser/browser_theme_provider.cc527
-rw-r--r--chrome/browser/browser_theme_provider.h125
-rw-r--r--chrome/browser/browser_window.h4
-rw-r--r--chrome/browser/dom_ui/dom_ui.cc7
-rw-r--r--chrome/browser/dom_ui/dom_ui.h5
-rw-r--r--chrome/browser/dom_ui/dom_ui_theme_source.cc37
-rw-r--r--chrome/browser/dom_ui/dom_ui_theme_source.h30
-rw-r--r--chrome/browser/dom_ui/new_tab_ui.cc6
-rw-r--r--chrome/browser/extensions/extension.cc117
-rw-r--r--chrome/browser/extensions/extension.h32
-rw-r--r--chrome/browser/extensions/extensions_service.cc53
-rw-r--r--chrome/browser/extensions/extensions_service.h11
-rw-r--r--chrome/browser/extensions/extensions_service_unittest.cc77
-rw-r--r--chrome/browser/extensions/extensions_ui.cc2
-rw-r--r--chrome/browser/external_tab_container.h1
-rw-r--r--chrome/browser/profile.cc66
-rw-r--r--chrome/browser/profile.h21
-rw-r--r--chrome/browser/resources/new_tab.html2
-rw-r--r--chrome/browser/views/bookmark_bar_view.cc48
-rw-r--r--chrome/browser/views/constrained_window_impl.cc92
-rw-r--r--chrome/browser/views/frame/browser_frame.cc15
-rw-r--r--chrome/browser/views/frame/browser_frame.h9
-rw-r--r--chrome/browser/views/frame/browser_view.cc7
-rw-r--r--chrome/browser/views/frame/browser_view.h1
-rw-r--r--chrome/browser/views/frame/glass_browser_frame_view.cc101
-rw-r--r--chrome/browser/views/frame/glass_browser_frame_view.h2
-rw-r--r--chrome/browser/views/frame/opaque_browser_frame_view.cc412
-rw-r--r--chrome/browser/views/frame/opaque_browser_frame_view.h15
-rw-r--r--chrome/browser/views/infobars/infobars.cc3
-rw-r--r--chrome/browser/views/location_bar_view.cc2
-rw-r--r--chrome/browser/views/options/content_page_view.cc75
-rw-r--r--chrome/browser/views/options/content_page_view.h17
-rw-r--r--chrome/browser/views/options/cookies_view.cc1
-rw-r--r--chrome/browser/views/tabs/tab_renderer.cc196
-rw-r--r--chrome/browser/views/tabs/tab_renderer.h20
-rw-r--r--chrome/browser/views/toolbar_view.cc199
-rw-r--r--chrome/browser/views/toolbar_view.h4
39 files changed, 1674 insertions, 685 deletions
diff --git a/chrome/browser/browser.vcproj b/chrome/browser/browser.vcproj
index 91e7a36..7642876 100644
--- a/chrome/browser/browser.vcproj
+++ b/chrome/browser/browser.vcproj
@@ -290,6 +290,14 @@
>
</File>
<File
+ RelativePath=".\browser_theme_provider.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\browser_theme_provider.h"
+ >
+ </File>
+ <File
RelativePath=".\browser_url_handler.cc"
>
</File>
@@ -1542,6 +1550,14 @@
>
</File>
<File
+ RelativePath=".\dom_ui\dom_ui_theme_source.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\dom_ui\dom_ui_theme_source.h"
+ >
+ </File>
+ <File
RelativePath=".\dom_ui\dom_ui_thumbnail_source.cc"
>
</File>
diff --git a/chrome/browser/browser_init.cc b/chrome/browser/browser_init.cc
index fa6a00a..5488ef6 100644
--- a/chrome/browser/browser_init.cc
+++ b/chrome/browser/browser_init.cc
@@ -20,6 +20,7 @@
#include "chrome/browser/browser_list.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/extensions/extensions_service.h"
+#include "chrome/browser/extensions/user_script_master.h"
#include "chrome/browser/first_run.h"
#include "chrome/browser/net/dns_global.h"
#include "chrome/browser/profile.h"
diff --git a/chrome/browser/browser_theme_provider.cc b/chrome/browser/browser_theme_provider.cc
new file mode 100644
index 0000000..cb68ed0c
--- /dev/null
+++ b/chrome/browser/browser_theme_provider.cc
@@ -0,0 +1,527 @@
+// Copyright (c) 2009 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 "browser_theme_provider.h"
+
+#include "base/gfx/png_decoder.h"
+#include "base/string_util.h"
+#include "base/values.h"
+#include "chrome/browser/browser_list.h"
+#include "chrome/browser/extensions/extension.h"
+#include "chrome/browser/metrics/user_metrics.h"
+#include "chrome/browser/profile.h"
+#include "chrome/browser/theme_resources_util.h"
+#include "chrome/common/pref_names.h"
+#include "chrome/common/pref_service.h"
+#include "grit/theme_resources.h"
+#include "net/base/file_stream.h"
+#include "net/base/net_errors.h"
+#include "skia/ext/image_operations.h"
+#include "skia/ext/skia_utils.h"
+#include "SkBitmap.h"
+
+// Strings used by themes to identify colors for different parts of our UI.
+static const std::string kColorFrame = "frame";
+static const std::string kColorFrameInactive = "frame_inactive";
+static const std::string kColorFrameIncognito = "frame_incognito";
+static const std::string kColorFrameIncognitoInactive =
+ "frame_incognito_inactive";
+static const std::string kColorToolbar = "toolbar";
+static const std::string kColorTabText = "tab_text";
+static const std::string kColorBackgroundTabText = "background_tab_text";
+static const std::string kColorBookmarkText = "bookmark_text";
+static const std::string kColorNTPText = "ntp_text";
+static const std::string kColorNTPLink = "ntp_link";
+static const std::string kColorNTPSection = "ntp_section";
+
+// Strings used by themes to identify tints to apply to different parts of
+// our UI. The frame tints apply to the frame color and produce the
+// COLOR_FRAME* colors.
+static const std::string kTintButtons = "buttons";
+static const std::string kTintFrame = "frame";
+static const std::string kTintFrameInactive = "frame_inactive";
+static const std::string kTintFrameIncognito = "frame_incognito";
+static const std::string kTintFrameIncognitoInactive =
+ "frame_incognito_inactive";
+static const std::string kTintBackgroundTab = "background_tab";
+
+// Default colors.
+static const SkColor kDefaultColorFrame = SkColorSetRGB(77, 139, 217);
+static const SkColor kDefaultColorFrameInactive = SkColorSetRGB(152, 188, 233);
+static const SkColor kDefaultColorFrameIncognito = SkColorSetRGB(83, 106, 139);
+static const SkColor kDefaultColorFrameIncognitoInactive =
+ SkColorSetRGB(126, 139, 156);
+static const SkColor kDefaultColorToolbar = SkColorSetRGB(210, 225, 246);
+static const SkColor kDefaultColorTabText = SkColorSetRGB(0, 0, 0);
+static const SkColor kDefaultColorBackgroundTabText = SkColorSetRGB(64, 64, 64);
+
+// Default tints.
+static const skia::HSL kDefaultTintButtons = { -1, -1, -1 };
+static const skia::HSL kDefaultTintFrame = { -1, -1, -1 };
+static const skia::HSL kDefaultTintFrameInactive = { -1, 0.5f, 0.72f };
+static const skia::HSL kDefaultTintFrameIncognito = { -1, 0.2f, 0.35f };
+static const skia::HSL kDefaultTintFrameIncognitoInactive = { -1, 0.3f, 0.6f };
+static const skia::HSL kDefaultTintBackgroundTab = { -1, 0.5, 0.75 };
+
+// The image resources that will be tinted by the 'button' tint value.
+static 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_RELOAD, IDR_RELOAD_H, IDR_RELOAD_P,
+ IDR_HOME, IDR_HOME_H, IDR_HOME_P,
+ IDR_STAR, IDR_STAR_D, IDR_STAR_H, IDR_STAR_P,
+ IDR_STARRED, IDR_STARRED_H, IDR_STARRED_P,
+ IDR_GO, IDR_GO_H, IDR_GO_P,
+ IDR_STOP, IDR_STOP_H, IDR_STOP_P,
+ IDR_MENU_PAGE, IDR_MENU_PAGE_RTL,
+ IDR_MENU_CHROME, IDR_MENU_CHROME_RTL,
+ IDR_MENU_DROPARROW,
+ IDR_THROBBER,
+};
+
+// A map for kToolbarButtonIDs.
+static std::map<const int, bool> button_images_;
+
+// A map of frame image IDs to the tints for those ids.
+static std::map<const int, int> frame_tints_;
+
+BrowserThemeProvider::BrowserThemeProvider()
+ : rb_(ResourceBundle::GetSharedInstance()) {
+ static bool initialized = false;
+ if (!initialized) {
+ for (int i = 0; i < sizeof(kToolbarButtonIDs); ++i) {
+ button_images_[kToolbarButtonIDs[i]] = true;
+ }
+ frame_tints_[IDR_THEME_FRAME] = TINT_FRAME;
+ frame_tints_[IDR_THEME_FRAME_INACTIVE] = TINT_FRAME_INACTIVE;
+ frame_tints_[IDR_THEME_FRAME_INCOGNITO] = TINT_FRAME_INCOGNITO;
+ frame_tints_[IDR_THEME_FRAME_INCOGNITO_INACTIVE] =
+ TINT_FRAME_INCOGNITO_INACTIVE;
+ }
+}
+
+BrowserThemeProvider::~BrowserThemeProvider() { }
+
+void BrowserThemeProvider::Init(Profile* profile) {
+ profile_ = profile;
+ LoadThemePrefs();
+}
+
+SkBitmap* BrowserThemeProvider::GetBitmapNamed(int id) {
+ // Check to see if we already have the Skia image in the cache.
+ ImageCache::const_iterator found = image_cache_.find(id);
+ if (found != image_cache_.end())
+ return found->second;
+
+ scoped_ptr<SkBitmap> result;
+
+ // Try to load the image from the extension.
+ result.reset(LoadThemeBitmap(id));
+
+ // If the extension doesn't provide the requested image, but has provided
+ // a custom frame, then we may be able to generate the image required.
+ if (!result.get())
+ result.reset(GenerateBitmap(id));
+
+ // If we still don't have an image, load it from resourcebundle.
+ if (!result.get())
+ result.reset(rb_.GetBitmapNamed(id));
+
+ // If the requested image is part of the toolbar button set, and we have
+ // a provided tint for that set, tint it appropriately.
+ if (button_images_.count(id) && tints_.count(kTintButtons)) {
+ SkBitmap* tinted =
+ new SkBitmap(TintBitmap(*result.release(), TINT_BUTTONS));
+ result.reset(tinted);
+ }
+
+ if (result.get()) {
+ // We loaded successfully. Cache the bitmap.
+ image_cache_[id] = result.get();
+ return result.release();
+ } else {
+ NOTREACHED() << "Failed to load a requested image";
+ return NULL;
+ }
+}
+
+SkColor BrowserThemeProvider::GetColor(int id) {
+ // TODO(glen): Figure out if we need to tint these. http://crbug.com/11578
+ switch (id) {
+ case COLOR_FRAME:
+ return (colors_.find(kColorFrame) != colors_.end()) ?
+ colors_[kColorFrame] : kDefaultColorFrame;
+ case COLOR_FRAME_INACTIVE:
+ return (colors_.find(kColorFrameInactive) != colors_.end()) ?
+ colors_[kColorFrameInactive] : kDefaultColorFrameInactive;
+ case COLOR_FRAME_INCOGNITO:
+ return (colors_.find(kColorFrameIncognito) != colors_.end()) ?
+ colors_[kColorFrameIncognito] : kDefaultColorFrameIncognito;
+ case COLOR_FRAME_INCOGNITO_INACTIVE:
+ return (colors_.find(kColorFrameIncognitoInactive) != colors_.end()) ?
+ colors_[kColorFrameIncognitoInactive] :
+ kDefaultColorFrameIncognitoInactive;
+ case COLOR_TOOLBAR:
+ return (colors_.find(kColorToolbar) != colors_.end()) ?
+ colors_[kColorToolbar] : kDefaultColorToolbar;
+ case COLOR_TAB_TEXT:
+ return (colors_.find(kColorTabText) != colors_.end()) ?
+ colors_[kColorTabText] : kDefaultColorTabText;
+ case COLOR_BACKGROUND_TAB_TEXT:
+ return (colors_.find(kColorBackgroundTabText) != colors_.end()) ?
+ colors_[kColorBackgroundTabText] :
+ kDefaultColorBackgroundTabText;
+ default:
+ NOTREACHED() << "Unknown color requested";
+ }
+
+ // Return a debugging red color.
+ return 0xffff0000;
+}
+
+void BrowserThemeProvider::SetTheme(Extension* extension) {
+ // Clear our image cache.
+ image_cache_.clear();
+
+ DCHECK(extension);
+ DCHECK(extension->IsTheme());
+ SetImageData(extension->GetThemeImages(),
+ extension->path());
+ SetColorData(extension->GetThemeColors());
+ SetTintData(extension->GetThemeTints());
+ GenerateFrameColors();
+ GenerateFrameImages();
+
+ SaveImageData(extension->GetThemeImages());
+ SaveColorData();
+ SaveTintData();
+
+ NotifyThemeChanged();
+ UserMetrics::RecordAction(L"Themes_Installed", profile_);
+}
+
+void BrowserThemeProvider::UseDefaultTheme() {
+ // Clear our image cache.
+ image_cache_.clear();
+
+ images_.clear();
+ colors_.clear();
+ tints_.clear();
+
+ SaveImageData(NULL);
+ SaveColorData();
+ SaveTintData();
+
+ NotifyThemeChanged();
+}
+
+SkBitmap* BrowserThemeProvider::LoadThemeBitmap(int id) {
+ // Attempt to find the image in our theme bundle.
+ std::vector<unsigned char> raw_data, png_data;
+ if (images_.count(id)) {
+ // First check to see if we have a registered theme extension and whether
+ // it can handle this resource.
+ FilePath path = FilePath(UTF8ToWide(images_[id]));
+ if (!path.empty()) {
+ net::FileStream file;
+ int flags = base::PLATFORM_FILE_OPEN | base::PLATFORM_FILE_READ;
+ if (file.Open(path, flags) == net::OK) {
+ int64 avail = file.Available();
+ if (avail > 0 && avail < INT_MAX) {
+ size_t size = static_cast<size_t>(avail);
+ raw_data.resize(size);
+ char* data = reinterpret_cast<char*>(&(raw_data.front()));
+ if (file.ReadUntilComplete(data, size) == avail) {
+ // Decode the PNG.
+ int image_width = 0;
+ int image_height = 0;
+
+ if (!PNGDecoder::Decode(&raw_data.front(), raw_data.size(),
+ PNGDecoder::FORMAT_BGRA, &png_data,
+ &image_width, &image_height)) {
+ NOTREACHED() << "Unable to decode theme image resource " << id;
+ return NULL;
+ }
+
+ return PNGDecoder::CreateSkBitmapFromBGRAFormat(png_data,
+ image_width,
+ image_height);
+ }
+ }
+ } else {
+ // TODO(glen): File no-longer exists, we're out of date. We should
+ // clear the theme (or maybe just the pref that points to this
+ // image).
+ return NULL;
+ }
+ }
+ }
+
+ return NULL;
+}
+
+skia::HSL BrowserThemeProvider::GetTint(int id) {
+ switch (id) {
+ case TINT_FRAME:
+ return (tints_.find(kTintFrame) != tints_.end()) ?
+ tints_[kTintFrame] : kDefaultTintFrame;
+ case TINT_FRAME_INACTIVE:
+ return (tints_.find(kTintFrameInactive) != tints_.end()) ?
+ tints_[kTintFrameInactive] : kDefaultTintFrameInactive;
+ case TINT_FRAME_INCOGNITO:
+ return (tints_.count(kTintFrameIncognito)) ?
+ tints_[kTintFrameIncognito] : kDefaultTintFrameIncognito;
+ case TINT_FRAME_INCOGNITO_INACTIVE:
+ return (tints_.count(kTintFrameIncognitoInactive)) ?
+ tints_[kTintFrameIncognitoInactive] :
+ kDefaultTintFrameIncognitoInactive;
+ case TINT_BUTTONS:
+ return (tints_.find(kTintButtons) != tints_.end()) ?
+ tints_[kTintButtons] :
+ kDefaultTintButtons;
+ case TINT_BACKGROUND_TAB:
+ return (tints_.find(kTintBackgroundTab) != tints_.end()) ?
+ tints_[kTintBackgroundTab] :
+ kDefaultTintBackgroundTab;
+ default:
+ NOTREACHED() << "Unknown tint requested";
+ }
+ skia::HSL result = {-1, -1, -1};
+ return result;
+}
+
+SkBitmap BrowserThemeProvider::TintBitmap(const SkBitmap& bitmap, int hsl_id) {
+ return skia::ImageOperations::CreateHSLShiftedBitmap(bitmap, GetTint(hsl_id));
+}
+
+void BrowserThemeProvider::SetImageData(DictionaryValue* images_value,
+ FilePath images_path) {
+ images_.clear();
+
+ if (images_value) {
+ DictionaryValue::key_iterator iter = images_value->begin_keys();
+ while (iter != images_value->end_keys()) {
+ std::string val;
+ if (images_value->GetString(*iter, &val)) {
+ int id = ThemeResourcesUtil::GetId(WideToUTF8(*iter));
+ if (id != -1) {
+ if (!images_path.empty()) {
+ images_[id] = WideToUTF8(images_path.AppendASCII(val)
+ .ToWStringHack());
+ } else {
+ images_[id] = val;
+ }
+ }
+ }
+ ++iter;
+ }
+ }
+}
+
+void BrowserThemeProvider::SetColorData(DictionaryValue* colors_value) {
+ colors_.clear();
+
+ if (colors_value) {
+ DictionaryValue::key_iterator iter = colors_value->begin_keys();
+ while (iter != colors_value->end_keys()) {
+ ListValue* color_list;
+ if (colors_value->GetList(*iter, &color_list) &&
+ color_list->GetSize() == 3) {
+ int r, g, b;
+ color_list->GetInteger(0, &r);
+ color_list->GetInteger(1, &g);
+ color_list->GetInteger(2, &b);
+ colors_[WideToUTF8(*iter)] = SkColorSetRGB(r, g, b);
+ }
+ ++iter;
+ }
+ }
+}
+
+void BrowserThemeProvider::SetTintData(DictionaryValue* tints_value) {
+ tints_.clear();
+
+ if (tints_value) {
+ DictionaryValue::key_iterator iter = tints_value->begin_keys();
+ while (iter != tints_value->end_keys()) {
+ ListValue* tint_list;
+ if (tints_value->GetList(*iter, &tint_list) &&
+ tint_list->GetSize() == 3) {
+ skia::HSL hsl = { -1, -1, -1 };
+ // TODO(glen): Make this work with integer values.
+ tint_list->GetReal(0, &hsl.h);
+ tint_list->GetReal(1, &hsl.s);
+ tint_list->GetReal(2, &hsl.l);
+ tints_[WideToUTF8(*iter)] = hsl;
+ }
+ ++iter;
+ }
+ }
+}
+
+void BrowserThemeProvider::GenerateFrameColors() {
+ // Generate any secondary frame colors that weren't provided.
+ skia::HSL frame_hsl = { 0, 0, 0 };
+ skia::SkColorToHSL(GetColor(COLOR_FRAME), frame_hsl);
+
+ if (colors_.find(kColorFrame) == colors_.end())
+ colors_[kColorFrame] = HSLShift(frame_hsl, GetTint(TINT_FRAME));
+ if (colors_.find(kColorFrameInactive) == colors_.end()) {
+ colors_[kColorFrameInactive] =
+ HSLShift(frame_hsl, GetTint(TINT_FRAME_INACTIVE));
+ }
+ if (colors_.find(kColorFrameIncognito) == colors_.end()) {
+ colors_[kColorFrameIncognito] =
+ HSLShift(frame_hsl, GetTint(TINT_FRAME_INCOGNITO));
+ }
+ if (colors_.find(kColorFrameIncognitoInactive) == colors_.end()) {
+ colors_[kColorFrameIncognitoInactive] =
+ HSLShift(frame_hsl, GetTint(TINT_FRAME_INCOGNITO_INACTIVE));
+ }
+}
+
+void BrowserThemeProvider::GenerateFrameImages() {
+ std::map<const int, int>::iterator iter = frame_tints_.begin();
+ while (iter != frame_tints_.end()) {
+ int id = iter->first;
+ SkBitmap* frame;
+ // If there's no frame image provided for the specified id, then load
+ // the default provided frame. If that's not provided, skip this whole
+ // thing and just use the default images.
+ int base_id = (id == IDR_THEME_FRAME_INCOGNITO ||
+ id == IDR_THEME_FRAME_INCOGNITO_INACTIVE) ?
+ IDR_THEME_FRAME_INCOGNITO : IDR_THEME_FRAME;
+
+ if (images_.find(id) != images_.end()) {
+ frame = LoadThemeBitmap(id);
+ } else if (base_id != id && images_.find(base_id) != images_.end()) {
+ frame = LoadThemeBitmap(base_id);
+ } else {
+ if (id == IDR_THEME_FRAME_INCOGNITO ||
+ id == IDR_THEME_FRAME_INCOGNITO_INACTIVE)
+ frame = rb_.GetBitmapNamed(IDR_THEME_FRAME_INCOGNITO);
+ else
+ frame = rb_.GetBitmapNamed(IDR_THEME_FRAME);
+ }
+
+ if (frame) {
+ SkBitmap* tinted = new SkBitmap(TintBitmap(*frame, frame_tints_[id]));
+ image_cache_[id] = tinted;
+ }
+ ++iter;
+ }
+}
+
+SkBitmap* BrowserThemeProvider::GenerateBitmap(int id) {
+ if (id == IDR_THEME_TAB_BACKGROUND ||
+ id == IDR_THEME_TAB_BACKGROUND_INCOGNITO) {
+ // The requested image is a background tab. Get a frame to create the
+ // tab against. As themes don't use the glass frame, we don't have to
+ // worry about compositing them together, as our default theme provides
+ // the necessary bitmaps.
+ SkBitmap* frame;
+ if (id == IDR_THEME_TAB_BACKGROUND)
+ frame = image_cache_.find(IDR_THEME_FRAME)->second;
+ else
+ frame = image_cache_.find(IDR_THEME_FRAME_INCOGNITO)->second;
+
+ if (frame) {
+ SkBitmap blurred =
+ skia::ImageOperations::CreateBlurredBitmap(*frame, 5);
+ SkBitmap* bg_tab =
+ new SkBitmap(TintBitmap(blurred, TINT_BACKGROUND_TAB));
+ return bg_tab;
+ }
+ }
+
+ return NULL;
+}
+
+void BrowserThemeProvider::NotifyThemeChanged() {
+ // TODO(glen): If we're in glass and IDR_THEME_FRAME has been provided,
+ // swap us back to opaque frame.
+
+ // Redraw!
+ for (BrowserList::const_iterator browser = BrowserList::begin();
+ browser != BrowserList::end(); ++browser) {
+ (*browser)->window()->UserChangedTheme();
+ }
+}
+
+void BrowserThemeProvider::LoadThemePrefs() {
+ PrefService* prefs = profile_->GetPrefs();
+
+ // Our prefs already have the extension path baked in, so we don't need
+ // to provide it.
+ SetImageData(prefs->GetMutableDictionary(prefs::kCurrentThemeImages),
+ FilePath());
+ SetColorData(prefs->GetMutableDictionary(prefs::kCurrentThemeColors));
+ SetTintData(prefs->GetMutableDictionary(prefs::kCurrentThemeTints));
+ GenerateFrameColors();
+ GenerateFrameImages();
+
+ // TODO(glen): Figure out if any custom prefs were loaded, and if so
+ // UMA-log the fact that a theme was loaded.
+ if (prefs->HasPrefPath(prefs::kCurrentThemeImages) ||
+ prefs->HasPrefPath(prefs::kCurrentThemeColors) ||
+ prefs->HasPrefPath(prefs::kCurrentThemeTints)) {
+ UserMetrics::RecordAction(L"Themes_loaded", profile_);
+ }
+}
+
+void BrowserThemeProvider::SaveImageData(DictionaryValue* images_value) {
+ // Save our images data.
+ DictionaryValue* pref_images =
+ profile_->GetPrefs()->GetMutableDictionary(prefs::kCurrentThemeImages);
+ pref_images->Clear();
+
+ DictionaryValue::key_iterator iter = images_value->begin_keys();
+ while (iter != images_value->end_keys()) {
+ std::string val;
+ if (images_value->GetString(*iter, &val)) {
+ int id = ThemeResourcesUtil::GetId(WideToUTF8(*iter));
+ if (id != -1)
+ pref_images->SetString(*iter, images_[id]);
+ }
+ ++iter;
+ }
+}
+
+void BrowserThemeProvider::SaveColorData() {
+ // Save our color data.
+ DictionaryValue* pref_colors =
+ profile_->GetPrefs()->GetMutableDictionary(prefs::kCurrentThemeColors);
+ pref_colors->Clear();
+ if (colors_.size()) {
+ ColorMap::iterator iter = colors_.begin();
+ while (iter != colors_.end()) {
+ SkColor rgb = (*iter).second;
+ ListValue* rgb_list = new ListValue();
+ rgb_list->Set(0, Value::CreateIntegerValue(SkColorGetR(rgb)));
+ rgb_list->Set(1, Value::CreateIntegerValue(SkColorGetG(rgb)));
+ rgb_list->Set(2, Value::CreateIntegerValue(SkColorGetB(rgb)));
+ pref_colors->Set(UTF8ToWide((*iter).first), rgb_list);
+ ++iter;
+ }
+ }
+}
+
+void BrowserThemeProvider::SaveTintData() {
+ // Save our tint data.
+ DictionaryValue* pref_tints =
+ profile_->GetPrefs()->GetMutableDictionary(prefs::kCurrentThemeTints);
+ pref_tints->Clear();
+ if (tints_.size()) {
+ TintMap::iterator iter = tints_.begin();
+ while (iter != tints_.end()) {
+ skia::HSL hsl = (*iter).second;
+ ListValue* hsl_list = new ListValue();
+ hsl_list->Set(0, Value::CreateRealValue(hsl.h));
+ hsl_list->Set(1, Value::CreateRealValue(hsl.s));
+ hsl_list->Set(2, Value::CreateRealValue(hsl.l));
+ pref_tints->Set(UTF8ToWide((*iter).first), hsl_list);
+ ++iter;
+ }
+ }
+}
diff --git a/chrome/browser/browser_theme_provider.h b/chrome/browser/browser_theme_provider.h
new file mode 100644
index 0000000..bf16a0d
--- /dev/null
+++ b/chrome/browser/browser_theme_provider.h
@@ -0,0 +1,125 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_BROWSER_THEME_PROVIDER_H_
+#define CHROME_BROWSER_THEME_PROVIDER_H_
+
+#include "app/resource_bundle.h"
+#include "app/theme_provider.h"
+#include "base/ref_counted.h"
+#include "skia/ext/skia_utils.h"
+
+class Extension;
+class Profile;
+class DictionaryValue;
+
+class BrowserThemeProvider :
+ public base::RefCountedThreadSafe<BrowserThemeProvider>,
+ public ThemeProvider {
+ public:
+ BrowserThemeProvider();
+ virtual ~BrowserThemeProvider();
+
+ enum {
+ COLOR_FRAME,
+ COLOR_FRAME_INACTIVE,
+ COLOR_FRAME_INCOGNITO,
+ COLOR_FRAME_INCOGNITO_INACTIVE,
+ COLOR_TOOLBAR,
+ COLOR_TAB_TEXT,
+ COLOR_BACKGROUND_TAB_TEXT,
+ COLOR_BOOKMARK_TEXT,
+ COLOR_NTP_TEXT,
+ COLOR_NTP_LINK,
+ COLOR_NTP_SECTION,
+ TINT_BUTTONS,
+ TINT_FRAME,
+ TINT_FRAME_INACTIVE,
+ TINT_FRAME_INCOGNITO,
+ TINT_FRAME_INCOGNITO_INACTIVE,
+ TINT_BACKGROUND_TAB
+ };
+
+ void Init(Profile* profile);
+
+ // ThemeProvider implementation.
+ virtual SkBitmap* GetBitmapNamed(int id);
+ virtual SkColor GetColor(int id);
+
+ // Set the current theme to the theme defined in |extension|.
+ void SetTheme(Extension* extension);
+
+ // Reset the theme to default.
+ void UseDefaultTheme();
+
+ private:
+ typedef std::map<const int, std::string> ImageMap;
+ typedef std::map<const std::string, SkColor> ColorMap;
+ typedef std::map<const std::string, skia::HSL> TintMap;
+
+ // Loads a bitmap from the theme, which may be tinted or
+ // otherwise modified, or an application default.
+ SkBitmap* LoadThemeBitmap(int id);
+
+ // Get the specified tint - |id| is one of the TINT_* enum values.
+ skia::HSL GetTint(int id);
+
+ // Tint |bitmap| with the tint specified by |hsl_id|
+ SkBitmap TintBitmap(const SkBitmap& bitmap, int hsl_id);
+
+ // The following load data from specified dictionaries (either from
+ // preferences or from an extension manifest) and update our theme
+ // data appropriately.
+ // Allow any ResourceBundle image to be overridden. |images| should
+ // contain keys defined in ThemeResourceMap, and values as paths to
+ // the images on-disk.
+ void SetImageData(DictionaryValue* images,
+ FilePath images_path);
+ // Set our theme colors. The keys of |colors| are any of the kColor*
+ // constants, and the values are a three-item list containing 8-bit
+ // RGB values.
+ void SetColorData(DictionaryValue* colors);
+ // Set tint data for our images and colors. The keys of |tints| are
+ // any of the kTint* contstants, and the values are a three-item list
+ // containing real numbers in the range 0-1 (and -1 for 'null').
+ void SetTintData(DictionaryValue* tints);
+
+ // Generate any frame colors that weren't specified.
+ void GenerateFrameColors();
+
+ // Generate any frame images that weren't specified. The resulting images
+ // will be stored in our cache.
+ void GenerateFrameImages();
+
+ // Create any images that aren't pregenerated (e.g. background tab images).
+ SkBitmap* GenerateBitmap(int id);
+
+ // Save our data - when saving images we need the original dictionary
+ // from the extension because it contains the text ids that we want to save.
+ void SaveImageData(DictionaryValue* images);
+ void SaveColorData();
+ void SaveTintData();
+
+ // Let all the browser views know that themes have changed.
+ void NotifyThemeChanged();
+
+ // Load theme data from preferences.
+ void LoadThemePrefs();
+
+ // Cached images. We cache all retrieved and generated bitmaps and keep
+ // track of the pointers.
+ typedef std::map<int, SkBitmap*> ImageCache;
+ ImageCache image_cache_;
+
+ ResourceBundle& rb_;
+ Profile* profile_;
+
+ ImageMap images_;
+ ColorMap colors_;
+ TintMap tints_;
+
+ DISALLOW_COPY_AND_ASSIGN(BrowserThemeProvider);
+};
+
+#endif // CHROME_BROWSER_THEME_PROVIDER_H_
diff --git a/chrome/browser/browser_window.h b/chrome/browser/browser_window.h
index 440c66d..c5e2027 100644
--- a/chrome/browser/browser_window.h
+++ b/chrome/browser/browser_window.h
@@ -176,6 +176,10 @@ class BrowserWindow {
virtual void ShowHTMLDialog(HtmlDialogUIDelegate* delegate,
void* parent_window) = 0;
+ // BrowserThemeProvider calls this when a user has changed his or her theme,
+ // indicating that it's time to redraw everything.
+ virtual void UserChangedTheme() = 0;
+
// Construct a BrowserWindow implementation for the specified |browser|.
static BrowserWindow* CreateBrowserWindow(Browser* browser);
diff --git a/chrome/browser/dom_ui/dom_ui.cc b/chrome/browser/dom_ui/dom_ui.cc
index f8e0405..ff49039 100644
--- a/chrome/browser/dom_ui/dom_ui.cc
+++ b/chrome/browser/dom_ui/dom_ui.cc
@@ -10,7 +10,10 @@
#include "base/stl_util-inl.h"
#include "base/string_util.h"
#include "base/values.h"
+#include "base/win_util.h"
+#include "chrome/browser/profile.h"
#include "chrome/browser/tab_contents/tab_contents.h"
+#include "chrome/browser/tab_contents/tab_contents_view.h"
DOMUI::DOMUI(TabContents* contents)
: hide_favicon_(false),
@@ -79,6 +82,10 @@ void DOMUI::CallJavascriptFunction(
ExecuteJavascript(javascript);
}
+ThemeProvider* DOMUI::GetThemeProvider() const {
+ return tab_contents_->profile()->GetThemeProvider();
+}
+
void DOMUI::RegisterMessageCallback(const std::string &message,
MessageCallback *callback) {
message_callbacks_.insert(std::make_pair(message, callback));
diff --git a/chrome/browser/dom_ui/dom_ui.h b/chrome/browser/dom_ui/dom_ui.h
index 3c2dc16..fadff05 100644
--- a/chrome/browser/dom_ui/dom_ui.h
+++ b/chrome/browser/dom_ui/dom_ui.h
@@ -21,6 +21,7 @@ class Profile;
class RenderViewHost;
class Value;
class TabContents;
+class ThemeProvider;
// A DOMUI sets up the datasources and message handlers for a given HTML-based
// UI. It is contained by a DOMUIManager.
@@ -44,7 +45,7 @@ class DOMUI {
bool hide_favicon() const {
return hide_favicon_;
}
-
+
// Returns true if the bookmark bar should be forced to being visible,
// overriding the user's preference.
bool force_bookmark_bar_visible() const {
@@ -89,6 +90,8 @@ class DOMUI {
const Value& arg1,
const Value& arg2);
+ ThemeProvider* GetThemeProvider() const;
+
TabContents* tab_contents() { return tab_contents_; }
Profile* GetProfile();
diff --git a/chrome/browser/dom_ui/dom_ui_theme_source.cc b/chrome/browser/dom_ui/dom_ui_theme_source.cc
new file mode 100644
index 0000000..31ff1f8
--- /dev/null
+++ b/chrome/browser/dom_ui/dom_ui_theme_source.cc
@@ -0,0 +1,37 @@
+// Copyright (c) 2009 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/dom_ui/dom_ui_theme_source.h"
+
+#include "app/resource_bundle.h"
+#include "app/theme_provider.h"
+#include "base/gfx/png_encoder.h"
+#include "base/message_loop.h"
+#include "chrome/browser/profile.h"
+#include "chrome/browser/theme_resources_util.h"
+#include "chrome/common/url_constants.h"
+#include "grit/theme_resources.h"
+
+DOMUIThemeSource::DOMUIThemeSource(Profile* profile)
+ : DataSource(chrome::kChromeUIThemePath, MessageLoop::current()),
+ profile_(profile) {
+}
+
+void DOMUIThemeSource::StartDataRequest(const std::string& path,
+ int request_id) {
+ ThemeProvider* tp = profile_->GetThemeProvider();
+ if (tp) {
+ int id = ThemeResourcesUtil::GetId(path);
+ if (id != -1) {
+ SkBitmap* image = tp->GetBitmapNamed(id);
+ if (image) {
+ std::vector<unsigned char> png_bytes;
+ PNGEncoder::EncodeBGRASkBitmap(*image, false, &png_bytes);
+
+ scoped_refptr<RefCountedBytes> image_data = new RefCountedBytes(png_bytes);
+ SendResponse(request_id, image_data);
+ }
+ }
+ }
+}
diff --git a/chrome/browser/dom_ui/dom_ui_theme_source.h b/chrome/browser/dom_ui/dom_ui_theme_source.h
new file mode 100644
index 0000000..279042d
--- /dev/null
+++ b/chrome/browser/dom_ui/dom_ui_theme_source.h
@@ -0,0 +1,30 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_BROWSER_DOM_UI_DOM_UI_THEME_SOURCE_H_
+#define CHROME_BROWSER_DOM_UI_DOM_UI_THEME_SOURCE_H_
+
+#include "chrome/browser/dom_ui/chrome_url_data_manager.h"
+
+class Profile;
+
+// ThumbnailSource is the gateway between network-level chrome:
+// requests for thumbnails and the history backend that serves these.
+class DOMUIThemeSource : public ChromeURLDataManager::DataSource {
+public:
+ explicit DOMUIThemeSource(Profile* profile);
+
+ // Called when the network layer has requested a resource underneath
+ // the path we registered.
+ virtual void StartDataRequest(const std::string& path, int request_id);
+ virtual std::string GetMimeType(const std::string& path) const {
+ return "image/png";
+ }
+
+private:
+ Profile* profile_;
+ DISALLOW_COPY_AND_ASSIGN(DOMUIThemeSource);
+};
+
+#endif // CHROME_BROWSER_DOM_UI_DOM_UI_THEME_SOURCE_H_
diff --git a/chrome/browser/dom_ui/new_tab_ui.cc b/chrome/browser/dom_ui/new_tab_ui.cc
index d4bdf97..b13fbb2 100644
--- a/chrome/browser/dom_ui/new_tab_ui.cc
+++ b/chrome/browser/dom_ui/new_tab_ui.cc
@@ -16,6 +16,7 @@
#include "chrome/browser/browser_process.h"
#include "chrome/browser/dom_ui/dom_ui_favicon_source.h"
#include "chrome/browser/dom_ui/dom_ui_thumbnail_source.h"
+#include "chrome/browser/dom_ui/dom_ui_theme_source.h"
#include "chrome/browser/dom_ui/history_ui.h"
#include "chrome/browser/history/page_usage_data.h"
#include "chrome/browser/metrics/user_metrics.h"
@@ -1165,6 +1166,11 @@ NewTabUI::NewTabUI(TabContents* contents)
// In testing mode there may not be an I/O thread.
if (g_browser_process->io_thread()) {
+ g_browser_process->io_thread()->message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(&chrome_url_data_manager,
+ &ChromeURLDataManager::AddDataSource,
+ new DOMUIThemeSource(GetProfile())));
+
NewTabHTMLSource* html_source = new NewTabHTMLSource();
g_browser_process->io_thread()->message_loop()->PostTask(FROM_HERE,
NewRunnableMethod(&chrome_url_data_manager,
diff --git a/chrome/browser/extensions/extension.cc b/chrome/browser/extensions/extension.cc
index 0f9c24e..e1533db6 100644
--- a/chrome/browser/extensions/extension.cc
+++ b/chrome/browser/extensions/extension.cc
@@ -30,6 +30,9 @@ const wchar_t* Extension::kPluginsDirKey = L"plugins_dir";
const wchar_t* Extension::kBackgroundKey = L"background_page";
const wchar_t* Extension::kRunAtKey = L"run_at";
const wchar_t* Extension::kThemeKey = L"theme";
+const wchar_t* Extension::kThemeImagesKey = L"images";
+const wchar_t* Extension::kThemeColorsKey = L"colors";
+const wchar_t* Extension::kThemeTintsKey = L"tints";
const wchar_t* Extension::kToolstripsKey = L"toolstrips";
const wchar_t* Extension::kTooltipKey = L"tooltip";
const wchar_t* Extension::kTypeKey = L"type";
@@ -110,6 +113,16 @@ const char* Extension::kMissingFileError =
"At least one js or css file is required for 'content_scripts[*]'.";
const char* Extension::kMissingPageActionIcon =
"Unable to find 'page_actions[*].icon'";
+const char* Extension::kInvalidThemeError =
+ "Invalid value for 'theme'.";
+const char* Extension::kInvalidThemeImagesError =
+ "Invalid value for theme images - images must be strings.";
+const char* Extension::kInvalidThemeImagesMissingError =
+ "Am image specified in the theme is missing.";
+const char* Extension::kInvalidThemeColorsError =
+ "Invalid value for theme colors - colors must be integers";
+const char* Extension::kInvalidThemeTintsError =
+ "Invalid value for theme images - tints must be decimal numbers.";
const size_t Extension::kIdSize = 20; // SHA1 (160 bits) == 20 bytes
@@ -124,7 +137,10 @@ Extension::Extension(const Extension& rhs)
page_actions_(rhs.page_actions_),
plugins_dir_(rhs.plugins_dir_),
zip_hash_(rhs.zip_hash_),
- theme_paths_(rhs.theme_paths_) {
+ theme_images_(rhs.theme_images_),
+ theme_colors_(rhs.theme_colors_),
+ theme_tints_(rhs.theme_tints_),
+ is_theme_(rhs.is_theme_) {
}
Extension::~Extension() {
@@ -149,14 +165,6 @@ GURL Extension::GetResourceURL(const GURL& extension_url,
return ret_val;
}
-FilePath Extension::GetThemeResourcePath(const int resource_id) {
- std::wstring id = IntToWString(resource_id);
- std::string path = theme_paths_[id];
- if (path.size())
- return path_.AppendASCII(path.c_str());
- return FilePath();
-}
-
const PageAction* Extension::GetPageAction(std::string id) const {
PageActionMap::const_iterator it = page_actions_.find(id);
if (it == page_actions_.end())
@@ -500,6 +508,81 @@ bool Extension::InitFromValue(const DictionaryValue& source, bool require_id,
}
}
+ // Initialize themes. If a theme is included, no other items may be processed
+ // (we currently don't want people bundling themes and extension stuff
+ // together).
+ //
+ // TODO(glen): Error if other items *are* included.
+ is_theme_ = false;
+ if (source.HasKey(kThemeKey)) {
+ DictionaryValue* theme_value;
+ if (!source.GetDictionary(kThemeKey, &theme_value)) {
+ *error = kInvalidThemeError;
+ return false;
+ }
+ is_theme_ = true;
+
+ theme_images_ = new DictionaryValue;
+ DictionaryValue* images_value;
+ if (theme_value->GetDictionary(kThemeImagesKey, &images_value)) {
+ // Validate that the images are all strings
+ DictionaryValue::key_iterator iter = images_value->begin_keys();
+ while (iter != images_value->end_keys()) {
+ std::string val;
+ if (!images_value->GetString(*iter, &val)) {
+ *error = kInvalidThemeImagesError;
+ return false;
+ }
+ ++iter;
+ }
+ theme_images_ = static_cast<DictionaryValue*>(images_value->DeepCopy());
+ }
+
+ theme_colors_ = new DictionaryValue;
+ DictionaryValue* colors_value;
+ if (theme_value->GetDictionary(kThemeColorsKey, &colors_value)) {
+ // Validate that the colors are all three-item lists
+ DictionaryValue::key_iterator iter = colors_value->begin_keys();
+ while (iter != colors_value->end_keys()) {
+ std::string val;
+ int color = 0;
+ ListValue* color_list;
+ if (!colors_value->GetList(*iter, &color_list) ||
+ color_list->GetSize() != 3 ||
+ !color_list->GetInteger(0, &color) ||
+ !color_list->GetInteger(1, &color) ||
+ !color_list->GetInteger(2, &color)) {
+ *error = kInvalidThemeColorsError;
+ return false;
+ }
+ ++iter;
+ }
+ theme_colors_ = static_cast<DictionaryValue*>(colors_value->DeepCopy());
+ }
+
+ theme_tints_ = new DictionaryValue;
+ DictionaryValue* tints_value;
+ if (theme_value->GetDictionary(kThemeTintsKey, &tints_value)) {
+ // Validate that the tints are all reals.
+ DictionaryValue::key_iterator iter = tints_value->begin_keys();
+ while (iter != tints_value->end_keys()) {
+ ListValue* tint_list;
+ double hue = 0;
+ if (!tints_value->GetList(*iter, &tint_list) &&
+ tint_list->GetSize() != 3 ||
+ !tint_list->GetReal(0, &hue) ||
+ !tint_list->GetReal(1, &hue) ||
+ !tint_list->GetReal(2, &hue)) {
+ *error = kInvalidThemeTintsError;
+ return false;
+ }
+ ++iter;
+ }
+ theme_tints_ = static_cast<DictionaryValue*>(tints_value->DeepCopy());
+ }
+ return true;
+ }
+
// Initialize plugins dir (optional).
if (source.HasKey(kPluginsDirKey)) {
std::string plugins_dir;
@@ -538,22 +621,6 @@ bool Extension::InitFromValue(const DictionaryValue& source, bool require_id,
}
}
- if (source.HasKey(kThemeKey)) {
- DictionaryValue* dict_value;
- if (source.GetDictionary(kThemeKey, &dict_value)) {
- DictionaryValue::key_iterator iter = dict_value->begin_keys();
- while (iter != dict_value->end_keys()) {
- std::string val;
- if (dict_value->GetString(*iter, &val)) {
- std::wstring id = *iter;
- theme_paths_[id] = val;
- }
- ++iter;
- }
- ResourceBundle::GetSharedInstance().SetThemeExtension(*this);
- }
- }
-
// Initialize content scripts (optional).
if (source.HasKey(kContentScriptsKey)) {
ListValue* list_value;
diff --git a/chrome/browser/extensions/extension.h b/chrome/browser/extensions/extension.h
index c612203..4e19c9a 100644
--- a/chrome/browser/extensions/extension.h
+++ b/chrome/browser/extensions/extension.h
@@ -18,6 +18,7 @@
#include "chrome/common/extensions/url_pattern.h"
#include "chrome/common/page_action.h"
#include "googleurl/src/gurl.h"
+#include "SkColor.h"
// Represents a Chromium extension.
class Extension {
@@ -45,6 +46,9 @@ class Extension {
static const wchar_t* kBackgroundKey;
static const wchar_t* kRunAtKey;
static const wchar_t* kThemeKey;
+ static const wchar_t* kThemeImagesKey;
+ static const wchar_t* kThemeColorsKey;
+ static const wchar_t* kThemeTintsKey;
static const wchar_t* kToolstripsKey;
static const wchar_t* kTooltipKey;
static const wchar_t* kTypeKey;
@@ -87,6 +91,11 @@ class Extension {
static const char* kInvalidPermissionError;
static const char* kInvalidPermissionSchemeError;
static const char* kInvalidZipHashError;
+ static const char* kInvalidThemeError;
+ static const char* kInvalidThemeImagesMissingError;
+ static const char* kInvalidThemeImagesError;
+ static const char* kInvalidThemeColorsError;
+ static const char* kInvalidThemeTintsError;
static const char* kMissingFileError;
static const char* kMissingPageActionIcon;
@@ -115,19 +124,17 @@ class Extension {
return GetResourcePath(path(), relative_path);
}
+ DictionaryValue* GetThemeImages() { return theme_images_; }
+ DictionaryValue* GetThemeColors() { return theme_colors_; }
+ DictionaryValue* GetThemeTints() { return theme_tints_; }
+ bool IsTheme() { return is_theme_; }
+
// Initialize the extension from a parsed manifest.
// If |require_id| is true, will return an error if the "id" key is missing
// from the value.
bool InitFromValue(const DictionaryValue& value, bool require_id,
std::string* error);
- // Returns an absolute path to a resource inside of an extension if the
- // extension has a theme defined with the given |resource_id|. Otherwise
- // the path will be empty. Note that this method is not static as it is
- // only intended to be called on an extension which has registered itself
- // as providing a theme.
- FilePath GetThemeResourcePath(const int resource_id);
-
// Retrieves a page action by |id|.
const PageAction* GetPageAction(std::string id) const;
@@ -207,7 +214,16 @@ class Extension {
std::string zip_hash_;
// A map of resource id's to relative file paths.
- std::map<const std::wstring, std::string> theme_paths_;
+ DictionaryValue* theme_images_;
+
+ // A map of color names to colors.
+ DictionaryValue* theme_colors_;
+
+ // A map of color names to colors.
+ DictionaryValue* theme_tints_;
+
+ // Whether the extension is a theme - if it is, certain things are disabled.
+ bool is_theme_;
// The sites this extension has permission to talk to (using XHR, etc).
std::vector<URLPattern> permissions_;
diff --git a/chrome/browser/extensions/extensions_service.cc b/chrome/browser/extensions/extensions_service.cc
index da5db0f1..2a152e1 100644
--- a/chrome/browser/extensions/extensions_service.cc
+++ b/chrome/browser/extensions/extensions_service.cc
@@ -183,11 +183,30 @@ void ExtensionsService::OnExtensionInstalled(Extension* extension,
NotificationType::EXTENSION_INSTALLED,
NotificationService::AllSources(),
Details<Extension>(extension));
+
+ // If the extension is a theme, tell the profile (and therefore ThemeProvider)
+ // to apply it.
+ if (extension->IsTheme()) {
+ NotificationService::current()->Notify(
+ NotificationType::THEME_INSTALLED,
+ NotificationService::AllSources(),
+ Details<Extension>(extension));
+ }
+}
+
+void ExtensionsService::OnExtensionVersionReinstalled(const std::string& id) {
+ Extension* extension = GetExtensionByID(id);
+ if (extension && extension->IsTheme()) {
+ NotificationService::current()->Notify(
+ NotificationType::THEME_INSTALLED,
+ NotificationService::AllSources(),
+ Details<Extension>(extension));
+ }
}
Extension* ExtensionsService::GetExtensionByID(std::string id) {
for (ExtensionList::const_iterator iter = extensions_.begin();
- iter != extensions_.end(); ++iter) {
+ iter != extensions_.end(); ++iter) {
if ((*iter)->id() == id)
return *iter;
}
@@ -329,7 +348,24 @@ Extension* ExtensionsServiceBackend::LoadExtension(
return NULL;
}
- // Validate that claimed resources actually exist.
+ // Validate that claimed image resources actually exist.
+ DictionaryValue* images_value = extension->GetThemeImages();
+ if (images_value) {
+ DictionaryValue::key_iterator iter = images_value->begin_keys();
+ while (iter != images_value->end_keys()) {
+ std::string val;
+ images_value->GetString(*iter, &val);
+ const FilePath& path = extension->path().AppendASCII(val);
+ if (!file_util::PathExists(path)) {
+ ReportExtensionLoadError(extension_path,
+ StringPrintf("Could not load '%s' for theme.",
+ WideToUTF8(path.ToWStringHack()).c_str()));
+ return NULL;
+ }
+ ++iter;
+ }
+ }
+ // Validate that claimed script resources actually exist.
for (size_t i = 0; i < extension->content_scripts().size(); ++i) {
const UserScript& script = extension->content_scripts()[i];
@@ -532,8 +568,9 @@ bool ExtensionsServiceBackend::CheckCurrentVersion(
// has actually loaded successfully.
FilePath version_dir = dest_dir.AppendASCII(current_version_str);
if (file_util::PathExists(version_dir)) {
- ReportExtensionInstallError(dest_dir,
- "Existing version is already up to date.");
+ std::string id = WideToASCII(dest_dir.BaseName().ToWStringHack());
+ StringToLowerASCII(&id);
+ ReportExtensionVersionReinstalled(id);
return false;
}
}
@@ -727,6 +764,14 @@ void ExtensionsServiceBackend::ReportExtensionInstallError(
ExtensionErrorReporter::GetInstance()->ReportError(message, alert_on_error_);
}
+void ExtensionsServiceBackend::ReportExtensionVersionReinstalled(
+ const std::string& id) {
+ frontend_->GetMessageLoop()->PostTask(FROM_HERE, NewRunnableMethod(
+ frontend_,
+ &ExtensionsServiceFrontendInterface::OnExtensionVersionReinstalled,
+ id));
+}
+
void ExtensionsServiceBackend::ReportExtensionInstalled(
const FilePath& path, bool update) {
// After it's installed, load it right away with the same settings.
diff --git a/chrome/browser/extensions/extensions_service.h b/chrome/browser/extensions/extensions_service.h
index 9a83c99..31b8f90e 100644
--- a/chrome/browser/extensions/extensions_service.h
+++ b/chrome/browser/extensions/extensions_service.h
@@ -51,8 +51,15 @@ class ExtensionsServiceFrontendInterface
// installed extension rather than a new installation.
virtual void OnExtensionInstalled(Extension* extension, bool is_update) = 0;
+ // Called when an existing extension is installed by the user. We may wish to
+ // notify the user about the prior existence of the extension, or take some
+ // action using the fact that the user chose to reinstall the extension as a
+ // signal (for example, setting the default theme to the extension).
+ virtual void OnExtensionVersionReinstalled(const std::string& id) = 0;
+
// Lookup an extension by |id|.
virtual Extension* GetExtensionByID(std::string id) = 0;
+
};
@@ -77,6 +84,7 @@ class ExtensionsService : public ExtensionsServiceFrontendInterface {
virtual void OnExtensionsLoaded(ExtensionList* extensions);
virtual void OnExtensionInstalled(Extension* extension, bool is_update);
virtual Extension* GetExtensionByID(std::string id);
+ virtual void OnExtensionVersionReinstalled(const std::string& id);
// The name of the file that the current active version number is stored in.
static const char* kCurrentVersionFileName;
@@ -176,6 +184,9 @@ class ExtensionsServiceBackend
void ReportExtensionInstallError(const FilePath& extension_path,
const std::string& error);
+ // Notify the frontend that the extension had already been installed.
+ void ReportExtensionVersionReinstalled(const std::string& id);
+
// Notify the frontend that extensions were installed.
// |is_update| is true if this was an update to an existing extension.
void ReportExtensionInstalled(const FilePath& path, bool is_update);
diff --git a/chrome/browser/extensions/extensions_service_unittest.cc b/chrome/browser/extensions/extensions_service_unittest.cc
index e6c27df..dc9be78 100644
--- a/chrome/browser/extensions/extensions_service_unittest.cc
+++ b/chrome/browser/extensions/extensions_service_unittest.cc
@@ -66,10 +66,19 @@ class ExtensionsServiceTestFrontend
return &extensions_;
}
+ void ClearInstalledReinstalled() {
+ installed_ = NULL;
+ reinstalled_id_ = std::string();
+ }
+
Extension* installed() {
return installed_;
}
+ std::string reinstalled_id() {
+ return reinstalled_id_;
+ }
+
// ExtensionsServiceFrontendInterface
virtual MessageLoop* GetMessageLoop() {
return &message_loop_;
@@ -94,6 +103,10 @@ class ExtensionsServiceTestFrontend
installed_ = extension;
}
+ virtual void OnExtensionVersionReinstalled(const std::string& id) {
+ reinstalled_id_ = id;
+ }
+
virtual Extension* GetExtensionByID(std::string id) {
return NULL;
}
@@ -122,11 +135,11 @@ class ExtensionsServiceTestFrontend
ExtensionErrorReporter::GetInstance()->ClearErrors();
}
-
private:
MessageLoop message_loop_;
ExtensionList extensions_;
Extension* installed_;
+ std::string reinstalled_id_;
};
// make the test a PlatformTest to setup autorelease pools properly on mac
@@ -277,9 +290,6 @@ TEST_F(ExtensionsServiceTest, InstallExtension) {
frontend->TestInstallExtension(path, backend, true);
// TODO(erikkay): verify the contents of the installed extension.
- // Installing the same extension twice should fail.
- frontend->TestInstallExtension(path, backend, false);
-
// 0-length extension file.
path = extensions_path.AppendASCII("not_an_extension.crx");
frontend->TestInstallExtension(path, backend, false);
@@ -300,6 +310,65 @@ TEST_F(ExtensionsServiceTest, InstallExtension) {
// TODO(erikkay): add tests for upgrade cases.
}
+TEST_F(ExtensionsServiceTest, ReinstallExtension) {
+ // In this test, we install two extensions, verify that they both install
+ // correctly, then install the first extension again and verify that it was
+ // not installed, and that VersionReinstalled was called instead.
+ FilePath extensions_path;
+ ASSERT_TRUE(PathService::Get(chrome::DIR_TEST_DATA, &extensions_path));
+ extensions_path = extensions_path.AppendASCII("extensions");
+
+ FilePath install_dir;
+ file_util::CreateNewTempDirectory(FILE_PATH_LITERAL("ext_test"),
+ &install_dir);
+ scoped_refptr<ExtensionsServiceBackend> backend(
+ new ExtensionsServiceBackend(install_dir));
+ scoped_refptr<ExtensionsServiceTestFrontend> frontend(
+ new ExtensionsServiceTestFrontend);
+
+ FilePath path = extensions_path.AppendASCII("good.crx");
+ FilePath path2 = extensions_path.AppendASCII("theme.crx");
+
+ // Verify that our extensions are valid.
+ ASSERT_TRUE(file_util::PathExists(path));
+ ASSERT_TRUE(file_util::PathExists(path2));
+
+ frontend->ClearInstalledReinstalled();
+ // Install an extension.
+ backend->InstallExtension(path,
+ scoped_refptr<ExtensionsServiceFrontendInterface>(frontend.get()));
+ frontend->GetMessageLoop()->RunAllPending();
+ std::vector<std::string> errors = GetErrors();
+
+ // Verify that it was installed.
+ EXPECT_TRUE(frontend->installed()) << path.value();
+ EXPECT_EQ(0u, errors.size()) << path.value();
+
+ // Install our second extension.
+ frontend->ClearInstalledReinstalled();
+ backend->InstallExtension(path2,
+ scoped_refptr<ExtensionsServiceFrontendInterface>(frontend.get()));
+ frontend->GetMessageLoop()->RunAllPending();
+ errors = GetErrors();
+
+ // Verify that it was installed without reinstall getting called.
+ EXPECT_TRUE(frontend->installed()) << path2.value();
+ EXPECT_TRUE(frontend->reinstalled_id().empty());
+ EXPECT_EQ(0u, errors.size()) << path.value();
+
+ // Install the first extension again.
+ frontend->ClearInstalledReinstalled();
+ backend->InstallExtension(path,
+ scoped_refptr<ExtensionsServiceFrontendInterface>(frontend.get()));
+ frontend->GetMessageLoop()->RunAllPending();
+ errors = GetErrors();
+
+ // Verify that reinstall was called and installed was not.
+ EXPECT_FALSE(frontend->installed()) << path.value();
+ EXPECT_FALSE(frontend->reinstalled_id().empty()) << path.value();
+ EXPECT_EQ(0u, errors.size()) << path.value();
+}
+
TEST_F(ExtensionsServiceTest, LoadExtension) {
FilePath extensions_path;
ASSERT_TRUE(PathService::Get(chrome::DIR_TEST_DATA, &extensions_path));
diff --git a/chrome/browser/extensions/extensions_ui.cc b/chrome/browser/extensions/extensions_ui.cc
index d74b802..d05e39a 100644
--- a/chrome/browser/extensions/extensions_ui.cc
+++ b/chrome/browser/extensions/extensions_ui.cc
@@ -9,12 +9,14 @@
#include "base/string_util.h"
#include "base/thread.h"
#include "chrome/browser/browser_process.h"
+#include "chrome/browser/extensions/extension.h"
#include "chrome/browser/extensions/extension_message_service.h"
#include "chrome/browser/extensions/extensions_service.h"
#include "chrome/browser/extensions/extension_error_reporter.h"
#include "chrome/browser/profile.h"
#include "chrome/browser/renderer_host/render_widget_host.h"
#include "chrome/browser/renderer_host/render_view_host.h"
+#include "chrome/common/extensions/user_script.h"
#include "chrome/common/extensions/url_pattern.h"
#include "chrome/common/jstemplate_builder.h"
#include "chrome/common/url_constants.h"
diff --git a/chrome/browser/external_tab_container.h b/chrome/browser/external_tab_container.h
index c0effce..f3a186d 100644
--- a/chrome/browser/external_tab_container.h
+++ b/chrome/browser/external_tab_container.h
@@ -113,6 +113,7 @@ class ExternalTabContainer : public TabContentsDelegate,
virtual gfx::NativeView GetNativeView() const;
virtual void PaintNow(const gfx::Rect& update_rect);
virtual views::RootView* GetRootView();
+ virtual Widget* GetRootWidget() const { return NULL; }
virtual bool IsVisible() const;
virtual bool IsActive() const;
virtual bool GetAccelerator(int cmd_id,
diff --git a/chrome/browser/profile.cc b/chrome/browser/profile.cc
index 86f32b6..9406216 100644
--- a/chrome/browser/profile.cc
+++ b/chrome/browser/profile.cc
@@ -28,6 +28,7 @@
#include "chrome/browser/sessions/tab_restore_service.h"
#include "chrome/browser/spellchecker.h"
#include "chrome/browser/ssl/ssl_host_state.h"
+#include "chrome/browser/browser_theme_provider.h"
#include "chrome/browser/visitedlink_master.h"
#include "chrome/browser/webdata/web_data_service.h"
#include "chrome/common/chrome_constants.h"
@@ -38,7 +39,6 @@
#include "chrome/common/render_messages.h"
#include "grit/locale_settings.h"
-
using base::Time;
using base::TimeDelta;
@@ -59,6 +59,10 @@ void Profile::RegisterUserPrefs(PrefService* prefs) {
IDS_SPELLCHECK_DICTIONARY);
prefs->RegisterBooleanPref(prefs::kEnableSpellCheck, true);
prefs->RegisterBooleanPref(prefs::kEnableUserScripts, false);
+ prefs->RegisterStringPref(prefs::kCurrentThemeID, L"");
+ prefs->RegisterDictionaryPref(prefs::kCurrentThemeImages);
+ prefs->RegisterDictionaryPref(prefs::kCurrentThemeColors);
+ prefs->RegisterDictionaryPref(prefs::kCurrentThemeTints);
prefs->RegisterBooleanPref(prefs::kEnableExtensions, false);
}
@@ -76,7 +80,7 @@ URLRequestContext* Profile::GetDefaultRequestContext() {
////////////////////////////////////////////////////////////////////////////////
//
// OffTheRecordProfileImpl is a profile subclass that wraps an existing profile
-// to make it suitable for the off the record mode.
+// to make it suitable for the off the record mode.
//
////////////////////////////////////////////////////////////////////////////////
class OffTheRecordProfileImpl : public Profile,
@@ -204,6 +208,22 @@ class OffTheRecordProfileImpl : public Profile,
return (download_manager_.get() != NULL);
}
+ virtual void InitThemes() {
+ GetOriginalProfile()->InitThemes();
+ }
+
+ virtual void SetTheme(Extension* extension) {
+ GetOriginalProfile()->SetTheme(extension);
+ }
+
+ virtual void ClearTheme() {
+ GetOriginalProfile()->ClearTheme();
+ }
+
+ virtual ThemeProvider* GetThemeProvider() {
+ return GetOriginalProfile()->GetThemeProvider();
+ }
+
virtual URLRequestContext* GetRequestContext() {
return request_context_;
}
@@ -341,6 +361,9 @@ class OffTheRecordProfileImpl : public Profile,
// The download manager that only stores downloaded items in memory.
scoped_refptr<DownloadManager> download_manager_;
+ // The download manager that only stores downloaded items in memory.
+ scoped_refptr<BrowserThemeProvider> theme_provider_;
+
// We don't want SSLHostState from the OTR profile to leak back to the main
// profile because then the main profile would learn some of the host names
// the user visited while OTR.
@@ -362,6 +385,7 @@ ProfileImpl::ProfileImpl(const FilePath& path)
history_service_created_(false),
created_web_data_service_(false),
created_download_manager_(false),
+ created_theme_provider_(false),
start_time_(Time::Now()),
spellchecker_(NULL),
shutdown_session_service_(false) {
@@ -387,6 +411,10 @@ ProfileImpl::ProfileImpl(const FilePath& path)
prefs->transient()->SetString(prefs::kHomePage, "about:linux-splash");
prefs->transient()->SetBoolean(prefs::kHomePageIsNewTabPage, false);
#endif
+
+ // Listen for theme installation.
+ NotificationService::current()->AddObserver(this,
+ NotificationType::THEME_INSTALLED, NotificationService::AllSources());
}
void ProfileImpl::InitExtensions() {
@@ -440,6 +468,9 @@ ProfileImpl::~ProfileImpl() {
// before the history is shutdown so it can properly cancel all requests.
download_manager_ = NULL;
+ // The theme provider provides bitmaps to whoever wants them.
+ theme_provider_ = NULL;
+
// Remove pref observers.
PrefService* prefs = GetPrefs();
prefs->RemovePrefObserver(prefs::kSpellCheckDictionary, this);
@@ -449,6 +480,10 @@ ProfileImpl::~ProfileImpl() {
personalization_.reset();
#endif
+ // Remove theme observer.
+ NotificationService::current()->RemoveObserver(this,
+ NotificationType::THEME_INSTALLED, NotificationService::AllSources());
+
// Both HistoryService and WebDataService maintain threads for background
// processing. Its possible each thread still has tasks on it that have
// increased the ref count of the service. In such a situation, when we
@@ -731,6 +766,30 @@ bool ProfileImpl::HasCreatedDownloadManager() const {
return created_download_manager_;
}
+void ProfileImpl::InitThemes() {
+ if (!created_theme_provider_) {
+ scoped_refptr<BrowserThemeProvider> themes(new BrowserThemeProvider);
+ themes->Init(this);
+ created_theme_provider_ = true;
+ theme_provider_.swap(themes);
+ }
+}
+
+void ProfileImpl::SetTheme(Extension* extension) {
+ InitThemes();
+ theme_provider_.get()->SetTheme(extension);
+}
+
+void ProfileImpl::ClearTheme() {
+ InitThemes();
+ theme_provider_.get()->UseDefaultTheme();
+}
+
+ThemeProvider* ProfileImpl::GetThemeProvider() {
+ InitThemes();
+ return theme_provider_.get();
+}
+
SessionService* ProfileImpl::GetSessionService() {
if (!session_service_.get() && !shutdown_session_service_) {
session_service_ = new SessionService(this);
@@ -911,6 +970,9 @@ void ProfileImpl::Observe(NotificationType type,
*pref_name_in == prefs::kEnableSpellCheck) {
InitializeSpellChecker(true);
}
+ } else if (NotificationType::THEME_INSTALLED == type) {
+ Extension* extension = Details<Extension>(details).ptr();
+ SetTheme(extension);
}
}
diff --git a/chrome/browser/profile.h b/chrome/browser/profile.h
index 2a96277..a83d76a 100644
--- a/chrome/browser/profile.h
+++ b/chrome/browser/profile.h
@@ -10,6 +10,7 @@
#include <set>
#include <string>
+#include "app/theme_provider.h"
#include "base/basictypes.h"
#include "base/file_path.h"
#include "base/scoped_ptr.h"
@@ -22,6 +23,7 @@
class BookmarkModel;
class ChromeURLRequestContext;
class DownloadManager;
+class Extension;
class ExtensionProcessManager;
class ExtensionsService;
class HistoryService;
@@ -33,6 +35,7 @@ class SSLHostState;
class TabRestoreService;
class TemplateURLFetcher;
class TemplateURLModel;
+class BrowserThemeProvider;
class URLRequestContext;
class UserScriptMaster;
class VisitedLinkMaster;
@@ -162,6 +165,18 @@ class Profile {
virtual DownloadManager* GetDownloadManager() = 0;
virtual bool HasCreatedDownloadManager() const = 0;
+ // Init our themes system.
+ virtual void InitThemes() = 0;
+
+ // Set the theme to the specified extension.
+ virtual void SetTheme(Extension* extension) = 0;
+
+ // Clear the theme and reset it to default.
+ virtual void ClearTheme() = 0;
+
+ // Returns or creates the ThemeProvider associated with this profile
+ virtual ThemeProvider* GetThemeProvider() = 0;
+
// Returns the request context information associated with this profile. Call
// this only on the UI thread, since it can send notifications that should
// happen on the UI thread.
@@ -289,6 +304,10 @@ class ProfileImpl : public Profile,
virtual TemplateURLModel* GetTemplateURLModel();
virtual TemplateURLFetcher* GetTemplateURLFetcher();
virtual DownloadManager* GetDownloadManager();
+ virtual void InitThemes();
+ virtual void SetTheme(Extension* extension);
+ virtual void ClearTheme();
+ virtual ThemeProvider* GetThemeProvider();
virtual bool HasCreatedDownloadManager() const;
virtual URLRequestContext* GetRequestContext();
virtual URLRequestContext* GetRequestContextForMedia();
@@ -363,9 +382,11 @@ class ProfileImpl : public Profile,
scoped_refptr<HistoryService> history_service_;
scoped_refptr<WebDataService> web_data_service_;
scoped_refptr<SessionService> session_service_;
+ scoped_refptr<BrowserThemeProvider> theme_provider_;
bool history_service_created_;
bool created_web_data_service_;
bool created_download_manager_;
+ bool created_theme_provider_;
// Whether or not the last session exited cleanly. This is set only once.
bool last_session_exited_cleanly_;
diff --git a/chrome/browser/resources/new_tab.html b/chrome/browser/resources/new_tab.html
index 2b27e00..5daf5c5 100644
--- a/chrome/browser/resources/new_tab.html
+++ b/chrome/browser/resources/new_tab.html
@@ -133,6 +133,8 @@ logEvent('log start');
body {
background-color:white;
margin:0px;
+ background-image:url(chrome-ui://theme/theme_newtab_background);
+ background-repeat:repeat-x;
}
html[firstview='true'] #main {
opacity:0.0;
diff --git a/chrome/browser/views/bookmark_bar_view.cc b/chrome/browser/views/bookmark_bar_view.cc
index 51beca0a..88264f2 100644
--- a/chrome/browser/views/bookmark_bar_view.cc
+++ b/chrome/browser/views/bookmark_bar_view.cc
@@ -87,7 +87,6 @@ static const SkColor kBackgroundColor = SkColorSetRGB(237, 244, 252);
// Border colors for the BookmarBarView.
static const SkColor kTopBorderColor = SkColorSetRGB(222, 234, 248);
-static const SkColor kBottomBorderColor = SkColorSetRGB(178, 178, 178);
// Background color for when the bookmarks bar is only being displayed on the
// new tab page - this color should match the background color of the new tab
@@ -222,30 +221,6 @@ class BookmarkButton : public views::TextButton {
return event_utils::IsPossibleDispositionEvent(e);
}
- virtual void Paint(ChromeCanvas *canvas) {
- views::TextButton::Paint(canvas);
-
- PaintAnimation(this, canvas, show_animation_->GetCurrentValue());
- }
-
- static void PaintAnimation(views::View* view,
- ChromeCanvas* canvas,
- double animation_value) {
- // Since we can't change the alpha of the button (it contains un-alphable
- // text), we paint the bar background over the front of the button. As the
- // bar background is a gradient, we have to paint the gradient at the
- // size of the parent (hence all the margin math below). We can't use
- // the parent's actual bounds because they differ from what is painted.
- SkPaint paint;
- paint.setAlpha(static_cast<int>((1.0 - animation_value) * 255));
- paint.setShader(skia::CreateGradientShader(0,
- view->height() + kTopMargin + kBottomMargin,
- kTopBorderColor,
- kBackgroundColor))->safeUnref();
- canvas->FillRectInt(0, -kTopMargin, view->width(),
- view->height() + kTopMargin + kBottomMargin, paint);
- }
-
private:
const GURL& url_;
Profile* profile_;
@@ -283,9 +258,6 @@ class BookmarkFolderButton : public views::MenuButton {
virtual void Paint(ChromeCanvas *canvas) {
views::MenuButton::Paint(canvas, false);
-
- BookmarkButton::PaintAnimation(this, canvas,
- show_animation_->GetCurrentValue());
}
private:
@@ -588,7 +560,8 @@ void BookmarkBarView::Paint(ChromeCanvas* canvas) {
canvas->FillRectInt(kNewtabBackgroundColor, 0, 0, width(), height());
// Draw the 'bottom' of the toolbar above our bubble.
- canvas->FillRectInt(kBottomBorderColor, 0, 0, width(), 1);
+ canvas->FillRectInt(ResourceBundle::toolbar_separator_color,
+ 0, 0, width(), 1);
SkRect rect;
@@ -632,15 +605,14 @@ void BookmarkBarView::Paint(ChromeCanvas* canvas) {
SkDoubleToScalar(roundness),
SkDoubleToScalar(roundness), border_paint);
} else {
- SkPaint paint;
- paint.setShader(skia::CreateGradientShader(0,
- height(),
- kTopBorderColor,
- kBackgroundColor))->safeUnref();
- canvas->FillRectInt(0, 0, width(), height(), paint);
-
- canvas->FillRectInt(kTopBorderColor, 0, 0, width(), 1);
- canvas->FillRectInt(kBottomBorderColor, 0, height() - 1, width(), 1);
+ gfx::Rect bounds = GetBounds(views::View::APPLY_MIRRORING_TRANSFORMATION);
+
+ // +1 is for toolbar outdent.
+ canvas->TileImageInt(*GetThemeProvider()->
+ GetBitmapNamed(IDR_THEME_TOOLBAR),
+ bounds.x() + 1, bounds.y(), 0, 0, width(), height());
+ canvas->FillRectInt(ResourceBundle::toolbar_separator_color,
+ 0, height() - 1, width(), 1);
}
}
diff --git a/chrome/browser/views/constrained_window_impl.cc b/chrome/browser/views/constrained_window_impl.cc
index cde0327..9b4ee70 100644
--- a/chrome/browser/views/constrained_window_impl.cc
+++ b/chrome/browser/views/constrained_window_impl.cc
@@ -61,16 +61,23 @@ enum {
FRAME_TOP_LEFT_CORNER,
FRAME_TOP_RIGHT_CORNER,
+ FRAME_WINDOW,
+ FRAME_WINDOW_INACTIVE,
+ FRAME_WINDOW_INCOGNITO,
+ FRAME_WINDOW_INCOGNITO_INACTIVE,
+
FRAME_PART_BITMAP_COUNT // Must be last.
};
static const int kXPFramePartIDs[] = {
0,
IDR_CLOSE_SA, IDR_CLOSE_SA_H, IDR_CLOSE_SA_P,
- IDR_CONSTRAINED_BOTTOM_CENTER, IDR_CONSTRAINED_BOTTOM_LEFT_CORNER,
- IDR_CONSTRAINED_BOTTOM_RIGHT_CORNER, IDR_CONSTRAINED_LEFT_SIDE,
- IDR_CONSTRAINED_RIGHT_SIDE, IDR_CONSTRAINED_TOP_CENTER,
- IDR_CONSTRAINED_TOP_LEFT_CORNER, IDR_CONSTRAINED_TOP_RIGHT_CORNER,
+ IDR_WINDOW_BOTTOM_CENTER, IDR_WINDOW_BOTTOM_LEFT_CORNER,
+ IDR_WINDOW_BOTTOM_RIGHT_CORNER, IDR_WINDOW_LEFT_SIDE,
+ IDR_WINDOW_RIGHT_SIDE, IDR_WINDOW_TOP_CENTER,
+ IDR_WINDOW_TOP_LEFT_CORNER, IDR_WINDOW_TOP_RIGHT_CORNER,
+ IDR_THEME_FRAME, IDR_THEME_FRAME_INACTIVE, IDR_THEME_FRAME_INCOGNITO,
+ IDR_THEME_FRAME_INCOGNITO_INACTIVE,
0 };
static const int kVistaFramePartIDs[] = {
0,
@@ -79,14 +86,8 @@ static const int kVistaFramePartIDs[] = {
IDR_CONSTRAINED_BOTTOM_RIGHT_CORNER_V, IDR_CONSTRAINED_LEFT_SIDE_V,
IDR_CONSTRAINED_RIGHT_SIDE_V, IDR_CONSTRAINED_TOP_CENTER_V,
IDR_CONSTRAINED_TOP_LEFT_CORNER_V, IDR_CONSTRAINED_TOP_RIGHT_CORNER_V,
- 0 };
-static const int kOTRFramePartIDs[] = {
- 0,
- IDR_CLOSE_SA, IDR_CLOSE_SA_H, IDR_CLOSE_SA_P,
- IDR_WINDOW_BOTTOM_CENTER_OTR, IDR_WINDOW_BOTTOM_LEFT_CORNER_OTR,
- IDR_WINDOW_BOTTOM_RIGHT_CORNER_OTR, IDR_WINDOW_LEFT_SIDE_OTR,
- IDR_WINDOW_RIGHT_SIDE_OTR, IDR_WINDOW_TOP_CENTER_OTR,
- IDR_WINDOW_TOP_LEFT_CORNER_OTR, IDR_WINDOW_TOP_RIGHT_CORNER_OTR,
+ IDR_THEME_FRAME, IDR_THEME_FRAME_INACTIVE, IDR_THEME_FRAME_INCOGNITO,
+ IDR_THEME_FRAME_INCOGNITO_INACTIVE,
0 };
class XPWindowResources : public views::WindowResources {
@@ -149,39 +150,8 @@ class VistaWindowResources : public views::WindowResources {
DISALLOW_EVIL_CONSTRUCTORS(VistaWindowResources);
};
-class OTRWindowResources : public views::WindowResources {
- public:
- OTRWindowResources() {
- InitClass();
- }
- virtual ~OTRWindowResources() {}
-
- virtual SkBitmap* GetPartBitmap(views::FramePartBitmap part_id) const {
- return bitmaps_[part_id];
- }
-
- private:
- static void InitClass() {
- static bool initialized = false;
- if (!initialized) {
- ResourceBundle& rb = ResourceBundle::GetSharedInstance();
- for (int i = 0; i < FRAME_PART_BITMAP_COUNT; ++i) {
- int id = kOTRFramePartIDs[i];
- if (id != 0)
- bitmaps_[i] = rb.GetBitmapNamed(id);
- }
- initialized = true;
- }
- }
-
- static SkBitmap* bitmaps_[FRAME_PART_BITMAP_COUNT];
-
- DISALLOW_EVIL_CONSTRUCTORS(OTRWindowResources);
-};
-
SkBitmap* XPWindowResources::bitmaps_[];
SkBitmap* VistaWindowResources::bitmaps_[];
-SkBitmap* OTRWindowResources::bitmaps_[];
////////////////////////////////////////////////////////////////////////////////
// ConstrainedWindowFrameView
@@ -299,7 +269,6 @@ const int kTitleCaptionSpacing = 5;
const int kCaptionTopSpacing = 1;
const SkColor kContentsBorderShadow = SkColorSetARGB(51, 0, 0, 0);
-const SkColor kContentsBorderColor = SkColorSetRGB(219, 235, 255);
}
////////////////////////////////////////////////////////////////////////////////
@@ -479,6 +448,29 @@ void ConstrainedWindowFrameView::PaintFrameBorder(ChromeCanvas* canvas) {
resources_->GetPartBitmap(FRAME_BOTTOM_RIGHT_CORNER);
SkBitmap* bottom_edge = resources_->GetPartBitmap(FRAME_BOTTOM_EDGE);
+ ResourceBundle& rb = ResourceBundle::GetSharedInstance();
+ SkBitmap* theme_frame = rb.GetBitmapNamed(IDR_THEME_FRAME);
+ SkColor frame_color = ResourceBundle::frame_color;
+
+ // Fill with the frame color first so we have a constant background for
+ // areas not covered by the theme image.
+ canvas->FillRectInt(frame_color, 0, 0, width(), theme_frame->height());
+ // Now fill down the sides
+ canvas->FillRectInt(frame_color,
+ 0, theme_frame->height(),
+ left_edge->width(), height() - theme_frame->height());
+ canvas->FillRectInt(frame_color,
+ width() - right_edge->width(), theme_frame->height(),
+ right_edge->width(), height() - theme_frame->height());
+ // Now fill the bottom area.
+ canvas->FillRectInt(frame_color,
+ left_edge->width(), height() - bottom_edge->height(),
+ width() - left_edge->width() - right_edge->width(),
+ bottom_edge->height());
+
+ // Draw the theme frame.
+ canvas->TileImageInt(*theme_frame, 0, 0, width(), theme_frame->height());
+
// Top.
canvas->DrawBitmapInt(*top_left_corner, 0, 0);
canvas->TileImageInt(*top_edge, top_left_corner->width(), 0,
@@ -526,7 +518,7 @@ void ConstrainedWindowFrameView::PaintClientEdge(ChromeCanvas* canvas) {
frame_shadow_bounds.y(), frame_shadow_bounds.width(),
frame_shadow_bounds.height());
- canvas->FillRectInt(kContentsBorderColor, client_edge_bounds.x(),
+ canvas->FillRectInt(ResourceBundle::toolbar_color, client_edge_bounds.x(),
client_edge_bounds.y(), client_edge_bounds.width(),
client_edge_bounds.height());
}
@@ -565,14 +557,10 @@ gfx::Rect ConstrainedWindowFrameView::CalculateClientAreaBounds(
}
void ConstrainedWindowFrameView::InitWindowResources() {
- if (container_->owner()->profile()->IsOffTheRecord()) {
- resources_.reset(new OTRWindowResources);
+ if (win_util::ShouldUseVistaFrame()) {
+ resources_.reset(new VistaWindowResources);
} else {
- if (win_util::ShouldUseVistaFrame()) {
- resources_.reset(new VistaWindowResources);
- } else {
- resources_.reset(new XPWindowResources);
- }
+ resources_.reset(new XPWindowResources);
}
}
diff --git a/chrome/browser/views/frame/browser_frame.cc b/chrome/browser/views/frame/browser_frame.cc
index c022b6a..0d39521 100644
--- a/chrome/browser/views/frame/browser_frame.cc
+++ b/chrome/browser/views/frame/browser_frame.cc
@@ -9,6 +9,7 @@
#include "app/resource_bundle.h"
#include "app/win_util.h"
+#include "chrome/browser/profile.h"
#include "chrome/browser/browser_list.h"
#include "chrome/browser/views/frame/browser_root_view.h"
#include "chrome/browser/views/frame/browser_view.h"
@@ -23,10 +24,11 @@ static const int kClientEdgeThickness = 3;
///////////////////////////////////////////////////////////////////////////////
// BrowserFrame, public:
-BrowserFrame::BrowserFrame(BrowserView* browser_view)
+BrowserFrame::BrowserFrame(BrowserView* browser_view, Profile* profile)
: WindowWin(browser_view),
browser_view_(browser_view),
- frame_initialized_(false) {
+ frame_initialized_(false),
+ profile_(profile) {
browser_view_->set_frame(this);
GetNonClientView()->SetFrameView(CreateFrameViewForWindow());
// Don't focus anything on creation, selecting a tab will set the focus.
@@ -60,6 +62,14 @@ void BrowserFrame::UpdateThrobber(bool running) {
browser_frame_view_->UpdateThrobber(running);
}
+ThemeProvider* BrowserFrame::GetThemeProvider() const {
+ return profile_->GetThemeProvider();
+}
+
+ThemeProvider* BrowserFrame::GetDialogThemeProvider() const {
+ return profile_->GetThemeProvider();
+}
+
#if defined(FRAME_WINDOW)
void BrowserFrame::SetBrowserNonClientFrameView(
BrowserNonClientFrameView* view) {
@@ -221,7 +231,6 @@ void BrowserFrame::UpdateFrameAfterFrameChange() {
UpdateDWMFrame();
}
-
views::RootView* BrowserFrame::CreateRootView() {
return new BrowserRootView(this);
}
diff --git a/chrome/browser/views/frame/browser_frame.h b/chrome/browser/views/frame/browser_frame.h
index 140f25e..792e90e 100644
--- a/chrome/browser/views/frame/browser_frame.h
+++ b/chrome/browser/views/frame/browser_frame.h
@@ -10,6 +10,7 @@
class AeroGlassNonClientView;
class BrowserView;
class NonClientFrameView;
+class Profile;
class TabStrip;
// A specialization of the NonClientFrameView object that provides additional
@@ -34,7 +35,7 @@ class BrowserNonClientFrameView : public views::NonClientFrameView {
//
class BrowserFrame : public views::WindowWin {
public:
- explicit BrowserFrame(BrowserView* browser_view);
+ explicit BrowserFrame(BrowserView* browser_view, Profile* profile);
virtual ~BrowserFrame();
// Initialize the frame. Creates the Window.
@@ -51,6 +52,10 @@ class BrowserFrame : public views::WindowWin {
// Tells the frame to update the throbber.
void UpdateThrobber(bool running);
+ // Overridden from views::Widget.
+ virtual ThemeProvider* GetThemeProvider() const;
+ virtual ThemeProvider* GetDialogThemeProvider() const;
+
BrowserView* browser_view() const { return browser_view_; }
#if defined(FRAME_WINDOW)
@@ -92,6 +97,8 @@ class BrowserFrame : public views::WindowWin {
bool frame_initialized_;
+ Profile* profile_;
+
DISALLOW_EVIL_CONSTRUCTORS(BrowserFrame);
};
diff --git a/chrome/browser/views/frame/browser_view.cc b/chrome/browser/views/frame/browser_view.cc
index a3acb80..f2f24b5 100644
--- a/chrome/browser/views/frame/browser_view.cc
+++ b/chrome/browser/views/frame/browser_view.cc
@@ -855,6 +855,11 @@ void BrowserView::ShowHTMLDialog(HtmlDialogUIDelegate* delegate,
html_view->window()->Show();
}
+void BrowserView::UserChangedTheme() {
+ frame_->GetRootView()->ThemeChanged();
+ frame_->GetRootView()->SchedulePaint();
+}
+
///////////////////////////////////////////////////////////////////////////////
// BrowserView, BrowserWindowTesting implementation:
@@ -1662,7 +1667,7 @@ void BrowserView::InitClass() {
// static
BrowserWindow* BrowserWindow::CreateBrowserWindow(Browser* browser) {
BrowserView* browser_view = new BrowserView(browser);
- (new BrowserFrame(browser_view))->Init();
+ (new BrowserFrame(browser_view, browser->profile()))->Init();
return browser_view;
}
diff --git a/chrome/browser/views/frame/browser_view.h b/chrome/browser/views/frame/browser_view.h
index 7357333..40afbc5 100644
--- a/chrome/browser/views/frame/browser_view.h
+++ b/chrome/browser/views/frame/browser_view.h
@@ -201,6 +201,7 @@ class BrowserView : public BrowserWindow,
virtual void ConfirmBrowserCloseWithPendingDownloads();
virtual void ShowHTMLDialog(HtmlDialogUIDelegate* delegate,
void* parent_window);
+ virtual void UserChangedTheme();
// Overridden from BrowserWindowTesting:
virtual BookmarkBarView* GetBookmarkBarView() const;
diff --git a/chrome/browser/views/frame/glass_browser_frame_view.cc b/chrome/browser/views/frame/glass_browser_frame_view.cc
index bbebd3a..6eaf547 100644
--- a/chrome/browser/views/frame/glass_browser_frame_view.cc
+++ b/chrome/browser/views/frame/glass_browser_frame_view.cc
@@ -6,66 +6,15 @@
#include "app/gfx/chrome_canvas.h"
#include "app/resource_bundle.h"
+#include "app/theme_provider.h"
+#include "chrome/browser/browser_theme_provider.h"
#include "chrome/browser/views/frame/browser_view.h"
#include "chrome/browser/views/tabs/tab_strip.h"
#include "grit/theme_resources.h"
#include "views/window/client_view.h"
#include "views/window/window_resources.h"
-// An enumeration of bitmap resources used by this window.
-enum {
- // Client Edge Border.
- FRAME_CLIENT_EDGE_TOP_LEFT,
- FRAME_CLIENT_EDGE_TOP,
- FRAME_CLIENT_EDGE_TOP_RIGHT,
- FRAME_CLIENT_EDGE_RIGHT,
- FRAME_CLIENT_EDGE_BOTTOM_RIGHT,
- FRAME_CLIENT_EDGE_BOTTOM,
- FRAME_CLIENT_EDGE_BOTTOM_LEFT,
- FRAME_CLIENT_EDGE_LEFT,
-
- FRAME_PART_BITMAP_COUNT // Must be last.
-};
-
-class GlassBrowserWindowResources {
- public:
- GlassBrowserWindowResources() {
- InitClass();
- }
- virtual ~GlassBrowserWindowResources() { }
-
- virtual SkBitmap* GetPartBitmap(views::FramePartBitmap part) const {
- return standard_frame_bitmaps_[part];
- }
-
- private:
- static void InitClass() {
- static bool initialized = false;
- if (!initialized) {
- static const int kFramePartBitmapIds[] = {
- IDR_CONTENT_TOP_LEFT_CORNER, IDR_CONTENT_TOP_CENTER,
- IDR_CONTENT_TOP_RIGHT_CORNER, IDR_CONTENT_RIGHT_SIDE,
- IDR_CONTENT_BOTTOM_RIGHT_CORNER, IDR_CONTENT_BOTTOM_CENTER,
- IDR_CONTENT_BOTTOM_LEFT_CORNER, IDR_CONTENT_LEFT_SIDE,
- };
-
- ResourceBundle& rb = ResourceBundle::GetSharedInstance();
- for (int i = 0; i < FRAME_PART_BITMAP_COUNT; ++i)
- standard_frame_bitmaps_[i] = rb.GetBitmapNamed(kFramePartBitmapIds[i]);
-
- initialized = true;
- }
- }
-
- static SkBitmap* standard_frame_bitmaps_[FRAME_PART_BITMAP_COUNT];
-
- DISALLOW_EVIL_CONSTRUCTORS(GlassBrowserWindowResources);
-};
-
// static
-SkBitmap* GlassBrowserWindowResources::standard_frame_bitmaps_[];
-
-GlassBrowserWindowResources* GlassBrowserFrameView::resources_ = NULL;
SkBitmap* GlassBrowserFrameView::distributor_logo_ = NULL;
HICON GlassBrowserFrameView::throbber_icons_[
GlassBrowserFrameView::kThrobberIconCount];
@@ -281,23 +230,34 @@ void GlassBrowserFrameView::PaintDistributorLogo(ChromeCanvas* canvas) {
}
void GlassBrowserFrameView::PaintToolbarBackground(ChromeCanvas* canvas) {
+ ThemeProvider* tp = GetThemeProvider();
+
gfx::Rect toolbar_bounds(browser_view_->GetToolbarBounds());
gfx::Point toolbar_origin(toolbar_bounds.origin());
View::ConvertPointToView(frame_->GetClientView(), this, &toolbar_origin);
toolbar_bounds.set_origin(toolbar_origin);
+ SkBitmap* theme_toolbar = tp->GetBitmapNamed(IDR_THEME_TOOLBAR);
+
+ // Draw the toolbar background, setting src_y of the paint to the tab
+ // strip height as the toolbar background begins at the top of the tabs.
+ canvas->TileImageInt(*theme_toolbar,
+ 0, browser_view_->GetTabStripHeight() - 1,
+ toolbar_bounds.x() - 1, toolbar_bounds.y() + 2,
+ toolbar_bounds.width() + 2, theme_toolbar->height());
+
SkBitmap* toolbar_left =
- resources_->GetPartBitmap(FRAME_CLIENT_EDGE_TOP_LEFT);
+ tp->GetBitmapNamed(IDR_CONTENT_TOP_LEFT_CORNER);
canvas->DrawBitmapInt(*toolbar_left,
toolbar_bounds.x() - toolbar_left->width(),
toolbar_bounds.y());
SkBitmap* toolbar_center =
- resources_->GetPartBitmap(FRAME_CLIENT_EDGE_TOP);
+ tp->GetBitmapNamed(IDR_CONTENT_TOP_CENTER);
canvas->TileImageInt(*toolbar_center, toolbar_bounds.x(), toolbar_bounds.y(),
toolbar_bounds.width(), toolbar_center->height());
- canvas->DrawBitmapInt(*resources_->GetPartBitmap(FRAME_CLIENT_EDGE_TOP_RIGHT),
+ canvas->DrawBitmapInt(*tp->GetBitmapNamed(IDR_CONTENT_TOP_RIGHT_CORNER),
toolbar_bounds.right(), toolbar_bounds.y());
}
@@ -314,35 +274,50 @@ void GlassBrowserFrameView::PaintOTRAvatar(ChromeCanvas* canvas) {
}
void GlassBrowserFrameView::PaintRestoredClientEdge(ChromeCanvas* canvas) {
+ ResourceBundle& rb = ResourceBundle::GetSharedInstance();
+
// The client edges start below the toolbar upper corner images regardless
// of how tall the toolbar itself is.
int client_area_top =
frame_->GetClientView()->y() + browser_view_->GetToolbarBounds().y() +
- resources_->GetPartBitmap(FRAME_CLIENT_EDGE_TOP_LEFT)->height();
+ rb.GetBitmapNamed(IDR_CONTENT_TOP_LEFT_CORNER)->height();
gfx::Rect client_area_bounds = CalculateClientAreaBounds(width(), height());
int client_area_bottom =
std::max(client_area_top, height() - NonClientBorderThickness());
int client_area_height = client_area_bottom - client_area_top;
- SkBitmap* right = resources_->GetPartBitmap(FRAME_CLIENT_EDGE_RIGHT);
+ SkBitmap* right = rb.GetBitmapNamed(IDR_CONTENT_RIGHT_SIDE);
canvas->TileImageInt(*right, client_area_bounds.right(), client_area_top,
right->width(), client_area_height);
+ // Draw the toolbar color so that the one pixel areas down the sides
+ // show the right color even if not covered by the toolbar image.
+ canvas->DrawRectInt(GetThemeProvider()->
+ GetColor(BrowserThemeProvider::COLOR_TOOLBAR),
+ client_area_bounds.x() - 1, client_area_top,
+ client_area_bounds.width() + 1, client_area_bottom - client_area_top);
+
+ // Draw the content/toolbar separator.
+ canvas->DrawLineInt(ResourceBundle::toolbar_separator_color,
+ client_area_bounds.x(), client_area_top,
+ client_area_bounds.x() + client_area_bounds.width(),
+ client_area_top);
+
canvas->DrawBitmapInt(
- *resources_->GetPartBitmap(FRAME_CLIENT_EDGE_BOTTOM_RIGHT),
+ *rb.GetBitmapNamed(IDR_CONTENT_BOTTOM_RIGHT_CORNER),
client_area_bounds.right(), client_area_bottom);
- SkBitmap* bottom = resources_->GetPartBitmap(FRAME_CLIENT_EDGE_BOTTOM);
+ SkBitmap* bottom = rb.GetBitmapNamed(IDR_CONTENT_BOTTOM_CENTER);
canvas->TileImageInt(*bottom, client_area_bounds.x(),
client_area_bottom, client_area_bounds.width(),
bottom->height());
SkBitmap* bottom_left =
- resources_->GetPartBitmap(FRAME_CLIENT_EDGE_BOTTOM_LEFT);
+ rb.GetBitmapNamed(IDR_CONTENT_BOTTOM_LEFT_CORNER);
canvas->DrawBitmapInt(*bottom_left,
client_area_bounds.x() - bottom_left->width(), client_area_bottom);
- SkBitmap* left = resources_->GetPartBitmap(FRAME_CLIENT_EDGE_LEFT);
+ SkBitmap* left = rb.GetBitmapNamed(IDR_CONTENT_LEFT_SIDE);
canvas->TileImageInt(*left, client_area_bounds.x() - left->width(),
client_area_top, left->width(), client_area_height);
}
@@ -437,8 +412,6 @@ void GlassBrowserFrameView::InitThrobberIcons() {
void GlassBrowserFrameView::InitClass() {
static bool initialized = false;
if (!initialized) {
- resources_ = new GlassBrowserWindowResources;
-
#if defined(GOOGLE_CHROME_BUILD)
distributor_logo_ = ResourceBundle::GetSharedInstance().
GetBitmapNamed(IDR_DISTRIBUTOR_LOGO);
diff --git a/chrome/browser/views/frame/glass_browser_frame_view.h b/chrome/browser/views/frame/glass_browser_frame_view.h
index d110fe9..3a00db3 100644
--- a/chrome/browser/views/frame/glass_browser_frame_view.h
+++ b/chrome/browser/views/frame/glass_browser_frame_view.h
@@ -10,7 +10,6 @@
#include "views/window/non_client_view.h"
class BrowserView;
-class GlassBrowserWindowResources;
class SkBitmap;
class GlassBrowserFrameView : public BrowserNonClientFrameView {
@@ -95,7 +94,6 @@ class GlassBrowserFrameView : public BrowserNonClientFrameView {
static void InitClass();
static SkBitmap* distributor_logo_;
- static GlassBrowserWindowResources* resources_;
static const int kThrobberIconCount = 24;
static HICON throbber_icons_[kThrobberIconCount];
diff --git a/chrome/browser/views/frame/opaque_browser_frame_view.cc b/chrome/browser/views/frame/opaque_browser_frame_view.cc
index 10c604c..317e307 100644
--- a/chrome/browser/views/frame/opaque_browser_frame_view.cc
+++ b/chrome/browser/views/frame/opaque_browser_frame_view.cc
@@ -9,7 +9,9 @@
#include "app/gfx/path.h"
#include "app/l10n_util.h"
#include "app/resource_bundle.h"
+#include "app/theme_provider.h"
#include "app/win_util.h"
+#include "chrome/browser/browser_theme_provider.h"
#include "chrome/browser/views/frame/browser_frame.h"
#include "chrome/browser/views/frame/browser_view.h"
#include "chrome/browser/views/tabs/tab_strip.h"
@@ -20,244 +22,7 @@
#include "views/widget/root_view.h"
#include "views/window/window_resources.h"
-// An enumeration of bitmap resources used by this window.
-enum {
- // Window Controls.
- FRAME_CLOSE_BUTTON_ICON,
- FRAME_CLOSE_BUTTON_ICON_H,
- FRAME_CLOSE_BUTTON_ICON_P,
- FRAME_CLOSE_BUTTON_ICON_SA,
- FRAME_CLOSE_BUTTON_ICON_SA_H,
- FRAME_CLOSE_BUTTON_ICON_SA_P,
- FRAME_RESTORE_BUTTON_ICON,
- FRAME_RESTORE_BUTTON_ICON_H,
- FRAME_RESTORE_BUTTON_ICON_P,
- FRAME_MAXIMIZE_BUTTON_ICON,
- FRAME_MAXIMIZE_BUTTON_ICON_H,
- FRAME_MAXIMIZE_BUTTON_ICON_P,
- FRAME_MINIMIZE_BUTTON_ICON,
- FRAME_MINIMIZE_BUTTON_ICON_H,
- FRAME_MINIMIZE_BUTTON_ICON_P,
-
- // Window Frame Border.
- FRAME_BOTTOM_EDGE,
- FRAME_BOTTOM_LEFT_CORNER,
- FRAME_BOTTOM_RIGHT_CORNER,
- FRAME_LEFT_EDGE,
- FRAME_RIGHT_EDGE,
- FRAME_TOP_EDGE,
- FRAME_TOP_LEFT_CORNER,
- FRAME_TOP_RIGHT_CORNER,
-
- // Client Edge Border.
- FRAME_CLIENT_EDGE_TOP_LEFT,
- FRAME_CLIENT_EDGE_TOP,
- FRAME_CLIENT_EDGE_TOP_RIGHT,
- FRAME_CLIENT_EDGE_RIGHT,
- FRAME_CLIENT_EDGE_BOTTOM_RIGHT,
- FRAME_CLIENT_EDGE_BOTTOM,
- FRAME_CLIENT_EDGE_BOTTOM_LEFT,
- FRAME_CLIENT_EDGE_LEFT,
-
- // No-toolbar client edge.
- FRAME_NO_TOOLBAR_TOP_LEFT,
- FRAME_NO_TOOLBAR_TOP_CENTER,
- FRAME_NO_TOOLBAR_TOP_RIGHT,
-
- FRAME_PART_BITMAP_COUNT // Must be last.
-};
-
-class ActiveWindowResources : public views::WindowResources {
- public:
- ActiveWindowResources() {
- InitClass();
- }
- virtual ~ActiveWindowResources() { }
-
- // WindowResources implementation:
- virtual SkBitmap* GetPartBitmap(views::FramePartBitmap part) const {
- return standard_frame_bitmaps_[part];
- }
-
- private:
- static void InitClass() {
- static bool initialized = false;
- if (!initialized) {
- static const int kFramePartBitmapIds[] = {
- IDR_CLOSE, IDR_CLOSE_H, IDR_CLOSE_P,
- IDR_CLOSE_SA, IDR_CLOSE_SA_H, IDR_CLOSE_SA_P,
- IDR_RESTORE, IDR_RESTORE_H, IDR_RESTORE_P,
- IDR_MAXIMIZE, IDR_MAXIMIZE_H, IDR_MAXIMIZE_P,
- IDR_MINIMIZE, IDR_MINIMIZE_H, IDR_MINIMIZE_P,
- IDR_WINDOW_BOTTOM_CENTER, IDR_WINDOW_BOTTOM_LEFT_CORNER,
- IDR_WINDOW_BOTTOM_RIGHT_CORNER, IDR_WINDOW_LEFT_SIDE,
- IDR_WINDOW_RIGHT_SIDE, IDR_WINDOW_TOP_CENTER,
- IDR_WINDOW_TOP_LEFT_CORNER, IDR_WINDOW_TOP_RIGHT_CORNER,
- IDR_CONTENT_TOP_LEFT_CORNER, IDR_CONTENT_TOP_CENTER,
- IDR_CONTENT_TOP_RIGHT_CORNER, IDR_CONTENT_RIGHT_SIDE,
- IDR_CONTENT_BOTTOM_RIGHT_CORNER, IDR_CONTENT_BOTTOM_CENTER,
- IDR_CONTENT_BOTTOM_LEFT_CORNER, IDR_CONTENT_LEFT_SIDE,
- IDR_APP_TOP_LEFT, IDR_APP_TOP_CENTER, IDR_APP_TOP_RIGHT,
- };
-
- ResourceBundle& rb = ResourceBundle::GetSharedInstance();
- for (int i = 0; i < FRAME_PART_BITMAP_COUNT; ++i)
- standard_frame_bitmaps_[i] = rb.GetBitmapNamed(kFramePartBitmapIds[i]);
- initialized = true;
- }
- }
-
- static SkBitmap* standard_frame_bitmaps_[FRAME_PART_BITMAP_COUNT];
-
- DISALLOW_EVIL_CONSTRUCTORS(ActiveWindowResources);
-};
-
-class InactiveWindowResources : public views::WindowResources {
- public:
- InactiveWindowResources() {
- InitClass();
- }
- virtual ~InactiveWindowResources() { }
-
- // WindowResources implementation:
- virtual SkBitmap* GetPartBitmap(views::FramePartBitmap part) const {
- return standard_frame_bitmaps_[part];
- }
-
- private:
- static void InitClass() {
- static bool initialized = false;
- if (!initialized) {
- static const int kFramePartBitmapIds[] = {
- IDR_CLOSE, IDR_CLOSE_H, IDR_CLOSE_P,
- IDR_CLOSE_SA, IDR_CLOSE_SA_H, IDR_CLOSE_SA_P,
- IDR_RESTORE, IDR_RESTORE_H, IDR_RESTORE_P,
- IDR_MAXIMIZE, IDR_MAXIMIZE_H, IDR_MAXIMIZE_P,
- IDR_MINIMIZE, IDR_MINIMIZE_H, IDR_MINIMIZE_P,
- IDR_DEWINDOW_BOTTOM_CENTER, IDR_DEWINDOW_BOTTOM_LEFT_CORNER,
- IDR_DEWINDOW_BOTTOM_RIGHT_CORNER, IDR_DEWINDOW_LEFT_SIDE,
- IDR_DEWINDOW_RIGHT_SIDE, IDR_DEWINDOW_TOP_CENTER,
- IDR_DEWINDOW_TOP_LEFT_CORNER, IDR_DEWINDOW_TOP_RIGHT_CORNER,
- IDR_CONTENT_TOP_LEFT_CORNER, IDR_CONTENT_TOP_CENTER,
- IDR_CONTENT_TOP_RIGHT_CORNER, IDR_CONTENT_RIGHT_SIDE,
- IDR_CONTENT_BOTTOM_RIGHT_CORNER, IDR_CONTENT_BOTTOM_CENTER,
- IDR_CONTENT_BOTTOM_LEFT_CORNER, IDR_CONTENT_LEFT_SIDE,
- IDR_APP_TOP_LEFT, IDR_APP_TOP_CENTER, IDR_APP_TOP_RIGHT,
- };
-
- ResourceBundle& rb = ResourceBundle::GetSharedInstance();
- for (int i = 0; i < FRAME_PART_BITMAP_COUNT; ++i)
- standard_frame_bitmaps_[i] = rb.GetBitmapNamed(kFramePartBitmapIds[i]);
- initialized = true;
- }
- }
-
- static SkBitmap* standard_frame_bitmaps_[FRAME_PART_BITMAP_COUNT];
-
- DISALLOW_EVIL_CONSTRUCTORS(InactiveWindowResources);
-};
-
-class OTRActiveWindowResources : public views::WindowResources {
- public:
- OTRActiveWindowResources() {
- InitClass();
- }
- virtual ~OTRActiveWindowResources() { }
-
- // WindowResources implementation:
- virtual SkBitmap* GetPartBitmap(views::FramePartBitmap part) const {
- return standard_frame_bitmaps_[part];
- }
-
- private:
- static void InitClass() {
- static bool initialized = false;
- if (!initialized) {
- static const int kFramePartBitmapIds[] = {
- IDR_CLOSE, IDR_CLOSE_H, IDR_CLOSE_P,
- IDR_CLOSE_SA, IDR_CLOSE_SA_H, IDR_CLOSE_SA_P,
- IDR_RESTORE, IDR_RESTORE_H, IDR_RESTORE_P,
- IDR_MAXIMIZE, IDR_MAXIMIZE_H, IDR_MAXIMIZE_P,
- IDR_MINIMIZE, IDR_MINIMIZE_H, IDR_MINIMIZE_P,
- IDR_WINDOW_BOTTOM_CENTER_OTR, IDR_WINDOW_BOTTOM_LEFT_CORNER_OTR,
- IDR_WINDOW_BOTTOM_RIGHT_CORNER_OTR, IDR_WINDOW_LEFT_SIDE_OTR,
- IDR_WINDOW_RIGHT_SIDE_OTR, IDR_WINDOW_TOP_CENTER_OTR,
- IDR_WINDOW_TOP_LEFT_CORNER_OTR, IDR_WINDOW_TOP_RIGHT_CORNER_OTR,
- IDR_CONTENT_TOP_LEFT_CORNER, IDR_CONTENT_TOP_CENTER,
- IDR_CONTENT_TOP_RIGHT_CORNER, IDR_CONTENT_RIGHT_SIDE,
- IDR_CONTENT_BOTTOM_RIGHT_CORNER, IDR_CONTENT_BOTTOM_CENTER,
- IDR_CONTENT_BOTTOM_LEFT_CORNER, IDR_CONTENT_LEFT_SIDE,
- IDR_APP_TOP_LEFT, IDR_APP_TOP_CENTER, IDR_APP_TOP_RIGHT,
- };
-
- ResourceBundle& rb = ResourceBundle::GetSharedInstance();
- for (int i = 0; i < FRAME_PART_BITMAP_COUNT; ++i)
- standard_frame_bitmaps_[i] = rb.GetBitmapNamed(kFramePartBitmapIds[i]);
- initialized = true;
- }
- }
-
- static SkBitmap* standard_frame_bitmaps_[FRAME_PART_BITMAP_COUNT];
-
- DISALLOW_EVIL_CONSTRUCTORS(OTRActiveWindowResources);
-};
-
-class OTRInactiveWindowResources : public views::WindowResources {
- public:
- OTRInactiveWindowResources() {
- InitClass();
- }
- virtual ~OTRInactiveWindowResources() { }
-
- // WindowResources implementation:
- virtual SkBitmap* GetPartBitmap(views::FramePartBitmap part) const {
- return standard_frame_bitmaps_[part];
- }
-
- private:
- static void InitClass() {
- static bool initialized = false;
- if (!initialized) {
- static const int kFramePartBitmapIds[] = {
- IDR_CLOSE, IDR_CLOSE_H, IDR_CLOSE_P,
- IDR_CLOSE_SA, IDR_CLOSE_SA_H, IDR_CLOSE_SA_P,
- IDR_RESTORE, IDR_RESTORE_H, IDR_RESTORE_P,
- IDR_MAXIMIZE, IDR_MAXIMIZE_H, IDR_MAXIMIZE_P,
- IDR_MINIMIZE, IDR_MINIMIZE_H, IDR_MINIMIZE_P,
- IDR_DEWINDOW_BOTTOM_CENTER_OTR, IDR_DEWINDOW_BOTTOM_LEFT_CORNER_OTR,
- IDR_DEWINDOW_BOTTOM_RIGHT_CORNER_OTR, IDR_DEWINDOW_LEFT_SIDE_OTR,
- IDR_DEWINDOW_RIGHT_SIDE_OTR, IDR_DEWINDOW_TOP_CENTER_OTR,
- IDR_DEWINDOW_TOP_LEFT_CORNER_OTR,
- IDR_DEWINDOW_TOP_RIGHT_CORNER_OTR,
- IDR_CONTENT_TOP_LEFT_CORNER, IDR_CONTENT_TOP_CENTER,
- IDR_CONTENT_TOP_RIGHT_CORNER, IDR_CONTENT_RIGHT_SIDE,
- IDR_CONTENT_BOTTOM_RIGHT_CORNER, IDR_CONTENT_BOTTOM_CENTER,
- IDR_CONTENT_BOTTOM_LEFT_CORNER, IDR_CONTENT_LEFT_SIDE,
- IDR_APP_TOP_LEFT, IDR_APP_TOP_CENTER, IDR_APP_TOP_RIGHT,
- };
-
- ResourceBundle& rb = ResourceBundle::GetSharedInstance();
- for (int i = 0; i < FRAME_PART_BITMAP_COUNT; ++i)
- standard_frame_bitmaps_[i] = rb.GetBitmapNamed(kFramePartBitmapIds[i]);
- initialized = true;
- }
- }
-
- static SkBitmap* standard_frame_bitmaps_[FRAME_PART_BITMAP_COUNT];
-
- DISALLOW_EVIL_CONSTRUCTORS(OTRInactiveWindowResources);
-};
-
// static
-SkBitmap* ActiveWindowResources::standard_frame_bitmaps_[];
-SkBitmap* InactiveWindowResources::standard_frame_bitmaps_[];
-SkBitmap* OTRActiveWindowResources::standard_frame_bitmaps_[];
-SkBitmap* OTRInactiveWindowResources::standard_frame_bitmaps_[];
-
-views::WindowResources* OpaqueBrowserFrameView::active_resources_ = NULL;
-views::WindowResources* OpaqueBrowserFrameView::inactive_resources_ = NULL;
-views::WindowResources* OpaqueBrowserFrameView::active_otr_resources_ = NULL;
-views::WindowResources* OpaqueBrowserFrameView::inactive_otr_resources_ = NULL;
SkBitmap* OpaqueBrowserFrameView::distributor_logo_ = NULL;
ChromeFont* OpaqueBrowserFrameView::title_font_ = NULL;
@@ -341,68 +106,57 @@ OpaqueBrowserFrameView::OpaqueBrowserFrameView(BrowserFrame* frame,
frame_(frame),
browser_view_(browser_view) {
InitClass();
- if (browser_view->IsOffTheRecord()) {
- if (!active_otr_resources_) {
- // Lazy load OTR resources only when we first show an OTR frame.
- active_otr_resources_ = new OTRActiveWindowResources;
- inactive_otr_resources_ = new OTRInactiveWindowResources;
- }
- current_active_resources_ = active_otr_resources_;
- current_inactive_resources_= inactive_otr_resources_;
- } else {
- current_active_resources_ = active_resources_;
- current_inactive_resources_ = inactive_resources_;
- }
- views::WindowResources* resources = current_active_resources_;
+ ThemeProvider* tp = frame_->GetThemeProvider();
+
minimize_button_->SetImage(
views::CustomButton::BS_NORMAL,
- resources->GetPartBitmap(FRAME_MINIMIZE_BUTTON_ICON));
+ tp->GetBitmapNamed(IDR_MINIMIZE));
minimize_button_->SetImage(
views::CustomButton::BS_HOT,
- resources->GetPartBitmap(FRAME_MINIMIZE_BUTTON_ICON_H));
+ tp->GetBitmapNamed(IDR_MINIMIZE_H));
minimize_button_->SetImage(
views::CustomButton::BS_PUSHED,
- resources->GetPartBitmap(FRAME_MINIMIZE_BUTTON_ICON_P));
+ tp->GetBitmapNamed(IDR_MINIMIZE_P));
minimize_button_->SetAccessibleName(
l10n_util::GetString(IDS_ACCNAME_MINIMIZE));
AddChildView(minimize_button_);
maximize_button_->SetImage(
views::CustomButton::BS_NORMAL,
- resources->GetPartBitmap(FRAME_MAXIMIZE_BUTTON_ICON));
+ tp->GetBitmapNamed(IDR_MAXIMIZE));
maximize_button_->SetImage(
views::CustomButton::BS_HOT,
- resources->GetPartBitmap(FRAME_MAXIMIZE_BUTTON_ICON_H));
+ tp->GetBitmapNamed(IDR_MAXIMIZE_H));
maximize_button_->SetImage(
views::CustomButton::BS_PUSHED,
- resources->GetPartBitmap(FRAME_MAXIMIZE_BUTTON_ICON_P));
+ tp->GetBitmapNamed(IDR_MAXIMIZE_P));
maximize_button_->SetAccessibleName(
l10n_util::GetString(IDS_ACCNAME_MAXIMIZE));
AddChildView(maximize_button_);
restore_button_->SetImage(
views::CustomButton::BS_NORMAL,
- resources->GetPartBitmap(FRAME_RESTORE_BUTTON_ICON));
+ tp->GetBitmapNamed(IDR_RESTORE));
restore_button_->SetImage(
views::CustomButton::BS_HOT,
- resources->GetPartBitmap(FRAME_RESTORE_BUTTON_ICON_H));
+ tp->GetBitmapNamed(IDR_RESTORE_H));
restore_button_->SetImage(
views::CustomButton::BS_PUSHED,
- resources->GetPartBitmap(FRAME_RESTORE_BUTTON_ICON_P));
+ tp->GetBitmapNamed(IDR_RESTORE_P));
restore_button_->SetAccessibleName(
l10n_util::GetString(IDS_ACCNAME_RESTORE));
AddChildView(restore_button_);
close_button_->SetImage(
views::CustomButton::BS_NORMAL,
- resources->GetPartBitmap(FRAME_CLOSE_BUTTON_ICON));
+ tp->GetBitmapNamed(IDR_CLOSE));
close_button_->SetImage(
views::CustomButton::BS_HOT,
- resources->GetPartBitmap(FRAME_CLOSE_BUTTON_ICON_H));
+ tp->GetBitmapNamed(IDR_CLOSE_H));
close_button_->SetImage(
views::CustomButton::BS_PUSHED,
- resources->GetPartBitmap(FRAME_CLOSE_BUTTON_ICON_P));
+ tp->GetBitmapNamed(IDR_CLOSE_P));
close_button_->SetAccessibleName(l10n_util::GetString(IDS_ACCNAME_CLOSE));
AddChildView(close_button_);
@@ -753,17 +507,61 @@ int OpaqueBrowserFrameView::IconSize(int* title_top_spacing_ptr,
}
void OpaqueBrowserFrameView::PaintRestoredFrameBorder(ChromeCanvas* canvas) {
- SkBitmap* top_left_corner = resources()->GetPartBitmap(FRAME_TOP_LEFT_CORNER);
+ ThemeProvider* tp = GetThemeProvider();
+
+ SkBitmap* top_left_corner = tp->GetBitmapNamed(IDR_WINDOW_TOP_LEFT_CORNER);
SkBitmap* top_right_corner =
- resources()->GetPartBitmap(FRAME_TOP_RIGHT_CORNER);
- SkBitmap* top_edge = resources()->GetPartBitmap(FRAME_TOP_EDGE);
- SkBitmap* right_edge = resources()->GetPartBitmap(FRAME_RIGHT_EDGE);
- SkBitmap* left_edge = resources()->GetPartBitmap(FRAME_LEFT_EDGE);
+ tp->GetBitmapNamed(IDR_WINDOW_TOP_RIGHT_CORNER);
+ SkBitmap* top_edge = tp->GetBitmapNamed(IDR_WINDOW_TOP_CENTER);
+ SkBitmap* right_edge = tp->GetBitmapNamed(IDR_WINDOW_RIGHT_SIDE);
+ SkBitmap* left_edge = tp->GetBitmapNamed(IDR_WINDOW_LEFT_SIDE);
SkBitmap* bottom_left_corner =
- resources()->GetPartBitmap(FRAME_BOTTOM_LEFT_CORNER);
+ tp->GetBitmapNamed(IDR_WINDOW_BOTTOM_LEFT_CORNER);
SkBitmap* bottom_right_corner =
- resources()->GetPartBitmap(FRAME_BOTTOM_RIGHT_CORNER);
- SkBitmap* bottom_edge = resources()->GetPartBitmap(FRAME_BOTTOM_EDGE);
+ tp->GetBitmapNamed(IDR_WINDOW_BOTTOM_RIGHT_CORNER);
+ SkBitmap* bottom_edge = tp->GetBitmapNamed(IDR_WINDOW_BOTTOM_CENTER);
+
+
+ // Window frame mode and color
+ SkBitmap* theme_frame;
+ SkColor frame_color;
+ if (!browser_view_->IsOffTheRecord()) {
+ if (frame_->IsActive()) {
+ theme_frame = tp->GetBitmapNamed(IDR_THEME_FRAME);
+ frame_color = tp->GetColor(BrowserThemeProvider::COLOR_FRAME);
+ } else {
+ theme_frame = tp->GetBitmapNamed(IDR_THEME_FRAME_INACTIVE);
+ frame_color = tp->GetColor(BrowserThemeProvider::COLOR_FRAME_INACTIVE);
+ }
+ } else {
+ if (frame_->IsActive()) {
+ theme_frame = tp->GetBitmapNamed(IDR_THEME_FRAME_INCOGNITO);
+ frame_color = tp->GetColor(BrowserThemeProvider::COLOR_FRAME_INCOGNITO);
+ } else {
+ theme_frame = tp->GetBitmapNamed(IDR_THEME_FRAME_INCOGNITO_INACTIVE);
+ frame_color = tp->GetColor(
+ BrowserThemeProvider::COLOR_FRAME_INCOGNITO_INACTIVE);
+ }
+ }
+
+ // Fill with the frame color first so we have a constant background for
+ // areas not covered by the theme image.
+ canvas->FillRectInt(frame_color, 0, 0, width(), theme_frame->height());
+ // Now fill down the sides
+ canvas->FillRectInt(frame_color,
+ 0, theme_frame->height(),
+ left_edge->width(), height() - theme_frame->height());
+ canvas->FillRectInt(frame_color,
+ width() - right_edge->width(), theme_frame->height(),
+ right_edge->width(), height() - theme_frame->height());
+ // Now fill the bottom area.
+ canvas->FillRectInt(frame_color,
+ left_edge->width(), height() - bottom_edge->height(),
+ width() - left_edge->width() - right_edge->width(),
+ bottom_edge->height());
+
+ // Draw the theme frame.
+ canvas->TileImageInt(*theme_frame, 0, 0, width(), theme_frame->height());
// Top.
int top_left_height = std::min(top_left_corner->height(),
@@ -804,8 +602,11 @@ void OpaqueBrowserFrameView::PaintRestoredFrameBorder(ChromeCanvas* canvas) {
height() - top_left_height - bottom_left_corner->height());
}
+
void OpaqueBrowserFrameView::PaintMaximizedFrameBorder(ChromeCanvas* canvas) {
- SkBitmap* top_edge = resources()->GetPartBitmap(FRAME_TOP_EDGE);
+ ThemeProvider* tp = GetThemeProvider();
+
+ SkBitmap* top_edge = tp->GetBitmapNamed(IDR_WINDOW_TOP_CENTER);
canvas->TileImageInt(*top_edge, 0, FrameBorderThickness(), width(),
top_edge->height());
@@ -814,7 +615,7 @@ void OpaqueBrowserFrameView::PaintMaximizedFrameBorder(ChromeCanvas* canvas) {
// edge. The graphic we use for this has a built in client edge, so we clip
// it off the bottom.
SkBitmap* top_center =
- resources()->GetPartBitmap(FRAME_NO_TOOLBAR_TOP_CENTER);
+ tp->GetBitmapNamed(IDR_APP_TOP_CENTER);
int edge_height = top_center->height() - kClientEdgeThickness;
canvas->TileImageInt(*top_center, 0,
frame_->GetClientView()->y() - edge_height, width(), edge_height);
@@ -851,19 +652,32 @@ void OpaqueBrowserFrameView::PaintToolbarBackground(ChromeCanvas* canvas) {
if (!browser_view_->IsToolbarVisible())
return;
+ ThemeProvider* tp = GetThemeProvider();
gfx::Rect toolbar_bounds(browser_view_->GetToolbarBounds());
gfx::Point toolbar_origin(toolbar_bounds.origin());
View::ConvertPointToView(frame_->GetClientView(), this, &toolbar_origin);
toolbar_bounds.set_origin(toolbar_origin);
+ int strip_height = browser_view_->GetTabStripHeight();
+ SkBitmap* theme_toolbar = tp->GetBitmapNamed(IDR_THEME_TOOLBAR);
+
+ canvas->TileImageInt(*theme_toolbar,
+ 0, strip_height - 1, // crop src
+ toolbar_bounds.x() - 1, toolbar_bounds.y() + 2,
+ toolbar_bounds.width() + 2, theme_toolbar->height());
+
+ SkBitmap* toolbar_left =
+ tp->GetBitmapNamed(IDR_CONTENT_TOP_LEFT_CORNER);
+ canvas->DrawBitmapInt(*toolbar_left,
+ toolbar_bounds.x() - toolbar_left->width(),
+ toolbar_bounds.y());
+
// Gross hack: We split the toolbar images into two pieces, since sometimes
// (popup mode) the toolbar isn't tall enough to show the whole image. The
// split happens between the top shadow section and the bottom gradient
// section so that we never break the gradient.
int split_point = kFrameShadowThickness * 2;
int bottom_y = toolbar_bounds.y() + split_point;
- SkBitmap* toolbar_left =
- resources()->GetPartBitmap(FRAME_CLIENT_EDGE_TOP_LEFT);
int bottom_edge_height =
std::min(toolbar_left->height(), toolbar_bounds.height()) - split_point;
@@ -876,7 +690,7 @@ void OpaqueBrowserFrameView::PaintToolbarBackground(ChromeCanvas* canvas) {
toolbar_left->width(), bottom_edge_height, false);
SkBitmap* toolbar_center =
- resources()->GetPartBitmap(FRAME_CLIENT_EDGE_TOP);
+ tp->GetBitmapNamed(IDR_CONTENT_TOP_CENTER);
canvas->TileImageInt(*toolbar_center, 0, 0, toolbar_bounds.x(),
toolbar_bounds.y(), toolbar_bounds.width(), split_point);
int bottom_center_height =
@@ -885,8 +699,7 @@ void OpaqueBrowserFrameView::PaintToolbarBackground(ChromeCanvas* canvas) {
toolbar_center->height() - bottom_center_height, toolbar_bounds.x(),
bottom_y, toolbar_bounds.width(), bottom_center_height);
- SkBitmap* toolbar_right =
- resources()->GetPartBitmap(FRAME_CLIENT_EDGE_TOP_RIGHT);
+ SkBitmap* toolbar_right = tp->GetBitmapNamed(IDR_CONTENT_TOP_RIGHT_CORNER);
canvas->DrawBitmapInt(*toolbar_right, 0, 0, toolbar_right->width(),
split_point, toolbar_bounds.right(), toolbar_bounds.y(),
toolbar_right->width(), split_point, false);
@@ -909,6 +722,7 @@ void OpaqueBrowserFrameView::PaintOTRAvatar(ChromeCanvas* canvas) {
}
void OpaqueBrowserFrameView::PaintRestoredClientEdge(ChromeCanvas* canvas) {
+ ThemeProvider* tp = GetThemeProvider();
int client_area_top = frame_->GetClientView()->y();
gfx::Rect client_area_bounds = CalculateClientAreaBounds(width(), height());
@@ -916,9 +730,9 @@ void OpaqueBrowserFrameView::PaintRestoredClientEdge(ChromeCanvas* canvas) {
// The client edges start below the toolbar or its corner images, whichever
// is shorter.
gfx::Rect toolbar_bounds(browser_view_->GetToolbarBounds());
- client_area_top += toolbar_bounds.y() + std::min(
- resources()->GetPartBitmap(FRAME_CLIENT_EDGE_TOP_LEFT)->height(),
- toolbar_bounds.height());
+ client_area_top += browser_view_->GetToolbarBounds().y() +
+ std::min(tp->GetBitmapNamed(IDR_CONTENT_TOP_LEFT_CORNER)->height(),
+ toolbar_bounds.height());
} else {
// The toolbar isn't going to draw a client edge for us, so draw one
// ourselves.
@@ -926,12 +740,11 @@ void OpaqueBrowserFrameView::PaintRestoredClientEdge(ChromeCanvas* canvas) {
// shorter than the top left and right bitmaps. We need their top edges to
// line up, and we need the left and right edges to start below the corners'
// bottoms.
- SkBitmap* top_left = resources()->GetPartBitmap(FRAME_NO_TOOLBAR_TOP_LEFT);
- SkBitmap* top_center =
- resources()->GetPartBitmap(FRAME_NO_TOOLBAR_TOP_CENTER);
- SkBitmap* top_right =
- resources()->GetPartBitmap(FRAME_NO_TOOLBAR_TOP_RIGHT);
+ SkBitmap* top_left = tp->GetBitmapNamed(IDR_APP_TOP_LEFT);
+ SkBitmap* top_center = tp->GetBitmapNamed(IDR_APP_TOP_CENTER);
+ SkBitmap* top_right = tp->GetBitmapNamed(IDR_APP_TOP_RIGHT);
int top_edge_y = client_area_top - top_center->height();
+
client_area_top = std::min(top_edge_y + top_left->height(),
height() - NonClientBorderThickness());
int height = client_area_top - top_edge_y;
@@ -945,28 +758,42 @@ void OpaqueBrowserFrameView::PaintRestoredClientEdge(ChromeCanvas* canvas) {
top_right->width(), height, false);
}
+
int client_area_bottom =
std::max(client_area_top, height() - NonClientBorderThickness());
int client_area_height = client_area_bottom - client_area_top;
- SkBitmap* right = resources()->GetPartBitmap(FRAME_CLIENT_EDGE_RIGHT);
+
+ // Draw the toolbar color so that the one pixel areas down the sides
+ // show the right color even if not covered by the toolbar image.
+ canvas->DrawRectInt(ResourceBundle::toolbar_color,
+ client_area_bounds.x() - kClientEdgeThickness, client_area_top,
+ client_area_bounds.width() + kClientEdgeThickness,
+ client_area_bottom - client_area_top);
+
+ // Draw the content/toolbar separator.
+ canvas->DrawLineInt(ResourceBundle::toolbar_separator_color,
+ client_area_bounds.x(), client_area_top,
+ client_area_bounds.x() + client_area_bounds.width(),
+ client_area_top);
+
+ SkBitmap* right = tp->GetBitmapNamed(IDR_CONTENT_RIGHT_SIDE);
canvas->TileImageInt(*right, client_area_bounds.right(), client_area_top,
right->width(), client_area_height);
-
canvas->DrawBitmapInt(
- *resources()->GetPartBitmap(FRAME_CLIENT_EDGE_BOTTOM_RIGHT),
+ *tp->GetBitmapNamed(IDR_CONTENT_BOTTOM_RIGHT_CORNER),
client_area_bounds.right(), client_area_bottom);
- SkBitmap* bottom = resources()->GetPartBitmap(FRAME_CLIENT_EDGE_BOTTOM);
+ SkBitmap* bottom = tp->GetBitmapNamed(IDR_CONTENT_BOTTOM_CENTER);
canvas->TileImageInt(*bottom, client_area_bounds.x(),
client_area_bottom, client_area_bounds.width(),
bottom->height());
SkBitmap* bottom_left =
- resources()->GetPartBitmap(FRAME_CLIENT_EDGE_BOTTOM_LEFT);
+ tp->GetBitmapNamed(IDR_CONTENT_BOTTOM_LEFT_CORNER);
canvas->DrawBitmapInt(*bottom_left,
client_area_bounds.x() - bottom_left->width(), client_area_bottom);
- SkBitmap* left = resources()->GetPartBitmap(FRAME_CLIENT_EDGE_LEFT);
+ SkBitmap* left = tp->GetBitmapNamed(IDR_CONTENT_LEFT_SIDE);
canvas->TileImageInt(*left, client_area_bounds.x() - left->width(),
client_area_top, left->width(), client_area_height);
}
@@ -1102,9 +929,6 @@ gfx::Rect OpaqueBrowserFrameView::CalculateClientAreaBounds(int width,
void OpaqueBrowserFrameView::InitClass() {
static bool initialized = false;
if (!initialized) {
- active_resources_ = new ActiveWindowResources;
- inactive_resources_ = new InactiveWindowResources;
-
#if defined(GOOGLE_CHROME_BUILD)
distributor_logo_ = ResourceBundle::GetSharedInstance().
GetBitmapNamed(IDR_DISTRIBUTOR_LOGO_LIGHT);
diff --git a/chrome/browser/views/frame/opaque_browser_frame_view.h b/chrome/browser/views/frame/opaque_browser_frame_view.h
index 1e917e9..08aee7e 100644
--- a/chrome/browser/views/frame/opaque_browser_frame_view.h
+++ b/chrome/browser/views/frame/opaque_browser_frame_view.h
@@ -16,7 +16,6 @@ class TabContents;
class TabStrip;
namespace views {
class ImageButton;
-class WindowResources;
}
class OpaqueBrowserFrameView : public BrowserNonClientFrameView,
@@ -118,12 +117,6 @@ class OpaqueBrowserFrameView : public BrowserNonClientFrameView,
// Returns the bounds of the client area for the specified view size.
gfx::Rect CalculateClientAreaBounds(int width, int height) const;
- // Returns the set of resources to use to paint this view.
- views::WindowResources* resources() const {
- return frame_->IsActive() || paint_as_active() ?
- current_active_resources_ : current_inactive_resources_;
- }
-
// The layout rect of the title, if visible.
gfx::Rect title_bounds_;
@@ -151,20 +144,12 @@ class OpaqueBrowserFrameView : public BrowserNonClientFrameView,
// The bounds of the ClientView.
gfx::Rect client_view_bounds_;
- // The resources currently used to paint this view.
- views::WindowResources* current_active_resources_;
- views::WindowResources* current_inactive_resources_;
-
// The accessible name of this view.
std::wstring accessible_name_;
static void InitClass();
static void InitAppWindowResources();
static SkBitmap* distributor_logo_;
- static views::WindowResources* active_resources_;
- static views::WindowResources* inactive_resources_;
- static views::WindowResources* active_otr_resources_;
- static views::WindowResources* inactive_otr_resources_;
static ChromeFont* title_font_;
DISALLOW_EVIL_CONSTRUCTORS(OpaqueBrowserFrameView);
diff --git a/chrome/browser/views/infobars/infobars.cc b/chrome/browser/views/infobars/infobars.cc
index 4a04040..6749dec 100644
--- a/chrome/browser/views/infobars/infobars.cc
+++ b/chrome/browser/views/infobars/infobars.cc
@@ -34,7 +34,6 @@ static const SkColor kBackgroundColorTop = SkColorSetRGB(255, 242, 183);
static const SkColor kBackgroundColorBottom = SkColorSetRGB(250, 230, 145);
static const int kSeparatorLineHeight = 1;
-static const SkColor kSeparatorColor = SkColorSetRGB(165, 165, 165);
namespace {
// Returns a centered y-position of a control of height specified in |prefsize|
@@ -70,7 +69,7 @@ class InfoBarBackground : public views::Background {
gradient_background_->Paint(canvas, view);
// Now paint the separator line.
- canvas->FillRectInt(kSeparatorColor, 0,
+ canvas->FillRectInt(ResourceBundle::toolbar_separator_color, 0,
view->height() - kSeparatorLineHeight, view->width(),
kSeparatorLineHeight);
}
diff --git a/chrome/browser/views/location_bar_view.cc b/chrome/browser/views/location_bar_view.cc
index c1354f1..8490c58 100644
--- a/chrome/browser/views/location_bar_view.cc
+++ b/chrome/browser/views/location_bar_view.cc
@@ -16,6 +16,7 @@
#include "chrome/browser/browser_list.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/command_updater.h"
+#include "chrome/browser/extensions/extension.h"
#include "chrome/browser/extensions/extension_browser_event_router.h"
#include "chrome/browser/extensions/extension_tabs_module.h"
#include "chrome/browser/extensions/extensions_service.h"
@@ -27,6 +28,7 @@
#include "chrome/browser/views/info_bubble.h"
#include "chrome/browser/views/first_run_bubble.h"
#include "chrome/browser/views/page_info_window.h"
+#include "chrome/common/page_action.h"
#include "grit/generated_resources.h"
#include "grit/theme_resources.h"
#include "views/background.h"
diff --git a/chrome/browser/views/options/content_page_view.cc b/chrome/browser/views/options/content_page_view.cc
index 6e3aff2..badd2f6 100644
--- a/chrome/browser/views/options/content_page_view.cc
+++ b/chrome/browser/views/options/content_page_view.cc
@@ -12,6 +12,7 @@
#include "app/gfx/chrome_canvas.h"
#include "app/l10n_util.h"
#include "app/resource_bundle.h"
+#include "base/command_line.h"
#include "base/file_util.h"
#include "base/gfx/native_theme.h"
#include "chrome/browser/browser_process.h"
@@ -19,6 +20,7 @@
#include "chrome/browser/views/options/fonts_languages_window_view.h"
#include "chrome/browser/views/options/options_group_view.h"
#include "chrome/browser/views/options/passwords_exceptions_window_view.h"
+#include "chrome/common/chrome_switches.h"
#include "chrome/common/pref_names.h"
#include "chrome/common/pref_service.h"
#include "grit/generated_resources.h"
@@ -181,6 +183,8 @@ ContentPageView::ContentPageView(Profile* profile)
passwords_neversave_radio_(NULL),
fonts_lang_group_(NULL),
fonts_and_languages_label_(NULL),
+ themes_group_(NULL),
+ themes_reset_button_(NULL),
change_content_fonts_button_(NULL),
OptionsPageView(profile) {
}
@@ -257,6 +261,9 @@ void ContentPageView::ButtonPressed(views::Button* sender) {
GetRootWindow(),
gfx::Rect(),
new FontsLanguagesWindowView(profile()))->Show();
+ } else if (sender == themes_reset_button_) {
+ UserMetricsRecordAction(L"Options_ThemesReset", profile()->GetPrefs());
+ profile()->ClearTheme();
}
}
@@ -299,6 +306,14 @@ void ContentPageView::InitControlLayout() {
layout->AddView(form_autofill_group_);
layout->AddPaddingRow(0, kRelatedControlVerticalSpacing);
+ if (CommandLine::ForCurrentProcess()->
+ HasSwitch(switches::kEnableExtensions)) {
+ layout->StartRow(0, single_column_view_set_id);
+ InitThemesGroup();
+ layout->AddView(themes_group_);
+ layout->AddPaddingRow(0, kRelatedControlVerticalSpacing);
+ }
+
// Init member prefs so we can update the controls if prefs change.
default_download_location_.Init(prefs::kDownloadDefaultDirectory,
profile()->GetPrefs(), this);
@@ -445,6 +460,39 @@ void ContentPageView::InitPasswordSavingGroup() {
true);
}
+void ContentPageView::InitFontsLangGroup() {
+ fonts_and_languages_label_ = new views::Label(
+ l10n_util::GetString(IDS_OPTIONS_FONTSETTINGS_INFO));
+ fonts_and_languages_label_->SetHorizontalAlignment(views::Label::ALIGN_LEFT);
+ fonts_and_languages_label_->SetMultiLine(true);
+ change_content_fonts_button_ = new views::NativeButton(
+ this,
+ l10n_util::GetString(IDS_OPTIONS_FONTSETTINGS_CONFIGUREFONTS_BUTTON));
+
+ using views::GridLayout;
+ using views::ColumnSet;
+
+ views::View* contents = new views::View;
+ GridLayout* layout = new GridLayout(contents);
+ contents->SetLayoutManager(layout);
+
+ const int single_column_view_set_id = 1;
+ ColumnSet* column_set = layout->AddColumnSet(single_column_view_set_id);
+ column_set->AddColumn(GridLayout::LEADING, GridLayout::CENTER, 1,
+ GridLayout::USE_PREF, 0, 0);
+
+ layout->StartRow(0, single_column_view_set_id);
+ layout->AddView(fonts_and_languages_label_);
+ layout->AddPaddingRow(0, kRelatedControlVerticalSpacing);
+ layout->StartRow(0, single_column_view_set_id);
+ layout->AddView(change_content_fonts_button_);
+
+ fonts_lang_group_ = new OptionsGroupView(
+ contents,
+ l10n_util::GetString(IDS_OPTIONS_FONTSANDLANGUAGES_GROUP_NAME),
+ L"", true);
+}
+
void ContentPageView::InitFormAutofillGroup() {
form_autofill_checkbox_ = new views::Checkbox(
l10n_util::GetString(IDS_AUTOFILL_SAVEFORMS));
@@ -468,17 +516,12 @@ void ContentPageView::InitFormAutofillGroup() {
form_autofill_group_ = new OptionsGroupView(
contents, l10n_util::GetString(IDS_AUTOFILL_SETTING_WINDOWS_GROUP_NAME),
- L"", false);
+ L"", true);
}
-void ContentPageView::InitFontsLangGroup() {
- fonts_and_languages_label_ = new views::Label(
- l10n_util::GetString(IDS_OPTIONS_FONTSETTINGS_INFO));
- fonts_and_languages_label_->SetHorizontalAlignment(views::Label::ALIGN_LEFT);
- fonts_and_languages_label_->SetMultiLine(true);
- change_content_fonts_button_ = new views::NativeButton(
- this,
- l10n_util::GetString(IDS_OPTIONS_FONTSETTINGS_CONFIGUREFONTS_BUTTON));
+void ContentPageView::InitThemesGroup() {
+ themes_reset_button_ = new views::NativeButton(this,
+ l10n_util::GetString(IDS_THEMES_RESET_BUTTON));
using views::GridLayout;
using views::ColumnSet;
@@ -490,18 +533,14 @@ void ContentPageView::InitFontsLangGroup() {
const int single_column_view_set_id = 1;
ColumnSet* column_set = layout->AddColumnSet(single_column_view_set_id);
column_set->AddColumn(GridLayout::LEADING, GridLayout::CENTER, 1,
- GridLayout::USE_PREF, 0, 0);
+ GridLayout::USE_PREF, 0, 0);
layout->StartRow(0, single_column_view_set_id);
- layout->AddView(fonts_and_languages_label_);
- layout->AddPaddingRow(0, kRelatedControlVerticalSpacing);
- layout->StartRow(0, single_column_view_set_id);
- layout->AddView(change_content_fonts_button_);
+ layout->AddView(themes_reset_button_);
- fonts_lang_group_ = new OptionsGroupView(
- contents,
- l10n_util::GetString(IDS_OPTIONS_FONTSANDLANGUAGES_GROUP_NAME),
- L"", true);
+ themes_group_ = new OptionsGroupView(
+ contents, l10n_util::GetString(IDS_THEMES_GROUP_NAME),
+ L"", false);
}
void ContentPageView::UpdateDownloadDirectoryDisplay() {
diff --git a/chrome/browser/views/options/content_page_view.h b/chrome/browser/views/options/content_page_view.h
index 7d96e00..ff133e3 100644
--- a/chrome/browser/views/options/content_page_view.h
+++ b/chrome/browser/views/options/content_page_view.h
@@ -51,8 +51,9 @@ class ContentPageView : public OptionsPageView,
// Init all the dialog controls.
void InitDownloadLocation();
void InitPasswordSavingGroup();
- void InitFormAutofillGroup();
void InitFontsLangGroup();
+ void InitFormAutofillGroup();
+ void InitThemesGroup();
// Updates the directory displayed in the default download location view with
// the current value of the pref.
@@ -71,20 +72,24 @@ class ContentPageView : public OptionsPageView,
views::RadioButton* passwords_asktosave_radio_;
views::RadioButton* passwords_neversave_radio_;
+ // Controls for the Fonts and Languages group.
+ OptionsGroupView* fonts_lang_group_;
+ views::Label* fonts_and_languages_label_;
+ views::NativeButton* change_content_fonts_button_;
+
// Controls for the Form Autofill group
OptionsGroupView* form_autofill_group_;
views::Checkbox* form_autofill_checkbox_;
+ // Controls for the Themes group
+ OptionsGroupView* themes_group_;
+ views::NativeButton* themes_reset_button_;
+
// Controls for the Popup Blocking group.
OptionsGroupView* popups_group_;
views::RadioButton* popups_show_minimized_radio_;
views::RadioButton* popups_block_all_radio_;
- // Controls for the Fonts and Languages group.
- OptionsGroupView* fonts_lang_group_;
- views::Label* fonts_and_languages_label_;
- views::NativeButton* change_content_fonts_button_;
-
StringPrefMember default_download_location_;
BooleanPrefMember ask_for_save_location_;
BooleanPrefMember ask_to_save_passwords_;
diff --git a/chrome/browser/views/options/cookies_view.cc b/chrome/browser/views/options/cookies_view.cc
index 593487b..cc8d65c 100644
--- a/chrome/browser/views/options/cookies_view.cc
+++ b/chrome/browser/views/options/cookies_view.cc
@@ -10,6 +10,7 @@
#include "app/gfx/color_utils.h"
#include "app/l10n_util.h"
#include "app/resource_bundle.h"
+#include "base/message_loop.h"
#include "base/string_util.h"
#include "base/time_format.h"
#include "chrome/browser/profile.h"
diff --git a/chrome/browser/views/tabs/tab_renderer.cc b/chrome/browser/views/tabs/tab_renderer.cc
index 87d0d01..e7b664b 100644
--- a/chrome/browser/views/tabs/tab_renderer.cc
+++ b/chrome/browser/views/tabs/tab_renderer.cc
@@ -12,6 +12,7 @@
#include "app/resource_bundle.h"
#include "app/win_util.h"
#include "chrome/browser/browser.h"
+#include "chrome/browser/browser_theme_provider.h"
#include "chrome/browser/profile.h"
#include "chrome/browser/tab_contents/tab_contents.h"
#include "chrome/browser/tabs/tab_strip_model.h"
@@ -43,7 +44,6 @@ static const int kPulseDurationMs = 200;
// How opaque to make the hover state (out of 1).
static const double kHoverOpacity = 0.33;
-static const double kHoverOpacityVista = 0.7;
// TODO(beng): (Cleanup) This stuff should move onto the class.
static ChromeFont* title_font = NULL;
@@ -53,6 +53,7 @@ static SkBitmap* close_button_h = NULL;
static SkBitmap* close_button_p = NULL;
static int close_button_height = 0;
static int close_button_width = 0;
+
static SkBitmap* waiting_animation_frames = NULL;
static SkBitmap* loading_animation_frames = NULL;
static SkBitmap* crashed_fav_icon = NULL;
@@ -63,16 +64,16 @@ static SkBitmap* download_icon = NULL;
static int download_icon_width = 0;
static int download_icon_height = 0;
+TabRenderer::TabImage TabRenderer::tab_alpha = {0};
TabRenderer::TabImage TabRenderer::tab_active = {0};
TabRenderer::TabImage TabRenderer::tab_inactive = {0};
-TabRenderer::TabImage TabRenderer::tab_inactive_otr = {0};
-TabRenderer::TabImage TabRenderer::tab_hover = {0};
namespace {
void InitResources() {
static bool initialized = false;
if (!initialized) {
+ // TODO(glen): Allow theming of these.
ResourceBundle& rb = ResourceBundle::GetSharedInstance();
title_font = new ChromeFont(rb.GetFont(ResourceBundle::BaseFont));
title_font_height = title_font->height();
@@ -83,7 +84,7 @@ void InitResources() {
close_button_width = close_button_n->width();
close_button_height = close_button_n->height();
- TabRenderer::LoadTabImages(win_util::ShouldUseVistaFrame());
+ TabRenderer::LoadTabImages();
// The loading animation image is a strip of states. Each state must be
// square, so the height must divide the width evenly.
@@ -215,7 +216,8 @@ TabRenderer::TabRenderer()
showing_close_button_(false),
crash_animation_(NULL),
fav_icon_hiding_offset_(0),
- should_display_crashed_favicon_(false) {
+ should_display_crashed_favicon_(false),
+ theme_provider_(NULL) {
InitResources();
// Add the Close Button.
@@ -236,6 +238,24 @@ TabRenderer::~TabRenderer() {
delete crash_animation_;
}
+void TabRenderer::ViewHierarchyChanged(bool is_add, View* parent, View* child) {
+ if (parent->GetThemeProvider())
+ SetThemeProvider(parent->GetThemeProvider());
+}
+
+ThemeProvider* TabRenderer::GetThemeProvider() {
+ ThemeProvider* tp = View::GetThemeProvider();
+ if (tp)
+ return tp;
+
+ if (theme_provider_)
+ return theme_provider_;
+
+ // return contents->profile()->GetThemeProvider();
+ NOTREACHED() << "Unable to find a theme provider";
+ return NULL;
+}
+
void TabRenderer::UpdateData(TabContents* contents, bool loading_only) {
DCHECK(contents);
if (!loading_only) {
@@ -246,6 +266,9 @@ void TabRenderer::UpdateData(TabContents* contents, bool loading_only) {
data_.favicon = contents->GetFavIcon();
}
+ // TODO(glen): Temporary hax.
+ theme_provider_ = contents->profile()->GetThemeProvider();
+
// Loading state also involves whether we show the favicon, since that's where
// we display the throbber.
data_.loading = contents->is_loading();
@@ -411,8 +434,11 @@ void TabRenderer::Paint(ChromeCanvas* canvas) {
Browser::FormatTitleForDisplay(&title);
}
- SkColor title_color = IsSelected() ? kSelectedTitleColor
- : kUnselectedTitleColor;
+ SkColor title_color = GetThemeProvider()->
+ GetColor(IsSelected() ?
+ BrowserThemeProvider::COLOR_TAB_TEXT :
+ BrowserThemeProvider::COLOR_BACKGROUND_TAB_TEXT);
+
canvas->DrawStringInt(title, *title_font, title_color, title_bounds_.x(),
title_bounds_.y(), title_bounds_.width(),
title_bounds_.height());
@@ -507,8 +533,7 @@ void TabRenderer::OnMouseExited(const views::MouseEvent& e) {
}
void TabRenderer::ThemeChanged() {
- if (GetWindow())
- LoadTabImages(GetWindow()->GetNonClientView()->UseNativeFrame());
+ LoadTabImages();
View::ThemeChanged();
}
@@ -541,33 +566,111 @@ void TabRenderer::PaintTabBackground(ChromeCanvas* canvas) {
Animation* animation = hover_animation_.get();
if (pulse_animation_->IsAnimating())
animation = pulse_animation_.get();
+
+ PaintInactiveTabBackground(canvas);
if (animation->GetCurrentValue() > 0) {
- PaintHoverTabBackground(canvas, animation->GetCurrentValue() *
- (GetWindow()->GetNonClientView()->UseNativeFrame() ?
- kHoverOpacityVista : kHoverOpacity));
- } else {
- PaintInactiveTabBackground(canvas);
+ SkRect bounds;
+ bounds.set(0, 0, SkIntToScalar(width()), SkIntToScalar(height()));
+ canvas->saveLayerAlpha(&bounds,
+ static_cast<int>(animation->GetCurrentValue() * kHoverOpacity * 0xff),
+ SkCanvas::kARGB_ClipLayer_SaveFlag);
+ canvas->drawARGB(0, 255, 255, 255, SkPorterDuff::kClear_Mode);
+ PaintActiveTabBackground(canvas);
+ canvas->restore();
}
}
}
void TabRenderer::PaintInactiveTabBackground(ChromeCanvas* canvas) {
bool is_otr = data_.off_the_record;
- canvas->DrawBitmapInt(is_otr ? *tab_inactive_otr.image_l :
- *tab_inactive.image_l, 0, 0);
- canvas->TileImageInt(is_otr ? *tab_inactive_otr.image_c :
- *tab_inactive.image_c, tab_inactive.l_width, 0,
- width() - tab_inactive.l_width -
- tab_inactive.r_width, height());
- canvas->DrawBitmapInt(is_otr ? *tab_inactive_otr.image_r :
- *tab_inactive.image_r,
+
+ // The tab image needs to be lined up with the background image
+ // so that it feels partially transparent.
+ int offset = GetX(views::View::APPLY_MIRRORING_TRANSFORMATION) + 1;
+ int offset_y = 20;
+
+ int tab_id;
+ if (GetWidget() &&
+ GetWidget()->GetWindow()->GetNonClientView()->UseNativeFrame()) {
+ tab_id = IDR_THEME_TAB_BACKGROUND_V;
+ } else {
+ tab_id = is_otr ? IDR_THEME_TAB_BACKGROUND_INCOGNITO :
+ IDR_THEME_TAB_BACKGROUND;
+ }
+
+ SkBitmap* tab_bg = GetThemeProvider()->GetBitmapNamed(tab_id);
+
+ // Draw left edge.
+ SkBitmap tab_l = skia::ImageOperations::CreateTiledBitmap(
+ *tab_bg, offset, offset_y,
+ tab_active.l_width, height());
+ SkBitmap theme_l = skia::ImageOperations::CreateMaskedBitmap(
+ tab_l, *tab_alpha.image_l);
+ canvas->DrawBitmapInt(theme_l,
+ 0, 0, theme_l.width(), theme_l.height() - 1,
+ 0, 0, theme_l.width(), theme_l.height() - 1,
+ false);
+
+ // Draw right edge.
+ SkBitmap tab_r = skia::ImageOperations::CreateTiledBitmap(
+ *tab_bg,
+ offset + width() - tab_active.r_width, offset_y,
+ tab_active.r_width, height());
+ SkBitmap theme_r = skia::ImageOperations::CreateMaskedBitmap(
+ tab_r, *tab_alpha.image_r);
+ canvas->DrawBitmapInt(theme_r,
+ 0, 0, theme_r.width(), theme_r.height() - 1,
+ width() - theme_r.width(), 0, theme_r.width(), theme_r.height() - 1,
+ false);
+
+ // Draw center.
+ canvas->TileImageInt(*tab_bg,
+ offset + tab_active.l_width, 2 + offset_y, // 2 is the drop shadow offset.
+ tab_active.l_width, 2,
+ width() - tab_active.l_width - tab_active.r_width, height() - 3);
+
+ canvas->DrawBitmapInt(*tab_inactive.image_l, 0, 0);
+ canvas->TileImageInt(*tab_inactive.image_c,
+ tab_inactive.l_width, 0,
+ width() - tab_inactive.l_width - tab_inactive.r_width,
+ height());
+ canvas->DrawBitmapInt(*tab_inactive.image_r,
width() - tab_inactive.r_width, 0);
}
void TabRenderer::PaintActiveTabBackground(ChromeCanvas* canvas) {
+ int offset = GetX(views::View::APPLY_MIRRORING_TRANSFORMATION) + 1;
+ ThemeProvider* tp = GetThemeProvider();
+ if (!tp)
+ NOTREACHED() << "Unable to get theme provider";
+
+ SkBitmap* tab_bg = GetThemeProvider()->GetBitmapNamed(IDR_THEME_TOOLBAR);
+
+ // Draw left edge.
+ SkBitmap tab_l = skia::ImageOperations::CreateTiledBitmap(
+ *tab_bg, offset, 0, tab_active.l_width, height());
+ SkBitmap theme_l = skia::ImageOperations::CreateMaskedBitmap(
+ tab_l, *tab_alpha.image_l);
+ canvas->DrawBitmapInt(theme_l, 0, 0);
+
+ // Draw right edge.
+ SkBitmap tab_r = skia::ImageOperations::CreateTiledBitmap(
+ *tab_bg,
+ offset + width() - tab_active.r_width, 0,
+ tab_active.r_width, height());
+ SkBitmap theme_r = skia::ImageOperations::CreateMaskedBitmap(
+ tab_r, *tab_alpha.image_r);
+ canvas->DrawBitmapInt(theme_r, width() - tab_active.r_width, 0);
+
+ // Draw center.
+ canvas->TileImageInt(*tab_bg,
+ offset + tab_active.l_width, 2,
+ tab_active.l_width, 2,
+ width() - tab_active.l_width - tab_active.r_width, height() - 2);
+
canvas->DrawBitmapInt(*tab_active.image_l, 0, 0);
canvas->TileImageInt(*tab_active.image_c, tab_active.l_width, 0,
- width() - tab_active.l_width - tab_active.r_width, height());
+ width() - tab_active.l_width - tab_active.r_width, height());
canvas->DrawBitmapInt(*tab_active.image_r, width() - tab_active.r_width, 0);
}
@@ -575,15 +678,11 @@ void TabRenderer::PaintHoverTabBackground(ChromeCanvas* canvas,
double opacity) {
bool is_otr = data_.off_the_record;
SkBitmap left = skia::ImageOperations::CreateBlendedBitmap(
- (is_otr ? *tab_inactive_otr.image_l :
- *tab_inactive.image_l), *tab_hover.image_l, opacity);
+ *tab_inactive.image_l, *tab_active.image_l, opacity);
SkBitmap center = skia::ImageOperations::CreateBlendedBitmap(
- (is_otr ? *tab_inactive_otr.image_c :
- *tab_inactive.image_c), *tab_hover.image_c, opacity);
+ *tab_inactive.image_c, *tab_active.image_c, opacity);
SkBitmap right = skia::ImageOperations::CreateBlendedBitmap(
- (is_otr ? *tab_inactive_otr.image_r :
- *tab_inactive.image_r),
- *tab_hover.image_r, opacity);
+ *tab_inactive.image_r, *tab_active.image_r, opacity);
canvas->DrawBitmapInt(left, 0, 0);
canvas->TileImageInt(center, tab_active.l_width, 0,
@@ -668,41 +767,26 @@ void TabRenderer::ResetCrashedFavIcon() {
}
// static
-void TabRenderer::LoadTabImages(bool use_vista_images) {
+void TabRenderer::LoadTabImages() {
+ // We're not letting people override tab images just yet.
ResourceBundle& rb = ResourceBundle::GetSharedInstance();
+ tab_alpha.image_l = rb.GetBitmapNamed(IDR_TAB_ALPHA_LEFT);
+ tab_alpha.image_r = rb.GetBitmapNamed(IDR_TAB_ALPHA_RIGHT);
+
tab_active.image_l = rb.GetBitmapNamed(IDR_TAB_ACTIVE_LEFT);
tab_active.image_c = rb.GetBitmapNamed(IDR_TAB_ACTIVE_CENTER);
tab_active.image_r = rb.GetBitmapNamed(IDR_TAB_ACTIVE_RIGHT);
tab_active.l_width = tab_active.image_l->width();
tab_active.r_width = tab_active.image_r->width();
- tab_hover.image_l = rb.GetBitmapNamed(IDR_TAB_HOVER_LEFT);
- tab_hover.image_c = rb.GetBitmapNamed(IDR_TAB_HOVER_CENTER);
- tab_hover.image_r = rb.GetBitmapNamed(IDR_TAB_HOVER_RIGHT);
-
- if (use_vista_images) {
- tab_inactive.image_l = rb.GetBitmapNamed(IDR_TAB_INACTIVE_LEFT_V);
- tab_inactive.image_c = rb.GetBitmapNamed(IDR_TAB_INACTIVE_CENTER_V);
- tab_inactive.image_r = rb.GetBitmapNamed(IDR_TAB_INACTIVE_RIGHT_V);
-
- // Our Vista frame doesn't change background color to show OTR,
- // so we continue to use the existing background tabs.
- tab_inactive_otr.image_l = rb.GetBitmapNamed(IDR_TAB_INACTIVE_LEFT_V);
- tab_inactive_otr.image_c = rb.GetBitmapNamed(IDR_TAB_INACTIVE_CENTER_V);
- tab_inactive_otr.image_r = rb.GetBitmapNamed(IDR_TAB_INACTIVE_RIGHT_V);
- } else {
- tab_inactive.image_l = rb.GetBitmapNamed(IDR_TAB_INACTIVE_LEFT);
- tab_inactive.image_c = rb.GetBitmapNamed(IDR_TAB_INACTIVE_CENTER);
- tab_inactive.image_r = rb.GetBitmapNamed(IDR_TAB_INACTIVE_RIGHT);
-
- tab_inactive_otr.image_l = rb.GetBitmapNamed(IDR_TAB_INACTIVE_LEFT_OTR);
- tab_inactive_otr.image_c = rb.GetBitmapNamed(IDR_TAB_INACTIVE_CENTER_OTR);
- tab_inactive_otr.image_r = rb.GetBitmapNamed(IDR_TAB_INACTIVE_RIGHT_OTR);
- }
+ tab_inactive.image_l = rb.GetBitmapNamed(IDR_TAB_INACTIVE_LEFT);
+ tab_inactive.image_c = rb.GetBitmapNamed(IDR_TAB_INACTIVE_CENTER);
+ tab_inactive.image_r = rb.GetBitmapNamed(IDR_TAB_INACTIVE_RIGHT);
tab_inactive.l_width = tab_inactive.image_l->width();
tab_inactive.r_width = tab_inactive.image_r->width();
- // tab_[hover,inactive_otr] width are not used and are initialized to 0
- // during static initialization.
+
+ loading_animation_frames = rb.GetBitmapNamed(IDR_THROBBER);
+ waiting_animation_frames = rb.GetBitmapNamed(IDR_THROBBER_WAITING);
}
diff --git a/chrome/browser/views/tabs/tab_renderer.h b/chrome/browser/views/tabs/tab_renderer.h
index f498c2a..d278fb5 100644
--- a/chrome/browser/views/tabs/tab_renderer.h
+++ b/chrome/browser/views/tabs/tab_renderer.h
@@ -36,6 +36,10 @@ class TabRenderer : public views::View,
TabRenderer();
virtual ~TabRenderer();
+ // Overridden from views:
+ void ViewHierarchyChanged(bool is_add, View* parent, View* child);
+ ThemeProvider* GetThemeProvider();
+
// Updates the data the Tab uses to render itself from the specified
// TabContents.
//
@@ -56,6 +60,13 @@ class TabRenderer : public views::View,
void StartPulse();
void StopPulse();
+ // Set the theme provider - because we get detached, we are frequently
+ // outside of a hierarchy with a theme provider at the top. This should be
+ // called whenever we're detached or attached to a hierarchy.
+ void SetThemeProvider(ThemeProvider* provider) {
+ theme_provider_ = provider;
+ }
+
// Returns the minimum possible size of a single unselected Tab.
static gfx::Size GetMinimumUnselectedSize();
// Returns the minimum possible size of a selected Tab. Selected tabs must
@@ -66,9 +77,8 @@ class TabRenderer : public views::View,
// available.
static gfx::Size GetStandardSize();
- // Loads the images to be used for the tab background. Uses the images for
- // Vista if |use_vista_images| is true.
- static void LoadTabImages(bool use_vista_images);
+ // Loads the images to be used for the tab background.
+ static void LoadTabImages();
protected:
views::ImageButton* close_button() const { return close_button_; }
@@ -166,7 +176,7 @@ class TabRenderer : public views::View,
};
static TabImage tab_active;
static TabImage tab_inactive;
- static TabImage tab_inactive_otr;
+ static TabImage tab_alpha;
static TabImage tab_hover;
// Whether we're showing the icon. It is cached so that we can detect when it
@@ -189,6 +199,8 @@ class TabRenderer : public views::View,
bool should_display_crashed_favicon_;
+ ThemeProvider* theme_provider_;
+
static void InitClass();
static bool initialized_;
diff --git a/chrome/browser/views/toolbar_view.cc b/chrome/browser/views/toolbar_view.cc
index c3c98d2..333dcf4 100644
--- a/chrome/browser/views/toolbar_view.cc
+++ b/chrome/browser/views/toolbar_view.cc
@@ -23,6 +23,7 @@
#include "chrome/browser/profile.h"
#include "chrome/browser/tab_contents/navigation_controller.h"
#include "chrome/browser/tab_contents/navigation_entry.h"
+#include "chrome/browser/browser_theme_provider.h"
#include "chrome/browser/user_data_manager.h"
#include "chrome/browser/views/bookmark_menu_button.h"
#include "chrome/browser/views/dom_view.h"
@@ -138,86 +139,49 @@ void BrowserToolbarView::SetProfile(Profile* profile) {
}
void BrowserToolbarView::CreateLeftSideControls() {
- ResourceBundle &rb = ResourceBundle::GetSharedInstance();
-
back_ = new views::ButtonDropDown(this, back_menu_model_.get());
back_->set_triggerable_event_flags(views::Event::EF_LEFT_BUTTON_DOWN |
views::Event::EF_MIDDLE_BUTTON_DOWN);
back_->set_tag(IDC_BACK);
back_->SetImageAlignment(views::ImageButton::ALIGN_RIGHT,
views::ImageButton::ALIGN_TOP);
- back_->SetImage(views::CustomButton::BS_NORMAL, rb.GetBitmapNamed(IDR_BACK));
- back_->SetImage(views::CustomButton::BS_HOT, rb.GetBitmapNamed(IDR_BACK_H));
- back_->SetImage(views::CustomButton::BS_PUSHED,
- rb.GetBitmapNamed(IDR_BACK_P));
- back_->SetImage(views::CustomButton::BS_DISABLED,
- rb.GetBitmapNamed(IDR_BACK_D));
back_->SetTooltipText(l10n_util::GetString(IDS_TOOLTIP_BACK));
back_->SetAccessibleName(l10n_util::GetString(IDS_ACCNAME_BACK));
back_->SetID(VIEW_ID_BACK_BUTTON);
- AddChildView(back_);
forward_ = new views::ButtonDropDown(this, forward_menu_model_.get());
forward_->set_triggerable_event_flags(views::Event::EF_LEFT_BUTTON_DOWN |
- views::Event::EF_MIDDLE_BUTTON_DOWN);
+ views::Event::EF_MIDDLE_BUTTON_DOWN);
forward_->set_tag(IDC_FORWARD);
- forward_->SetImage(views::CustomButton::BS_NORMAL,
- rb.GetBitmapNamed(IDR_FORWARD));
- forward_->SetImage(views::CustomButton::BS_HOT,
- rb.GetBitmapNamed(IDR_FORWARD_H));
- forward_->SetImage(views::CustomButton::BS_PUSHED,
- rb.GetBitmapNamed(IDR_FORWARD_P));
- forward_->SetImage(views::CustomButton::BS_DISABLED,
- rb.GetBitmapNamed(IDR_FORWARD_D));
forward_->SetTooltipText(l10n_util::GetString(IDS_TOOLTIP_FORWARD));
forward_->SetAccessibleName(l10n_util::GetString(IDS_ACCNAME_FORWARD));
forward_->SetID(VIEW_ID_FORWARD_BUTTON);
- AddChildView(forward_);
reload_ = new views::ImageButton(this);
reload_->set_tag(IDC_RELOAD);
- reload_->SetImage(views::CustomButton::BS_NORMAL,
- rb.GetBitmapNamed(IDR_RELOAD));
- reload_->SetImage(views::CustomButton::BS_HOT,
- rb.GetBitmapNamed(IDR_RELOAD_H));
- reload_->SetImage(views::CustomButton::BS_PUSHED,
- rb.GetBitmapNamed(IDR_RELOAD_P));
reload_->SetTooltipText(l10n_util::GetString(IDS_TOOLTIP_RELOAD));
reload_->SetAccessibleName(l10n_util::GetString(IDS_ACCNAME_RELOAD));
reload_->SetID(VIEW_ID_RELOAD_BUTTON);
- AddChildView(reload_);
home_ = new views::ImageButton(this);
home_->set_triggerable_event_flags(views::Event::EF_LEFT_BUTTON_DOWN |
views::Event::EF_MIDDLE_BUTTON_DOWN);
home_->set_tag(IDC_HOME);
- home_->SetImage(views::CustomButton::BS_NORMAL, rb.GetBitmapNamed(IDR_HOME));
- home_->SetImage(views::CustomButton::BS_HOT, rb.GetBitmapNamed(IDR_HOME_H));
- home_->SetImage(views::CustomButton::BS_PUSHED,
- rb.GetBitmapNamed(IDR_HOME_P));
home_->SetTooltipText(l10n_util::GetString(IDS_TOOLTIP_HOME));
home_->SetAccessibleName(l10n_util::GetString(IDS_ACCNAME_HOME));
home_->SetID(VIEW_ID_HOME_BUTTON);
+
+ LoadLeftSideControlsImages();
+
+ AddChildView(back_);
+ AddChildView(forward_);
+ AddChildView(reload_);
AddChildView(home_);
}
void BrowserToolbarView::CreateCenterStack(Profile *profile) {
- ResourceBundle &rb = ResourceBundle::GetSharedInstance();
-
star_ = new ToolbarStarToggle(this, this);
star_->set_tag(IDC_STAR);
- star_->SetImage(views::CustomButton::BS_NORMAL, rb.GetBitmapNamed(IDR_STAR));
- star_->SetImage(views::CustomButton::BS_HOT, rb.GetBitmapNamed(IDR_STAR_H));
- star_->SetImage(views::CustomButton::BS_PUSHED,
- rb.GetBitmapNamed(IDR_STAR_P));
- star_->SetImage(views::CustomButton::BS_DISABLED,
- rb.GetBitmapNamed(IDR_STAR_D));
- star_->SetToggledImage(views::CustomButton::BS_NORMAL,
- rb.GetBitmapNamed(IDR_STARRED));
- star_->SetToggledImage(views::CustomButton::BS_HOT,
- rb.GetBitmapNamed(IDR_STARRED_H));
- star_->SetToggledImage(views::CustomButton::BS_PUSHED,
- rb.GetBitmapNamed(IDR_STARRED_P));
star_->SetDragController(this);
star_->SetTooltipText(l10n_util::GetString(IDS_TOOLTIP_STAR));
star_->SetToggledTooltipText(l10n_util::GetString(IDS_TOOLTIP_STARRED));
@@ -229,68 +193,37 @@ void BrowserToolbarView::CreateCenterStack(Profile *profile) {
model_, this,
display_mode_ == DISPLAYMODE_LOCATION,
this);
- AddChildView(location_bar_);
- location_bar_->Init();
// The Go button.
go_ = new GoButton(location_bar_, browser_);
- go_->SetImage(views::CustomButton::BS_NORMAL, rb.GetBitmapNamed(IDR_GO));
- go_->SetImage(views::CustomButton::BS_HOT, rb.GetBitmapNamed(IDR_GO_H));
- go_->SetImage(views::CustomButton::BS_PUSHED, rb.GetBitmapNamed(IDR_GO_P));
- go_->SetToggledImage(views::CustomButton::BS_NORMAL,
- rb.GetBitmapNamed(IDR_STOP));
- go_->SetToggledImage(views::CustomButton::BS_HOT,
- rb.GetBitmapNamed(IDR_STOP_H));
- go_->SetToggledImage(views::CustomButton::BS_PUSHED,
- rb.GetBitmapNamed(IDR_STOP_P));
go_->SetAccessibleName(l10n_util::GetString(IDS_ACCNAME_GO));
go_->SetID(VIEW_ID_GO_BUTTON);
- AddChildView(go_);
-}
-
-void BrowserToolbarView::Update(TabContents* tab, bool should_restore_state) {
- tab_ = tab;
- if (!location_bar_)
- return;
+ LoadCenterStackImages();
- location_bar_->Update(should_restore_state ? tab : NULL);
-}
-
-void BrowserToolbarView::OnInputInProgress(bool in_progress) {
- // The edit should make sure we're only notified when something changes.
- DCHECK(model_->input_in_progress() != in_progress);
-
- model_->set_input_in_progress(in_progress);
- location_bar_->Update(NULL);
+ AddChildView(location_bar_);
+ location_bar_->Init();
+ AddChildView(go_);
}
void BrowserToolbarView::CreateRightSideControls(Profile* profile) {
- ResourceBundle &rb = ResourceBundle::GetSharedInstance();
+ ThemeProvider* tp = GetThemeProvider();
page_menu_ = new views::MenuButton(NULL, std::wstring(), this, false);
-
- // We use different menu button images if the locale is right-to-left.
- if (UILayoutIsRightToLeft())
- page_menu_->SetIcon(*rb.GetBitmapNamed(IDR_MENU_PAGE_RTL));
- else
- page_menu_->SetIcon(*rb.GetBitmapNamed(IDR_MENU_PAGE));
-
page_menu_->SetAccessibleName(l10n_util::GetString(IDS_ACCNAME_PAGE));
page_menu_->SetTooltipText(l10n_util::GetString(IDS_PAGEMENU_TOOLTIP));
page_menu_->SetID(VIEW_ID_PAGE_MENU);
- AddChildView(page_menu_);
- app_menu_ = new views::MenuButton(NULL, std::wstring(), this, false);
- if (UILayoutIsRightToLeft())
- app_menu_->SetIcon(*rb.GetBitmapNamed(IDR_MENU_CHROME_RTL));
- else
- app_menu_->SetIcon(*rb.GetBitmapNamed(IDR_MENU_CHROME));
+ app_menu_ = new views::MenuButton(NULL, std::wstring(), this, false);
app_menu_->SetAccessibleName(l10n_util::GetString(IDS_ACCNAME_APP));
app_menu_->SetTooltipText(l10n_util::GetStringF(IDS_APPMENU_TOOLTIP,
l10n_util::GetString(IDS_PRODUCT_NAME)));
app_menu_->SetID(VIEW_ID_APP_MENU);
+
+ LoadRightSideControlsImages();
+
+ AddChildView(page_menu_);
AddChildView(app_menu_);
if (CommandLine::ForCurrentProcess()->HasSwitch(switches::kBookmarkMenu)) {
@@ -301,6 +234,96 @@ void BrowserToolbarView::CreateRightSideControls(Profile* profile) {
}
}
+void BrowserToolbarView::LoadLeftSideControlsImages() {
+ ThemeProvider* tp = GetThemeProvider();
+
+ back_->SetImage(views::CustomButton::BS_NORMAL, tp->GetBitmapNamed(IDR_BACK));
+ back_->SetImage(views::CustomButton::BS_HOT, tp->GetBitmapNamed(IDR_BACK_H));
+ back_->SetImage(views::CustomButton::BS_PUSHED,
+ tp->GetBitmapNamed(IDR_BACK_P));
+ back_->SetImage(views::CustomButton::BS_DISABLED,
+ tp->GetBitmapNamed(IDR_BACK_D));
+
+ forward_->SetImage(views::CustomButton::BS_NORMAL,
+ tp->GetBitmapNamed(IDR_FORWARD));
+ forward_->SetImage(views::CustomButton::BS_HOT,
+ tp->GetBitmapNamed(IDR_FORWARD_H));
+ forward_->SetImage(views::CustomButton::BS_PUSHED,
+ tp->GetBitmapNamed(IDR_FORWARD_P));
+ forward_->SetImage(views::CustomButton::BS_DISABLED,
+ tp->GetBitmapNamed(IDR_FORWARD_D));
+
+ reload_->SetImage(views::CustomButton::BS_NORMAL,
+ tp->GetBitmapNamed(IDR_RELOAD));
+ reload_->SetImage(views::CustomButton::BS_HOT,
+ tp->GetBitmapNamed(IDR_RELOAD_H));
+ reload_->SetImage(views::CustomButton::BS_PUSHED,
+ tp->GetBitmapNamed(IDR_RELOAD_P));
+
+ home_->SetImage(views::CustomButton::BS_NORMAL, tp->GetBitmapNamed(IDR_HOME));
+ home_->SetImage(views::CustomButton::BS_HOT, tp->GetBitmapNamed(IDR_HOME_H));
+ home_->SetImage(views::CustomButton::BS_PUSHED,
+ tp->GetBitmapNamed(IDR_HOME_P));
+}
+
+void BrowserToolbarView::LoadCenterStackImages() {
+ ThemeProvider* tp = GetThemeProvider();
+
+ star_->SetImage(views::CustomButton::BS_NORMAL, tp->GetBitmapNamed(IDR_STAR));
+ star_->SetImage(views::CustomButton::BS_HOT, tp->GetBitmapNamed(IDR_STAR_H));
+ star_->SetImage(views::CustomButton::BS_PUSHED,
+ tp->GetBitmapNamed(IDR_STAR_P));
+ star_->SetImage(views::CustomButton::BS_DISABLED,
+ tp->GetBitmapNamed(IDR_STAR_D));
+ star_->SetToggledImage(views::CustomButton::BS_NORMAL,
+ tp->GetBitmapNamed(IDR_STARRED));
+ star_->SetToggledImage(views::CustomButton::BS_HOT,
+ tp->GetBitmapNamed(IDR_STARRED_H));
+ star_->SetToggledImage(views::CustomButton::BS_PUSHED,
+ tp->GetBitmapNamed(IDR_STARRED_P));
+
+ go_->SetImage(views::CustomButton::BS_NORMAL, tp->GetBitmapNamed(IDR_GO));
+ go_->SetImage(views::CustomButton::BS_HOT, tp->GetBitmapNamed(IDR_GO_H));
+ go_->SetImage(views::CustomButton::BS_PUSHED, tp->GetBitmapNamed(IDR_GO_P));
+ go_->SetToggledImage(views::CustomButton::BS_NORMAL,
+ tp->GetBitmapNamed(IDR_STOP));
+ go_->SetToggledImage(views::CustomButton::BS_HOT,
+ tp->GetBitmapNamed(IDR_STOP_H));
+ go_->SetToggledImage(views::CustomButton::BS_PUSHED,
+ tp->GetBitmapNamed(IDR_STOP_P));
+}
+
+void BrowserToolbarView::LoadRightSideControlsImages() {
+ ThemeProvider* tp = GetThemeProvider();
+
+ // We use different menu button images if the locale is right-to-left.
+ if (UILayoutIsRightToLeft())
+ page_menu_->SetIcon(*tp->GetBitmapNamed(IDR_MENU_PAGE_RTL));
+ else
+ page_menu_->SetIcon(*tp->GetBitmapNamed(IDR_MENU_PAGE));
+ if (UILayoutIsRightToLeft())
+ app_menu_->SetIcon(*tp->GetBitmapNamed(IDR_MENU_CHROME_RTL));
+ else
+ app_menu_->SetIcon(*tp->GetBitmapNamed(IDR_MENU_CHROME));
+}
+
+void BrowserToolbarView::Update(TabContents* tab, bool should_restore_state) {
+ tab_ = tab;
+
+ if (!location_bar_)
+ return;
+
+ location_bar_->Update(should_restore_state ? tab : NULL);
+}
+
+void BrowserToolbarView::OnInputInProgress(bool in_progress) {
+ // The edit should make sure we're only notified when something changes.
+ DCHECK(model_->input_in_progress() != in_progress);
+
+ model_->set_input_in_progress(in_progress);
+ location_bar_->Update(NULL);
+}
+
void BrowserToolbarView::Layout() {
// If we have not been initialized yet just do nothing.
if (back_ == NULL)
@@ -758,6 +781,12 @@ void BrowserToolbarView::SetAccessibleName(const std::wstring& name) {
accessible_name_.assign(name);
}
+void BrowserToolbarView::ThemeChanged() {
+ LoadLeftSideControlsImages();
+ LoadCenterStackImages();
+ LoadRightSideControlsImages();
+}
+
int BrowserToolbarView::GetNextAccessibleViewIndex(int view_index,
bool nav_left) {
int modifier = 1;
diff --git a/chrome/browser/views/toolbar_view.h b/chrome/browser/views/toolbar_view.h
index c5a9f83..b5f3861 100644
--- a/chrome/browser/views/toolbar_view.h
+++ b/chrome/browser/views/toolbar_view.h
@@ -63,6 +63,7 @@ class BrowserToolbarView : public views::View,
virtual bool GetAccessibleName(std::wstring* name);
virtual bool GetAccessibleRole(AccessibilityTypes::Role* role);
virtual void SetAccessibleName(const std::wstring& name);
+ virtual void ThemeChanged();
// Overridden from EncodingMenuControllerDelegate:
virtual bool IsItemChecked(int id) const;
@@ -147,6 +148,9 @@ class BrowserToolbarView : public views::View,
void CreateLeftSideControls();
void CreateCenterStack(Profile* profile);
void CreateRightSideControls(Profile* profile);
+ void LoadLeftSideControlsImages();
+ void LoadCenterStackImages();
+ void LoadRightSideControlsImages();
// Updates the controls to display the security appropriately (highlighted if
// secure).