blob: cf3fe5251f7e014cd67c944642bb9255f693c246 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ui/webui/theme_source.h"
#include <string_view>
#include "base/functional/bind.h"
#include "base/memory/ref_counted_memory.h"
#include "base/metrics/histogram_functions.h"
#include "base/ranges/algorithm.h"
#include "base/strings/strcat.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_split.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/timer/elapsed_timer.h"
#include "build/branding_buildflags.h"
#include "build/chromeos_buildflags.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/resources_util.h"
#include "chrome/browser/search/instant_service.h"
#include "chrome/browser/themes/browser_theme_pack.h"
#include "chrome/browser/themes/theme_properties.h"
#include "chrome/browser/themes/theme_service.h"
#include "chrome/browser/themes/theme_service_factory.h"
#include "chrome/browser/ui/color/chrome_color_id.h"
#include "chrome/browser/ui/color/chrome_color_provider_utils.h"
#include "chrome/browser/ui/webui/ntp/ntp_resource_cache.h"
#include "chrome/browser/ui/webui/ntp/ntp_resource_cache_factory.h"
#include "chrome/common/channel_info.h"
#include "chrome/common/url_constants.h"
#include "chrome/grit/theme_resources.h"
#include "components/version_info/version_info.h"
#include "content/public/browser/browser_task_traits.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/url_data_source.h"
#include "content/public/browser/web_contents.h"
#include "content/public/common/url_constants.h"
#include "net/base/url_util.h"
#include "services/network/public/mojom/content_security_policy.mojom.h"
#include "ui/base/resource/resource_bundle.h"
#include "ui/base/resource/resource_scale_factor.h"
#include "ui/base/webui/web_ui_util.h"
#include "ui/color/color_provider.h"
#include "ui/color/color_provider_utils.h"
#include "ui/gfx/codec/png_codec.h"
#include "ui/gfx/image/image_skia.h"
#include "ui/gfx/image/image_skia_rep.h"
#include "url/gurl.h"
#if BUILDFLAG(IS_CHROMEOS_ASH)
#include "chrome/grit/cros_styles_resources.h" // nogncheck crbug.com/1113869
#include "chromeos/constants/chromeos_features.h"
#include "ui/chromeos/styles/cros_tokens_color_mappings.h"
#endif // BUILDFLAG(IS_CHROMEOS_ASH)
namespace {
GURL GetThemeUrl(const std::string& path) {
return GURL(std::string(content::kChromeUIScheme) + "://" +
std::string(chrome::kChromeUIThemeHost) + "/" + path);
}
bool IsNewTabCssPath(const std::string& path) {
static const char kNewTabThemeCssPath[] = "css/new_tab_theme.css";
static const char kIncognitoTabThemeCssPath[] = "css/incognito_tab_theme.css";
return path == kNewTabThemeCssPath || path == kIncognitoTabThemeCssPath;
}
void ProcessImageOnUiThread(const gfx::ImageSkia& image,
float scale,
scoped_refptr<base::RefCountedBytes> data) {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
const gfx::ImageSkiaRep& rep = image.GetRepresentation(scale);
gfx::PNGCodec::EncodeBGRASkBitmap(
rep.GetBitmap(), false /* discard transparency */, &data->as_vector());
}
} // namespace
////////////////////////////////////////////////////////////////////////////////
// ThemeSource, public:
ThemeSource::ThemeSource(Profile* profile)
: profile_(profile), serve_untrusted_(false) {}
ThemeSource::ThemeSource(Profile* profile, bool serve_untrusted)
: profile_(profile), serve_untrusted_(serve_untrusted) {}
ThemeSource::~ThemeSource() = default;
std::string ThemeSource::GetSource() {
return serve_untrusted_ ? chrome::kChromeUIUntrustedThemeURL
: chrome::kChromeUIThemeHost;
}
void ThemeSource::StartDataRequest(
const GURL& url,
const content::WebContents::Getter& wc_getter,
content::URLDataSource::GotDataCallback callback) {
// TODO(crbug.com/40050262): Simplify usages of |path| since |url| is
// available.
const std::string path = content::URLDataSource::URLToRequestPath(url);
// Default scale factor if not specified.
float scale = 1.0f;
// All frames by default if not specified.
int frame = -1;
std::string parsed_path;
webui::ParsePathAndImageSpec(GetThemeUrl(path), &parsed_path, &scale, &frame);
if (IsNewTabCssPath(parsed_path)) {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
NTPResourceCache::WindowType type =
NTPResourceCache::GetWindowType(profile_);
NTPResourceCache* cache = NTPResourceCacheFactory::GetForProfile(profile_);
std::move(callback).Run(cache->GetNewTabCSS(type, wc_getter));
return;
}
// kColorsCssPath should stay consistent with COLORS_CSS_SELECTOR in
// colors_css_updater.js.
constexpr char kColorsCssPath[] = "colors.css";
if (parsed_path == kColorsCssPath) {
SendColorsCss(url, wc_getter, std::move(callback));
return;
}
#if BUILDFLAG(IS_CHROMEOS_ASH)
constexpr char kTypographyCssPath[] = "typography.css";
if (parsed_path == kTypographyCssPath) {
SendTypographyCss(std::move(callback));
return;
}
#endif
int resource_id = -1;
if (parsed_path == "current-channel-logo") {
switch (chrome::GetChannel()) {
#if BUILDFLAG(GOOGLE_CHROME_BRANDING)
case version_info::Channel::CANARY:
resource_id = IDR_PRODUCT_LOGO_32_CANARY;
break;
case version_info::Channel::DEV:
resource_id = IDR_PRODUCT_LOGO_32_DEV;
break;
case version_info::Channel::BETA:
resource_id = IDR_PRODUCT_LOGO_32_BETA;
break;
case version_info::Channel::STABLE:
resource_id = IDR_PRODUCT_LOGO_32;
break;
#else
case version_info::Channel::CANARY:
case version_info::Channel::DEV:
case version_info::Channel::BETA:
case version_info::Channel::STABLE:
NOTREACHED();
[[fallthrough]];
#endif
case version_info::Channel::UNKNOWN:
resource_id = IDR_PRODUCT_LOGO_32;
break;
}
} else {
resource_id = ResourcesUtil::GetThemeResourceId(parsed_path);
}
// Limit the maximum scale we'll respond to. Very large scale factors can
// take significant time to serve or, at worst, crash the browser due to OOM.
// We don't want to clamp to the max scale factor, though, for devices that
// use 2x scale without 2x data packs, as well as omnibox requests for larger
// (but still reasonable) scales (see below).
const float max_scale = ui::GetScaleForResourceScaleFactor(
ui::ResourceBundle::GetSharedInstance().GetMaxResourceScaleFactor());
const float unreasonable_scale = max_scale * 32;
// TODO(reveman): Add support frames beyond 0 (crbug.com/750064).
if ((resource_id == -1) || (scale >= unreasonable_scale) || (frame > 0)) {
// Either we have no data to send back, or the requested scale is
// unreasonably large. This shouldn't happen normally, as chrome://theme/
// URLs are only used by WebUI pages and component extensions. However, the
// user can also enter these into the omnibox, so we need to fail
// gracefully.
std::move(callback).Run(nullptr);
} else if ((GetMimeType(url) == "image/png") &&
((scale > max_scale) || (frame != -1))) {
// This will extract and scale frame 0 of animated images.
// TODO(reveman): Support scaling of animated images and avoid scaling and
// re-encode when specific frame is specified (crbug.com/750064).
DCHECK_LE(frame, 0);
SendThemeImage(std::move(callback), resource_id, scale);
} else {
SendThemeBitmap(std::move(callback), resource_id, scale);
}
}
std::string ThemeSource::GetMimeType(const GURL& url) {
const std::string_view file_path = url.path_piece();
if (base::EndsWith(file_path, ".css", base::CompareCase::INSENSITIVE_ASCII)) {
return "text/css";
}
return "image/png";
}
bool ThemeSource::AllowCaching() {
return false;
}
bool ThemeSource::ShouldServiceRequest(const GURL& url,
content::BrowserContext* browser_context,
int render_process_id) {
return url.SchemeIs(chrome::kChromeSearchScheme)
? InstantService::ShouldServiceRequest(url, browser_context,
render_process_id)
: URLDataSource::ShouldServiceRequest(url, browser_context,
render_process_id);
}
////////////////////////////////////////////////////////////////////////////////
// ThemeSource, private:
void ThemeSource::SendThemeBitmap(
content::URLDataSource::GotDataCallback callback,
int resource_id,
float scale) {
ui::ResourceScaleFactor scale_factor =
ui::GetSupportedResourceScaleFactor(scale);
if (BrowserThemePack::IsPersistentImageID(resource_id)) {
scoped_refptr<base::RefCountedMemory> image_data(
ThemeService::GetThemeProviderForProfile(profile_->GetOriginalProfile())
.GetRawData(resource_id, scale_factor));
std::move(callback).Run(image_data.get());
} else {
const ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
std::move(callback).Run(
rb.LoadDataResourceBytesForScale(resource_id, scale_factor));
}
}
void ThemeSource::SendThemeImage(
content::URLDataSource::GotDataCallback callback,
int resource_id,
float scale) {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
scoped_refptr<base::RefCountedBytes> data(new base::RefCountedBytes());
if (BrowserThemePack::IsPersistentImageID(resource_id)) {
const ui::ThemeProvider& tp = ThemeService::GetThemeProviderForProfile(
profile_->GetOriginalProfile());
ProcessImageOnUiThread(*tp.GetImageSkiaNamed(resource_id), scale, data);
} else {
ProcessImageOnUiThread(
*ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed(resource_id),
scale, data);
}
std::move(callback).Run(data.get());
}
void ThemeSource::SendColorsCss(
const GURL& url,
const content::WebContents::Getter& wc_getter,
content::URLDataSource::GotDataCallback callback) {
base::ElapsedTimer timer;
const ui::ColorProvider& color_provider = wc_getter.Run()->GetColorProvider();
std::string sets_param;
std::vector<std::string_view> color_id_sets;
bool generate_rgb_vars = false;
std::string generate_rgb_vars_query_value;
if (net::GetValueForKeyInQuery(url, "generate_rgb_vars",
&generate_rgb_vars_query_value)) {
generate_rgb_vars =
base::ToLowerASCII(generate_rgb_vars_query_value) == "true";
}
bool shadow_host = false;
std::string shadow_host_query_value;
if (net::GetValueForKeyInQuery(url, "shadow_host",
&shadow_host_query_value)) {
shadow_host = base::ToLowerASCII(shadow_host_query_value) == "true";
}
if (!net::GetValueForKeyInQuery(url, "sets", &sets_param)) {
LOG(ERROR)
<< "colors.css requires a 'sets' query parameter to specify the color "
"id sets returned e.g chrome://theme/colors.css?sets=ui,chrome";
std::move(callback).Run(nullptr);
return;
}
color_id_sets = base::SplitStringPiece(sets_param, ",", base::TRIM_WHITESPACE,
base::SPLIT_WANT_ALL);
using ColorIdCSSCallback = base::RepeatingCallback<std::string(ui::ColorId)>;
auto generate_color_mapping = [&color_id_sets, &color_provider,
&generate_rgb_vars](
std::string set_name, ui::ColorId start,
ui::ColorId end,
ColorIdCSSCallback color_css_name) {
// Only return these mappings if specified in the query parameter.
auto it = base::ranges::find(color_id_sets, set_name);
if (it == color_id_sets.end()) {
return std::string();
}
color_id_sets.erase(it);
std::string css_string;
for (ui::ColorId id = start; id < end; ++id) {
const SkColor color = color_provider.GetColor(id);
std::string css_id_to_color_mapping =
base::StringPrintf("%s:%s;", color_css_name.Run(id).c_str(),
ui::ConvertSkColorToCSSColor(color).c_str());
base::StrAppend(&css_string, {css_id_to_color_mapping});
if (generate_rgb_vars) {
// Also generate a r,g,b string for each color so apps can construct
// colors with their own opacities in css.
const std::string css_rgb_color_str =
color_utils::SkColorToRgbString(color);
const std::string css_id_to_rgb_color_mapping =
base::StringPrintf("%s-rgb:%s;", color_css_name.Run(id).c_str(),
css_rgb_color_str.c_str());
base::StrAppend(&css_string, {css_id_to_rgb_color_mapping});
}
}
return css_string;
};
// Convenience lambda for wrapping
// |ConvertColorProviderColorIdToCSSColorId|.
auto generate_color_provider_mapping = [&generate_color_mapping](
std::string set_name,
ui::ColorId start, ui::ColorId end,
std::string (*color_id_name)(
ui::ColorId)) {
auto color_id_to_css_name = base::BindRepeating(
[](std::string (*color_id_name)(ui::ColorId), ui::ColorId id) {
return ui::ConvertColorProviderColorIdToCSSColorId(color_id_name(id));
},
color_id_name);
return generate_color_mapping(set_name, start, end, color_id_to_css_name);
};
std::string css_selector;
if (shadow_host) {
css_selector = ":host";
} else {
// This selector requires more specificity than other existing CSS
// selectors that define variables. We increase the specifity by adding
// a pseudoselector.
css_selector = "html:not(#z)";
}
const auto* theme_service =
ThemeServiceFactory::GetForProfile(profile_->GetOriginalProfile());
std::string theme_id;
if (theme_service->GetIsGrayscale()) {
theme_id = "--user-color-source: baseline-grayscale;";
} else if (theme_service->GetIsBaseline()) {
theme_id = "--user-color-source: baseline-default;";
}
std::string css_string = base::StrCat(
{css_selector, "{", theme_id,
generate_color_provider_mapping("ui", ui::kUiColorsStart,
ui::kUiColorsEnd, ui::ColorIdName),
generate_color_provider_mapping("chrome", kChromeColorsStart,
kChromeColorsEnd, &ChromeColorIdName),
#if BUILDFLAG(IS_CHROMEOS_ASH)
generate_color_mapping("ref", cros_tokens::kCrosRefColorsStart,
cros_tokens::kCrosRefColorsEnd,
base::BindRepeating(cros_tokens::ColorIdName)),
generate_color_mapping("sys", cros_tokens::kCrosSysColorsStart,
cros_tokens::kCrosSysColorsEnd,
base::BindRepeating(cros_tokens::ColorIdName)),
generate_color_mapping("legacy", cros_tokens::kLegacySemanticColorsStart,
cros_tokens::kLegacySemanticColorsEnd,
base::BindRepeating(cros_tokens::ColorIdName)),
#endif // BUILDFLAG(IS_CHROMEOS_ASH)
"}"});
if (!color_id_sets.empty()) {
LOG(ERROR)
<< "Unrecognized color set(s) specified for chrome://theme/colors.css: "
<< base::JoinString(color_id_sets, ",");
std::move(callback).Run(nullptr);
return;
}
std::move(callback).Run(
base::MakeRefCounted<base::RefCountedString>(std::move(css_string)));
// Measures the time it takes to generate the colors.css and queue it for the
// renderer.
UmaHistogramTimes("WebUI.ColorsStylesheetServingDuration", timer.Elapsed());
}
std::string ThemeSource::GetAccessControlAllowOriginForOrigin(
const std::string& origin) {
std::string allowed_origin_prefix = content::kChromeUIScheme;
allowed_origin_prefix += "://";
if (base::StartsWith(origin, allowed_origin_prefix,
base::CompareCase::SENSITIVE)) {
return origin;
}
return content::URLDataSource::GetAccessControlAllowOriginForOrigin(origin);
}
#if BUILDFLAG(IS_CHROMEOS_ASH)
void ThemeSource::SendTypographyCss(
content::URLDataSource::GotDataCallback callback) {
if (!chromeos::features::IsJellyEnabled()) {
std::move(callback).Run(base::MakeRefCounted<base::RefCountedString>(
std::string("/* This file is intentionally blank */")));
return;
}
const ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
std::move(callback).Run(rb.LoadDataResourceBytesForScale(
IDR_CROS_STYLES_UI_CHROMEOS_STYLES_CROS_TYPOGRAPHY_CSS,
ui::kScaleFactorNone));
}
#endif
std::string ThemeSource::GetContentSecurityPolicy(
network::mojom::CSPDirectiveName directive) {
if (directive == network::mojom::CSPDirectiveName::DefaultSrc &&
serve_untrusted_) {
// TODO(crbug.com/40693568): Audit and tighten CSP.
return std::string();
}
return content::URLDataSource::GetContentSecurityPolicy(directive);
}