devtools_ui_theme_data_source.cc 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. // Copyright (c) 2024 The Chromium Authors. All rights reserved.
  2. // Use of this source code is governed by the MIT license that can be
  3. // found in the LICENSE file.
  4. #include "shell/browser/ui/devtools_ui_theme_data_source.h"
  5. #include <memory>
  6. #include <string>
  7. #include <string_view>
  8. #include <utility>
  9. #include "base/memory/ref_counted_memory.h"
  10. #include "base/metrics/histogram_functions.h"
  11. #include "base/strings/strcat.h"
  12. #include "base/strings/string_number_conversions.h"
  13. #include "base/strings/string_split.h"
  14. #include "base/strings/string_util.h"
  15. #include "chrome/browser/ui/color/chrome_color_id.h"
  16. #include "chrome/browser/ui/color/chrome_color_provider_utils.h"
  17. #include "chrome/common/webui_url_constants.h"
  18. #include "content/public/browser/devtools_frontend_host.h"
  19. #include "content/public/browser/url_data_source.h"
  20. #include "content/public/browser/web_contents.h"
  21. #include "content/public/browser/web_ui.h"
  22. #include "net/base/url_util.h"
  23. #include "ui/base/webui/web_ui_util.h"
  24. #include "ui/color/color_provider.h"
  25. #include "ui/color/color_provider_utils.h"
  26. namespace electron {
  27. namespace {
  28. GURL GetThemeUrl(const std::string& path) {
  29. return GURL(std::string(content::kChromeDevToolsScheme) +
  30. url::kStandardSchemeSeparator +
  31. std::string(chrome::kChromeUIThemeHost) + "/" + path);
  32. }
  33. scoped_refptr<base::RefCountedMemory> CreateNotFoundResponse() {
  34. const char kHttpNotFound[] = "HTTP/1.1 404 Not Found\n\n";
  35. return base::MakeRefCounted<base::RefCountedStaticMemory>(
  36. base::byte_span_from_cstring(kHttpNotFound));
  37. }
  38. std::string GetMimeTypeForUrl(const GURL& url) {
  39. std::string filename = url.ExtractFileName();
  40. if (base::EndsWith(filename, ".html", base::CompareCase::INSENSITIVE_ASCII)) {
  41. return "text/html";
  42. } else if (base::EndsWith(filename, ".css",
  43. base::CompareCase::INSENSITIVE_ASCII)) {
  44. return "text/css";
  45. } else if (base::EndsWith(filename, ".js",
  46. base::CompareCase::INSENSITIVE_ASCII) ||
  47. base::EndsWith(filename, ".mjs",
  48. base::CompareCase::INSENSITIVE_ASCII)) {
  49. return "application/javascript";
  50. } else if (base::EndsWith(filename, ".png",
  51. base::CompareCase::INSENSITIVE_ASCII)) {
  52. return "image/png";
  53. } else if (base::EndsWith(filename, ".map",
  54. base::CompareCase::INSENSITIVE_ASCII)) {
  55. return "application/json";
  56. } else if (base::EndsWith(filename, ".ts",
  57. base::CompareCase::INSENSITIVE_ASCII)) {
  58. return "application/x-typescript";
  59. } else if (base::EndsWith(filename, ".gif",
  60. base::CompareCase::INSENSITIVE_ASCII)) {
  61. return "image/gif";
  62. } else if (base::EndsWith(filename, ".svg",
  63. base::CompareCase::INSENSITIVE_ASCII)) {
  64. return "image/svg+xml";
  65. } else if (base::EndsWith(filename, ".manifest",
  66. base::CompareCase::INSENSITIVE_ASCII)) {
  67. return "text/cache-manifest";
  68. }
  69. return "text/html";
  70. }
  71. } // namespace
  72. std::string ThemeDataSource::GetSource() {
  73. // kChromeUIThemeHost
  74. return chrome::kChromeUIThemeHost;
  75. }
  76. void ThemeDataSource::StartDataRequest(
  77. const GURL& url,
  78. const content::WebContents::Getter& wc_getter,
  79. GotDataCallback callback) {
  80. // TODO(crbug.com/40050262): Simplify usages of |path| since |url| is
  81. // available.
  82. const std::string path = content::URLDataSource::URLToRequestPath(url);
  83. // Default scale factor if not specified.
  84. float scale = 1.0f;
  85. // All frames by default if not specified.
  86. int frame = -1;
  87. std::string parsed_path;
  88. webui::ParsePathAndImageSpec(GetThemeUrl(path), &parsed_path, &scale, &frame);
  89. // kColorsCssPath should stay consistent with COLORS_CSS_SELECTOR in
  90. // colors_css_updater.js.
  91. constexpr std::string_view kColorsCssPath = "colors.css";
  92. if (parsed_path == kColorsCssPath) {
  93. SendColorsCss(url, wc_getter, std::move(callback));
  94. return;
  95. }
  96. std::move(callback).Run(CreateNotFoundResponse());
  97. }
  98. std::string ThemeDataSource::GetMimeType(const GURL& url) {
  99. return GetMimeTypeForUrl(url);
  100. }
  101. void ThemeDataSource::SendColorsCss(
  102. const GURL& url,
  103. const content::WebContents::Getter& wc_getter,
  104. content::URLDataSource::GotDataCallback callback) {
  105. const ui::ColorProvider& color_provider = wc_getter.Run()->GetColorProvider();
  106. std::string sets_param;
  107. std::vector<std::string_view> color_id_sets;
  108. bool generate_rgb_vars = false;
  109. std::string generate_rgb_vars_query_value;
  110. if (net::GetValueForKeyInQuery(url, "generate_rgb_vars",
  111. &generate_rgb_vars_query_value)) {
  112. generate_rgb_vars =
  113. base::ToLowerASCII(generate_rgb_vars_query_value) == "true";
  114. }
  115. bool shadow_host = false;
  116. std::string shadow_host_query_value;
  117. if (net::GetValueForKeyInQuery(url, "shadow_host",
  118. &shadow_host_query_value)) {
  119. shadow_host = base::ToLowerASCII(shadow_host_query_value) == "true";
  120. }
  121. if (!net::GetValueForKeyInQuery(url, "sets", &sets_param)) {
  122. LOG(ERROR)
  123. << "colors.css requires a 'sets' query parameter to specify the "
  124. "color "
  125. "id sets returned e.g chrome://theme/colors.css?sets=ui,chrome";
  126. std::move(callback).Run(nullptr);
  127. return;
  128. }
  129. color_id_sets = base::SplitStringPiece(sets_param, ",", base::TRIM_WHITESPACE,
  130. base::SPLIT_WANT_ALL);
  131. using ColorIdCSSCallback = base::RepeatingCallback<std::string(ui::ColorId)>;
  132. auto generate_color_mapping =
  133. [&color_id_sets, &color_provider, &generate_rgb_vars](
  134. std::string set_name, ui::ColorId start, ui::ColorId end,
  135. ColorIdCSSCallback color_css_name) {
  136. // Only return these mappings if specified in the query parameter.
  137. auto it = std::ranges::find(color_id_sets, set_name);
  138. if (it == color_id_sets.end()) {
  139. return std::string();
  140. }
  141. color_id_sets.erase(it);
  142. std::string css_string;
  143. for (ui::ColorId id = start; id < end; ++id) {
  144. const SkColor color = color_provider.GetColor(id);
  145. std::string css_id_to_color_mapping =
  146. base::StringPrintf("%s:%s;", color_css_name.Run(id).c_str(),
  147. ui::ConvertSkColorToCSSColor(color).c_str());
  148. base::StrAppend(&css_string, {css_id_to_color_mapping});
  149. if (generate_rgb_vars) {
  150. // Also generate a r,g,b string for each color so apps can construct
  151. // colors with their own opacities in css.
  152. const std::string css_rgb_color_str =
  153. color_utils::SkColorToRgbString(color);
  154. const std::string css_id_to_rgb_color_mapping =
  155. base::StringPrintf("%s-rgb:%s;", color_css_name.Run(id).c_str(),
  156. css_rgb_color_str.c_str());
  157. base::StrAppend(&css_string, {css_id_to_rgb_color_mapping});
  158. }
  159. }
  160. return css_string;
  161. };
  162. // Convenience lambda for wrapping
  163. // |ConvertColorProviderColorIdToCSSColorId|.
  164. auto generate_color_provider_mapping = [&generate_color_mapping](
  165. std::string set_name,
  166. ui::ColorId start, ui::ColorId end,
  167. std::string (*color_id_name)(
  168. ui::ColorId)) {
  169. auto color_id_to_css_name = base::BindRepeating(
  170. [](std::string (*color_id_name)(ui::ColorId), ui::ColorId id) {
  171. return ui::ConvertColorProviderColorIdToCSSColorId(color_id_name(id));
  172. },
  173. color_id_name);
  174. return generate_color_mapping(set_name, start, end, color_id_to_css_name);
  175. };
  176. std::string css_selector;
  177. if (shadow_host) {
  178. css_selector = ":host";
  179. } else {
  180. // This selector requires more specificity than other existing CSS
  181. // selectors that define variables. We increase the specifity by adding
  182. // a pseudoselector.
  183. css_selector = "html:not(#z)";
  184. }
  185. std::string css_string = base::StrCat(
  186. {css_selector, "{", "--user-color-source: baseline-default;",
  187. generate_color_provider_mapping("ui", ui::kUiColorsStart,
  188. ui::kUiColorsEnd, ui::ColorIdName),
  189. generate_color_provider_mapping("chrome", kChromeColorsStart,
  190. kChromeColorsEnd, &ChromeColorIdName),
  191. "}"});
  192. if (!color_id_sets.empty()) {
  193. LOG(ERROR) << "Unrecognized color set(s) specified for "
  194. "chrome://theme/colors.css: "
  195. << base::JoinString(color_id_sets, ",");
  196. std::move(callback).Run(nullptr);
  197. return;
  198. }
  199. std::move(callback).Run(
  200. base::MakeRefCounted<base::RefCountedString>(std::move(css_string)));
  201. }
  202. } // namespace electron