summaryrefslogtreecommitdiffstats
path: root/devtools/client/shared/theme.js
blob: e3c3ff608dffa1a95841eaae38664d025d505ee2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

"use strict";

const variableFileContents = require("raw!chrome://devtools/skin/variables.css");

const THEME_SELECTOR_STRINGS = {
  light: ":root.theme-light {",
  dark: ":root.theme-dark {",
  root: ":root {",
};
const THEME_PREF = "devtools.theme";

/**
 * Takes a theme name and returns the contents of its variable rule block.
 * The first time this runs fetches the variables CSS file and caches it.
 */
function getThemeFile(name) {
  // If there's no theme expected for this name, use `light` as default.
  const selector = THEME_SELECTOR_STRINGS[name] || THEME_SELECTOR_STRINGS.light;

  // This is a pretty naive way to find the contents between:
  // selector {
  //   name: val;
  // }
  // There is test coverage for this feature (browser_theme.js)
  // so if an } is introduced in the variables file it will catch that.
  let theme = variableFileContents;
  theme = theme.substring(theme.indexOf(selector));
  theme = theme.substring(0, theme.indexOf("}"));

  return theme;
}

/**
 * Returns the "auto" theme.
 */
const getAutoTheme = (exports.getAutoTheme = () => {
  return Services.appinfo.chromeColorSchemeIsDark ? "dark" : "light";
});

/**
 * Returns the string value of the current theme,
 * like "dark" or "light".
 */
const getTheme = (exports.getTheme = () => {
  const theme = Services.prefs.getCharPref(THEME_PREF);
  if (theme == "auto") {
    return getAutoTheme();
  }
  return theme;
});

/**
 * Returns a color indicated by `type` (like "toolbar-background", or
 * "highlight-red"), with the ability to specify a theme, or use whatever the
 * current theme is if left unset. If theme not found, falls back to "light"
 * theme. Returns null if the type cannot be found for the theme given.
 */
/* eslint-disable no-unused-vars */
const getColor = (exports.getColor = (type, theme) => {
  const themeName = theme || getTheme();
  let themeFile = getThemeFile(themeName);
  let match = themeFile.match(new RegExp("--theme-" + type + ": (.*);"));
  const variableMatch = match ? match[1].match(/var\((.*)\)/) : null;

  // Check if the match is a color variable and retrieve the value of the color variable
  // if needed
  if (variableMatch) {
    themeFile = getThemeFile("root");
    match = themeFile.match(new RegExp(`${variableMatch[1]}: (.*);`));
  }

  // Return the appropriate variable in the theme, or otherwise, null.
  return match ? match[1] : null;
});

/**
 * Set the theme preference.
 */
const setTheme = (exports.setTheme = newTheme => {
  Services.prefs.setCharPref(THEME_PREF, newTheme);
});

/**
 * Add an observer for theme changes.
 */
const addThemeObserver = (exports.addThemeObserver = observer => {
  Services.obs.addObserver(observer, "look-and-feel-changed");
  Services.prefs.addObserver(THEME_PREF, observer);
});

/**
 * Remove an observer for theme changes.
 */
const removeThemeObserver = (exports.removeThemeObserver = observer => {
  Services.obs.removeObserver(observer, "look-and-feel-changed");
  Services.prefs.removeObserver(THEME_PREF, observer);
});
/* eslint-enable */