xref: /haiku/src/preferences/appearance/APRView.cpp (revision 9a6a20d4689307142a7ed26a1437ba47e244e73f)
1 /*
2  * Copyright 2002-2015 Haiku, Inc. All rights reserved.
3  * Distributed under the terms of the MIT License.
4  *
5  * Authors:
6  *		DarkWyrm, darkwyrm@earthlink.net
7  *		Rene Gollent, rene@gollent.com
8  *		John Scipione, jscipione@gmail.com
9  *		Joseph Groover <looncraz@looncraz.net>
10  */
11 
12 
13 #include "APRView.h"
14 
15 #include <stdio.h>
16 
17 #include <Alert.h>
18 #include <Catalog.h>
19 #include <DefaultColors.h>
20 #include <Directory.h>
21 #include <Entry.h>
22 #include <File.h>
23 #include <HSL.h>
24 #include <LayoutBuilder.h>
25 #include <Locale.h>
26 #include <Messenger.h>
27 #include <Path.h>
28 #include <SpaceLayoutItem.h>
29 
30 #include "APRWindow.h"
31 #include "defs.h"
32 #include "ColorPreview.h"
33 #include "Colors.h"
34 #include "ColorWhichListView.h"
35 #include "ColorWhichItem.h"
36 
37 
38 #undef B_TRANSLATION_CONTEXT
39 #define B_TRANSLATION_CONTEXT "Colors tab"
40 
41 #define COLOR_DROPPED 'cldp'
42 #define AUTO_ADJUST_CHANGED 'madj'
43 
44 
45 APRView::APRView(const char* name)
46 	:
47 	BView(name, B_WILL_DRAW)
48 {
49 	SetViewUIColor(B_PANEL_BACKGROUND_COLOR);
50 
51 	LoadSettings();
52 
53 	fAutoSelectCheckBox = new BCheckBox(B_TRANSLATE("Automatically pick secondary colors"),
54 		new BMessage(AUTO_ADJUST_CHANGED));
55 	fAutoSelectCheckBox->SetValue(true);
56 
57 	// Set up list of color attributes
58 	fAttrList = new ColorWhichListView("AttributeList");
59 
60 	fScrollView = new BScrollView("ScrollView", fAttrList, 0, false, true);
61 	fScrollView->SetViewUIColor(B_PANEL_BACKGROUND_COLOR);
62 
63 	_CreateItems();
64 
65 	fColorPreview = new ColorPreview(new BMessage(COLOR_DROPPED), 0);
66 	fColorPreview->SetExplicitAlignment(BAlignment(B_ALIGN_HORIZONTAL_CENTER,
67 		B_ALIGN_VERTICAL_CENTER));
68 
69 	fPicker = new BColorControl(B_ORIGIN, B_CELLS_32x8, 8.0,
70 		"picker", new BMessage(UPDATE_COLOR));
71 
72 	BLayoutBuilder::Group<>(this, B_VERTICAL)
73 		.Add(fAutoSelectCheckBox)
74 		.Add(fScrollView, 10.0)
75 		.AddGroup(B_HORIZONTAL, B_USE_DEFAULT_SPACING)
76 			.Add(fColorPreview)
77 			.AddGlue()
78 			.Add(fPicker)
79 			.End()
80 		.SetInsets(B_USE_WINDOW_SPACING);
81 
82 	fColorPreview->Parent()->SetExplicitMaxSize(
83 		BSize(B_SIZE_UNSET, fPicker->Bounds().Height()));
84 	fAttrList->SetSelectionMessage(new BMessage(ATTRIBUTE_CHOSEN));
85 }
86 
87 
88 APRView::~APRView()
89 {
90 }
91 
92 
93 void
94 APRView::AttachedToWindow()
95 {
96 	fAutoSelectCheckBox->SetTarget(this);
97 	fPicker->SetTarget(this);
98 	fAttrList->SetTarget(this);
99 	fColorPreview->SetTarget(this);
100 
101 	fAttrList->Select(0);
102 	SetViewUIColor(B_PANEL_BACKGROUND_COLOR);
103 }
104 
105 
106 void
107 APRView::MessageReceived(BMessage *msg)
108 {
109 	switch (msg->what) {
110 		case SET_COLOR:
111 		{
112 			rgb_color* color;
113 			ssize_t size;
114 			color_which which;
115 
116 			if (msg->FindData(kRGBColor, B_RGB_COLOR_TYPE,
117 					(const void**)&color, &size) == B_OK
118 				&& msg->FindUInt32(kWhich, (uint32*)&which) == B_OK) {
119 				_SetColor(which, *color);
120 				Window()->PostMessage(kMsgUpdate);
121 			}
122 			break;
123 		}
124 
125 		case SET_CURRENT_COLOR:
126 		{
127 			rgb_color* color;
128 			ssize_t size;
129 
130 			if (msg->FindData(kRGBColor, B_RGB_COLOR_TYPE,
131 					(const void**)&color, &size) == B_OK) {
132 				_SetCurrentColor(*color);
133 				Window()->PostMessage(kMsgUpdate);
134 			}
135 			break;
136 		}
137 
138 		case UPDATE_COLOR:
139 		{
140 			// Received from the color fPicker when its color changes
141 			rgb_color color = fPicker->ValueAsColor();
142 			_SetCurrentColor(color);
143 
144 			Window()->PostMessage(kMsgUpdate);
145 			break;
146 		}
147 
148 		case ATTRIBUTE_CHOSEN:
149 		{
150 			// Received when the user chooses a GUI fAttribute from the list
151 
152 			ColorWhichItem* item = (ColorWhichItem*)
153 				fAttrList->ItemAt(fAttrList->CurrentSelection());
154 			if (item == NULL)
155 				break;
156 
157 			fWhich = item->ColorWhich();
158 			rgb_color color = ui_color(fWhich);
159 			_SetCurrentColor(color);
160 			break;
161 		}
162 
163 		case AUTO_ADJUST_CHANGED:
164 		{
165 			_CreateItems();
166 			break;
167 		}
168 
169 		default:
170 			BView::MessageReceived(msg);
171 			break;
172 	}
173 }
174 
175 
176 void
177 APRView::LoadSettings()
178 {
179 	get_default_colors(&fDefaultColors);
180 	get_current_colors(&fCurrentColors);
181 	fPrevColors = fCurrentColors;
182 }
183 
184 
185 void
186 APRView::SetDefaults()
187 {
188 	_SetUIColors(fDefaultColors);
189 	_UpdatePreviews(fDefaultColors);
190 
191 	// Use a default color that stands out to show errors clearly
192 	rgb_color color = fDefaultColors.GetColor(ui_color_name(fWhich),
193 		make_color(255, 0, 255));
194 
195 	fPicker->SetValue(color);
196 	fColorPreview->SetColor(color);
197 	fColorPreview->Invalidate();
198 
199 	Window()->PostMessage(kMsgUpdate);
200 }
201 
202 
203 void
204 APRView::Revert()
205 {
206 	_SetUIColors(fPrevColors);
207 	_UpdatePreviews(fPrevColors);
208 
209 	rgb_color color = fPrevColors.GetColor(ui_color_name(fWhich),
210 		make_color(255, 0, 255));
211 	fPicker->SetValue(color);
212 	fColorPreview->SetColor(color);
213 	fColorPreview->Invalidate();
214 
215 	Window()->PostMessage(kMsgUpdate);
216 }
217 
218 
219 bool
220 APRView::IsDefaultable()
221 {
222 	return !fDefaultColors.HasSameData(fCurrentColors);
223 }
224 
225 
226 bool
227 APRView::IsRevertable()
228 {
229 	return !fPrevColors.HasSameData(fCurrentColors);
230 }
231 
232 
233 void
234 APRView::_CreateItems()
235 {
236 	while (fAttrList->CountItems() > 0)
237 		delete fAttrList->RemoveItem((int32)0);
238 
239 	const bool autoSelect = fAutoSelectCheckBox->Value();
240 	const int32 count = color_description_count();
241 	for (int32 i = 0; i < count; i++) {
242 		const ColorDescription& description = *get_color_description(i);
243 		const color_which which = description.which;
244 		if (autoSelect) {
245 			if (which != B_PANEL_BACKGROUND_COLOR
246 					&& which != B_STATUS_BAR_COLOR
247 					&& which != B_WINDOW_TAB_COLOR) {
248 				continue;
249 			}
250 		}
251 
252 		const char* text = B_TRANSLATE_NOCOLLECT(description.text);
253 		fAttrList->AddItem(new ColorWhichItem(text, which, ui_color(which)));
254 	}
255 
256 	if (Window() != NULL)
257 		fAttrList->Select(0);
258 }
259 
260 
261 void
262 APRView::_UpdatePreviews(const BMessage& colors)
263 {
264 	rgb_color color;
265 	for (int32 i = color_description_count() - 1; i >= 0; i--) {
266 		ColorWhichItem* item = static_cast<ColorWhichItem*>(fAttrList->ItemAt(i));
267 		if (item == NULL)
268 			continue;
269 
270 		color = colors.GetColor(ui_color_name(item->ColorWhich()),
271 			make_color(255, 0, 255));
272 
273 		item->SetColor(color);
274 		fAttrList->InvalidateItem(i);
275 	}
276 }
277 
278 
279 void
280 APRView::_SetUIColors(const BMessage& colors)
281 {
282 	set_ui_colors(&colors);
283 	fCurrentColors = colors;
284 }
285 
286 
287 void
288 APRView::_SetCurrentColor(rgb_color color)
289 {
290 	_SetColor(fWhich, color);
291 
292 	int32 currentIndex = fAttrList->CurrentSelection();
293 	ColorWhichItem* item = (ColorWhichItem*)fAttrList->ItemAt(currentIndex);
294 	if (item != NULL) {
295 		item->SetColor(color);
296 		fAttrList->InvalidateItem(currentIndex);
297 	}
298 
299 	fPicker->SetValue(color);
300 	fColorPreview->SetColor(color);
301 	fColorPreview->Invalidate();
302 }
303 
304 
305 void
306 APRView::_SetColor(color_which which, rgb_color color)
307 {
308 	_SetOneColor(which, color);
309 
310 	if (!fAutoSelectCheckBox->Value())
311 		return;
312 
313 	// Protect against accidentally overwriting colors.
314 	if (ui_color(which) == color)
315 		return;
316 
317 	if (which == B_PANEL_BACKGROUND_COLOR) {
318 		const bool isDark = color.IsDark();
319 
320 		_SetOneColor(B_MENU_BACKGROUND_COLOR, color);
321 		_SetOneColor(B_SCROLL_BAR_THUMB_COLOR, color);
322 
323 		const rgb_color menuSelectedBackground
324 			= tint_color(color, isDark ? 0.8 /* lighten "< 1" */ : B_DARKEN_2_TINT);
325 		_SetOneColor(B_MENU_SELECTED_BACKGROUND_COLOR, menuSelectedBackground);
326 
327 		const rgb_color controlBackground = tint_color(color, isDark
328 			? 0.8 /* lighten "< 1" */ : 0.25 /* lighten "> 2" */);
329 		_SetOneColor(B_CONTROL_BACKGROUND_COLOR, controlBackground);
330 
331 		const rgb_color controlBorder
332 			= tint_color(color, isDark ? 0.4875 : 1.20 /* lighten/darken "1.5" */);
333 		_SetOneColor(B_CONTROL_BORDER_COLOR, controlBorder);
334 
335 		const rgb_color windowBorder = tint_color(color, 0.75);
336 		_SetOneColor(B_WINDOW_BORDER_COLOR, windowBorder);
337 
338 		const rgb_color inactiveWindowBorder = tint_color(color, B_LIGHTEN_1_TINT);
339 		_SetOneColor(B_WINDOW_INACTIVE_TAB_COLOR, inactiveWindowBorder);
340 		_SetOneColor(B_WINDOW_INACTIVE_BORDER_COLOR, inactiveWindowBorder);
341 
342 		const rgb_color listSelectedBackground
343 			= tint_color(color, isDark ? 0.77 : 1.12 /* lighten/darken "< 1" */ );
344 		_SetOneColor(B_LIST_SELECTED_BACKGROUND_COLOR, listSelectedBackground);
345 
346 		const color_which fromDefaults[] = {
347 			B_MENU_ITEM_TEXT_COLOR,
348 			B_MENU_SELECTED_ITEM_TEXT_COLOR,
349 			B_MENU_SELECTED_BORDER_COLOR,
350 			B_PANEL_TEXT_COLOR,
351 			B_DOCUMENT_BACKGROUND_COLOR,
352 			B_DOCUMENT_TEXT_COLOR,
353 			B_CONTROL_TEXT_COLOR,
354 			B_NAVIGATION_PULSE_COLOR,
355 			B_WINDOW_INACTIVE_TEXT_COLOR,
356 			B_LIST_BACKGROUND_COLOR,
357 			B_LIST_ITEM_TEXT_COLOR,
358 			B_LIST_SELECTED_ITEM_TEXT_COLOR,
359 
360 			B_SHINE_COLOR,
361 			B_SHADOW_COLOR,
362 
363 			B_LINK_TEXT_COLOR,
364 			B_LINK_HOVER_COLOR,
365 			B_LINK_ACTIVE_COLOR,
366 			B_LINK_VISITED_COLOR,
367 		};
368 		for (size_t i = 0; i < B_COUNT_OF(fromDefaults); i++)
369 			_SetOneColor(fromDefaults[i], BPrivate::GetSystemColor(fromDefaults[i], isDark));
370 	} else if (which == B_STATUS_BAR_COLOR) {
371 		const hsl_color statusColorHSL = hsl_color::from_rgb(color);
372 
373 		hsl_color controlHighlight = statusColorHSL;
374 		controlHighlight.saturation = max_c(0.2f, controlHighlight.saturation / 2.f);
375 		_SetOneColor(B_CONTROL_HIGHLIGHT_COLOR, controlHighlight.to_rgb());
376 
377 		hsl_color controlMark = statusColorHSL;
378 		controlMark.saturation = max_c(0.2f, controlMark.saturation * 0.67f);
379 		controlMark.lightness = max_c(0.25f, controlMark.lightness * 0.55f);
380 		_SetOneColor(B_CONTROL_MARK_COLOR, controlMark.to_rgb());
381 
382 		rgb_color keyboardNav; {
383 			hsl_color keyboardNavHSL = statusColorHSL;
384 			keyboardNavHSL.lightness = max_c(0.2f, keyboardNavHSL.lightness * 0.75f);
385 			keyboardNav = keyboardNavHSL.to_rgb();
386 
387 			// Use primary color channel only.
388 			if (keyboardNav.blue >= max_c(keyboardNav.red, keyboardNav.green))
389 				keyboardNav.red = keyboardNav.green = 0;
390 			else if (keyboardNav.red >= max_c(keyboardNav.green, keyboardNav.blue))
391 				keyboardNav.green = keyboardNav.blue = 0;
392 			else
393 				keyboardNav.red = keyboardNav.blue = 0;
394 		}
395 		_SetOneColor(B_KEYBOARD_NAVIGATION_COLOR, keyboardNav);
396 	} else if (which == B_WINDOW_TAB_COLOR) {
397 		const bool isDark = color.IsDark();
398 		const hsl_color tabColorHSL = hsl_color::from_rgb(color);
399 		const float tabColorSaturation
400 			= tabColorHSL.saturation != 0 ? tabColorHSL.saturation : tabColorHSL.lightness;
401 
402 		_SetOneColor(B_WINDOW_TEXT_COLOR,
403 			BPrivate::GetSystemColor(B_WINDOW_TEXT_COLOR, isDark));
404 		_SetOneColor(B_TOOL_TIP_TEXT_COLOR,
405 			BPrivate::GetSystemColor(B_TOOL_TIP_TEXT_COLOR, isDark));
406 
407 		const rgb_color toolTipBackground = tint_color(color, isDark ? 1.7 : 0.15);
408 		_SetOneColor(B_TOOL_TIP_BACKGROUND_COLOR, toolTipBackground);
409 
410 		hsl_color success = hsl_color::from_rgb(BPrivate::GetSystemColor(B_SUCCESS_COLOR, isDark));
411 		success.saturation = max_c(0.25f, tabColorSaturation * 0.68f);
412 		_SetOneColor(B_SUCCESS_COLOR, success.to_rgb());
413 
414 		hsl_color failure = hsl_color::from_rgb(BPrivate::GetSystemColor(B_FAILURE_COLOR, isDark));
415 		failure.saturation = max_c(0.25f, tabColorSaturation);
416 		_SetOneColor(B_FAILURE_COLOR, failure.to_rgb());
417 	}
418 }
419 
420 
421 void
422 APRView::_SetOneColor(color_which which, rgb_color color)
423 {
424 	if (ui_color(which) == color)
425 		return;
426 
427 	set_ui_color(which, color);
428 	fCurrentColors.SetColor(ui_color_name(which), color);
429 }
430