1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-07-23 20:57:41 +00:00

WindowServer: New title bar vars for themes

The theming system can now control title bar height, title button
size, title stripe color and the title text shadow color.
The implemented theme metrics system could be later extended to LibGUI
to allow themes to change widget padding, border width, etc.
This commit is contained in:
Nullspeak 2020-07-17 10:27:55 +10:00 committed by Andreas Kling
parent 8e364b9780
commit 51b2b0d5e5
7 changed files with 142 additions and 10 deletions

View file

@ -0,0 +1,62 @@
[Colors]
DesktopBackground=#171717
ActiveWindowBorder1=black
ActiveWindowBorder2=#1f1f1f
ActiveWindowTitle=white
InactiveWindowBorder1=#171717
InactiveWindowBorder2=#1f1f1f
InactiveWindowTitle=#aaaaaa
MovingWindowBorder1=black
MovingWindowBorder2=#2f1f1f
MovingWindowTitle=white
HighlightWindowBorder1=black
HighlightWindowBorder2=#3f2f0f
HighlightWindowTitle=white
WindowTitleShadow=#00000000
WindowTitleStripes=#00000000
MenuBase=#1f1f1f
MenuBaseText=white
MenuStripe=#171717
MenuSelection=#ff7f00
MenuSelectionText=black
Window=#1f1f1f
WindowText=white
Button=#1f1f1f
ButtonText=white
Base=#1f1f1f
BaseText=white
ThreedHighlight=#2f2f2f
ThreedShadow1=#171717
ThreedShadow2=#0f0f0f
HoverHighlight=#272727
Selection=#ff7f00
SelectionText=black
InactiveSelection=#7f0000
InactiveSelectionText=black
RubberBandFill=#ff7f002f
RubberBandBorder=#ff7f00
Link=#88c
ActiveLink=#c88
VisitedLink=#c8c
Ruler=#0f0f0f
RulerBorder=#2f2f2f
RulerActiveText=white
RulerInactiveText=#2f2f2f
TextCursor=#ff4f4f
FocusOutline=#4f4f4f
SyntaxComment=#4fbf4f
SyntaxNumber=white
SyntaxString=#ff8f4f
SyntaxType=#6f8fff
SyntaxPunctuation=white
SyntaxOperator=white
SyntaxKeyword=#cf7fff
SyntaxControlKeyword=orchid
SyntaxIdentifier=white
SyntaxPreprocessorStatement=#ffafff
SyntaxPreprocessorValue=orange
[Metrics]
TitleHeight=24
TitleButtonWidth=32
TitleButtonHeight=18

View file

@ -12,6 +12,8 @@ MovingWindowTitle=white
HighlightWindowBorder1=#a10d0d HighlightWindowBorder1=#a10d0d
HighlightWindowBorder2=#fabbbb HighlightWindowBorder2=#fabbbb
HighlightWindowTitle=white HighlightWindowTitle=white
WindowTitleShadow=#421405
WindowTitleStripes=#6e2209
MenuBase=white MenuBase=white
MenuBaseText=black MenuBaseText=black
MenuStripe=#bbb7b0 MenuStripe=#bbb7b0
@ -53,3 +55,8 @@ SyntaxControlKeyword=black
SyntaxIdentifier=#092e64 SyntaxIdentifier=#092e64
SyntaxPreprocessorStatement=#008080 SyntaxPreprocessorStatement=#008080
SyntaxPreprocessorValue=#800000 SyntaxPreprocessorValue=#800000
[Metrics]
TitleHeight=19
TitleButtonWidth=15
TitleButtonHeight=15

View file

@ -61,6 +61,12 @@ Color PaletteImpl::color(ColorRole role) const
return theme().color[(int)role]; return theme().color[(int)role];
} }
int PaletteImpl::metric(MetricRole role) const
{
ASSERT((int)role < (int)MetricRole::__Count);
return theme().metric[(int)role];
}
NonnullRefPtr<PaletteImpl> PaletteImpl::clone() const NonnullRefPtr<PaletteImpl> PaletteImpl::clone() const
{ {
auto new_theme_buffer = SharedBuffer::create_with_size(m_theme_buffer->size()); auto new_theme_buffer = SharedBuffer::create_with_size(m_theme_buffer->size());
@ -76,6 +82,14 @@ void Palette::set_color(ColorRole role, Color color)
theme.color[(int)role] = color; theme.color[(int)role] = color;
} }
void Palette::set_metric(MetricRole role, int value)
{
if (m_impl->ref_count() != 1)
m_impl = m_impl->clone();
auto& theme = const_cast<SystemTheme&>(impl().theme());
theme.metric[(int)role] = value;
}
PaletteImpl::~PaletteImpl() PaletteImpl::~PaletteImpl()
{ {
} }

View file

@ -44,6 +44,7 @@ public:
NonnullRefPtr<PaletteImpl> clone() const; NonnullRefPtr<PaletteImpl> clone() const;
Color color(ColorRole) const; Color color(ColorRole) const;
int metric(MetricRole) const;
const SystemTheme& theme() const; const SystemTheme& theme() const;
void replace_internal_buffer(Badge<GUI::Application>, SharedBuffer& buffer); void replace_internal_buffer(Badge<GUI::Application>, SharedBuffer& buffer);
@ -79,6 +80,8 @@ public:
Color highlight_window_border1() const { return color(ColorRole::HighlightWindowBorder1); } Color highlight_window_border1() const { return color(ColorRole::HighlightWindowBorder1); }
Color highlight_window_border2() const { return color(ColorRole::HighlightWindowBorder2); } Color highlight_window_border2() const { return color(ColorRole::HighlightWindowBorder2); }
Color highlight_window_title() const { return color(ColorRole::HighlightWindowTitle); } Color highlight_window_title() const { return color(ColorRole::HighlightWindowTitle); }
Color window_title_stripes() const { return color(ColorRole::WindowTitleStripes); }
Color window_title_shadow() const { return color(ColorRole::WindowTitleShadow); }
Color menu_stripe() const { return color(ColorRole::MenuStripe); } Color menu_stripe() const { return color(ColorRole::MenuStripe); }
Color menu_base() const { return color(ColorRole::MenuBase); } Color menu_base() const { return color(ColorRole::MenuBase); }
Color menu_base_text() const { return color(ColorRole::MenuBaseText); } Color menu_base_text() const { return color(ColorRole::MenuBaseText); }
@ -117,9 +120,15 @@ public:
Color syntax_preprocessor_statement() const { return color(ColorRole::SyntaxPreprocessorStatement); } Color syntax_preprocessor_statement() const { return color(ColorRole::SyntaxPreprocessorStatement); }
Color syntax_preprocessor_value() const { return color(ColorRole::SyntaxPreprocessorValue); } Color syntax_preprocessor_value() const { return color(ColorRole::SyntaxPreprocessorValue); }
int window_title_height() const { return metric(MetricRole::TitleHeight); }
int window_title_button_width() const { return metric(MetricRole::TitleButtonWidth); }
int window_title_button_height() const { return metric(MetricRole::TitleButtonHeight); }
Color color(ColorRole role) const { return m_impl->color(role); } Color color(ColorRole role) const { return m_impl->color(role); }
int metric(MetricRole role) const { return m_impl->metric(role); }
void set_color(ColorRole, Color); void set_color(ColorRole, Color);
void set_metric(MetricRole, int);
const SystemTheme& theme() const { return m_impl->theme(); } const SystemTheme& theme() const { return m_impl->theme(); }

View file

@ -67,6 +67,24 @@ RefPtr<SharedBuffer> load_system_theme(const String& path)
return color.value(); return color.value();
}; };
auto get_metric = [&](auto& name, auto role) {
int metric = file->read_num_entry("Metrics", name, -1);
if (metric == -1) {
switch (role) {
case (int)MetricRole::TitleHeight:
return 19;
case (int)MetricRole::TitleButtonHeight:
return 15;
case (int)MetricRole::TitleButtonWidth:
return 15;
default:
dbg() << "Metric " << name << " has no fallback value!";
return 16;
}
}
return metric;
};
#define DO_COLOR(x) \ #define DO_COLOR(x) \
data->color[(int)ColorRole::x] = get_color(#x) data->color[(int)ColorRole::x] = get_color(#x)
@ -98,6 +116,8 @@ RefPtr<SharedBuffer> load_system_theme(const String& path)
DO_COLOR(HighlightWindowBorder1); DO_COLOR(HighlightWindowBorder1);
DO_COLOR(HighlightWindowBorder2); DO_COLOR(HighlightWindowBorder2);
DO_COLOR(HighlightWindowTitle); DO_COLOR(HighlightWindowTitle);
DO_COLOR(WindowTitleShadow);
DO_COLOR(WindowTitleStripes);
DO_COLOR(MenuStripe); DO_COLOR(MenuStripe);
DO_COLOR(MenuBase); DO_COLOR(MenuBase);
DO_COLOR(MenuBaseText); DO_COLOR(MenuBaseText);
@ -126,6 +146,13 @@ RefPtr<SharedBuffer> load_system_theme(const String& path)
DO_COLOR(SyntaxPreprocessorStatement); DO_COLOR(SyntaxPreprocessorStatement);
DO_COLOR(SyntaxPreprocessorValue); DO_COLOR(SyntaxPreprocessorValue);
#define DO_METRIC(x) \
data->metric[(int)MetricRole::x] = get_metric(#x, (int)MetricRole::x)
DO_METRIC(TitleHeight);
DO_METRIC(TitleButtonWidth);
DO_METRIC(TitleButtonHeight);
buffer->seal(); buffer->seal();
buffer->share_globally(); buffer->share_globally();

View file

@ -47,6 +47,8 @@ enum class ColorRole {
HighlightWindowBorder1, HighlightWindowBorder1,
HighlightWindowBorder2, HighlightWindowBorder2,
HighlightWindowTitle, HighlightWindowTitle,
WindowTitleShadow,
WindowTitleStripes,
MenuStripe, MenuStripe,
MenuBase, MenuBase,
MenuBaseText, MenuBaseText,
@ -95,8 +97,17 @@ enum class ColorRole {
DisabledText = ThreedShadow1, DisabledText = ThreedShadow1,
}; };
enum class MetricRole {
NoRole,
TitleHeight,
TitleButtonWidth,
TitleButtonHeight,
__Count,
};
struct SystemTheme { struct SystemTheme {
Color color[(int)ColorRole::__Count]; Color color[(int)ColorRole::__Count];
int metric[(int)MetricRole::__Count];
}; };
const SystemTheme& current_system_theme(); const SystemTheme& current_system_theme();

View file

@ -39,8 +39,6 @@
namespace WindowServer { namespace WindowServer {
static const int window_titlebar_height = 19;
static const char* s_close_button_bitmap_data = { static const char* s_close_button_bitmap_data = {
"## ##" "## ##"
"### ###" "### ###"
@ -153,6 +151,7 @@ void WindowFrame::did_set_maximized(Badge<Window>, bool maximized)
Gfx::IntRect WindowFrame::title_bar_rect() const Gfx::IntRect WindowFrame::title_bar_rect() const
{ {
auto window_titlebar_height = WindowManager::the().palette().window_title_height();
if (m_window.type() == WindowType::Notification) if (m_window.type() == WindowType::Notification)
return { m_window.width() + 3, 3, window_titlebar_height, m_window.height() }; return { m_window.width() + 3, 3, window_titlebar_height, m_window.height() };
return { 4, 4, m_window.width(), window_titlebar_height }; return { 4, 4, m_window.width(), window_titlebar_height };
@ -207,8 +206,8 @@ void WindowFrame::paint_notification_frame(Gfx::Painter& painter)
int stripe_top = m_buttons.last().relative_rect().bottom() + 4; int stripe_top = m_buttons.last().relative_rect().bottom() + 4;
int stripe_bottom = m_window.height() - 3; int stripe_bottom = m_window.height() - 3;
if (stripe_top && stripe_bottom && stripe_top < stripe_bottom) { if (stripe_top && stripe_bottom && stripe_top < stripe_bottom) {
for (int i = 2; i <= window_titlebar_height - 2; i += 2) { for (int i = 2; i <= palette.window_title_height() - 2; i += 2) {
painter.draw_line({ titlebar_rect.x() + i, stripe_top }, { titlebar_rect.x() + i, stripe_bottom }, palette.active_window_border1()); painter.draw_line({ titlebar_rect.x() + i, stripe_top }, { titlebar_rect.x() + i, stripe_bottom }, palette.window_title_stripes());
} }
} }
} }
@ -241,7 +240,7 @@ void WindowFrame::paint_normal_frame(Gfx::Painter& painter)
int stripe_right = leftmost_button_rect.left() - 3; int stripe_right = leftmost_button_rect.left() - 3;
if (stripe_left && stripe_right && stripe_left < stripe_right) { if (stripe_left && stripe_right && stripe_left < stripe_right) {
for (int i = 2; i <= titlebar_inner_rect.height() - 2; i += 2) { for (int i = 2; i <= titlebar_inner_rect.height() - 2; i += 2) {
painter.draw_line({ stripe_left, titlebar_inner_rect.y() + i }, { stripe_right, titlebar_inner_rect.y() + i }, border_color); painter.draw_line({ stripe_left, titlebar_inner_rect.y() + i }, { stripe_right, titlebar_inner_rect.y() + i }, palette.window_title_stripes());
} }
} }
@ -259,7 +258,7 @@ void WindowFrame::paint_normal_frame(Gfx::Painter& painter)
auto clipped_title_rect = titlebar_title_rect; auto clipped_title_rect = titlebar_title_rect;
clipped_title_rect.set_width(stripe_right - clipped_title_rect.x()); clipped_title_rect.set_width(stripe_right - clipped_title_rect.x());
if (!clipped_title_rect.is_empty()) { if (!clipped_title_rect.is_empty()) {
painter.draw_text(clipped_title_rect.translated(1, 2), title_text, wm.window_title_font(), Gfx::TextAlignment::CenterLeft, border_color.darkened(0.4), Gfx::TextElision::Right); painter.draw_text(clipped_title_rect.translated(1, 2), title_text, wm.window_title_font(), Gfx::TextAlignment::CenterLeft, palette.window_title_shadow(), Gfx::TextElision::Right);
// FIXME: The translated(0, 1) wouldn't be necessary if we could center text based on its baseline. // FIXME: The translated(0, 1) wouldn't be necessary if we could center text based on its baseline.
painter.draw_text(clipped_title_rect.translated(0, 1), title_text, wm.window_title_font(), Gfx::TextAlignment::CenterLeft, title_color, Gfx::TextElision::Right); painter.draw_text(clipped_title_rect.translated(0, 1), title_text, wm.window_title_font(), Gfx::TextAlignment::CenterLeft, title_color, Gfx::TextElision::Right);
} }
@ -293,6 +292,7 @@ static Gfx::IntRect frame_rect_for_window(Window& window, const Gfx::IntRect& re
return rect; return rect;
auto type = window.type(); auto type = window.type();
auto window_titlebar_height = WindowManager::the().palette().window_title_height();
switch (type) { switch (type) {
case WindowType::Normal: case WindowType::Normal:
@ -331,8 +331,9 @@ void WindowFrame::invalidate_title_bar()
void WindowFrame::notify_window_rect_changed(const Gfx::IntRect& old_rect, const Gfx::IntRect& new_rect) void WindowFrame::notify_window_rect_changed(const Gfx::IntRect& old_rect, const Gfx::IntRect& new_rect)
{ {
int window_button_width = 15; auto palette = WindowManager::the().palette();
int window_button_height = 15; int window_button_width = palette.window_title_button_width();
int window_button_height = palette.window_title_button_height();
int pos; int pos;
if (m_window.type() == WindowType::Notification) if (m_window.type() == WindowType::Notification)
pos = title_bar_rect().top() + 2; pos = title_bar_rect().top() + 2;
@ -341,10 +342,11 @@ void WindowFrame::notify_window_rect_changed(const Gfx::IntRect& old_rect, const
for (auto& button : m_buttons) { for (auto& button : m_buttons) {
if (m_window.type() == WindowType::Notification) { if (m_window.type() == WindowType::Notification) {
Gfx::IntRect rect { 0, pos, window_button_width, window_button_height }; // The button height & width have to be equal or it leaks out of its area
Gfx::IntRect rect { 0, pos, window_button_height, window_button_height };
rect.center_horizontally_within(title_bar_rect()); rect.center_horizontally_within(title_bar_rect());
button.set_relative_rect(rect); button.set_relative_rect(rect);
pos += window_button_width; pos += window_button_height;
} else { } else {
pos -= window_button_width; pos -= window_button_width;
Gfx::IntRect rect { pos, 0, window_button_width, window_button_height }; Gfx::IntRect rect { pos, 0, window_button_width, window_button_height };