mirror of
https://github.com/9ParsonsB/Pulsar.git
synced 2025-04-05 17:39:39 -04:00
* [Core] Visual improvements to notifications - Notification title was previously missing; it's back! - Fixed window and label backgrounds to be correctly/consistently transparent. The click-thru bit is only partially working (transparent areas are click-thru). * [Core] Reduce notification font size to avoid wrapping Now that titles work, I noticed a few notification titles were wrapping. Examples: - System Discovery Complete (from Evaluator) - Minimum Distance Reached (from BioInsights) To address, I made the following adjustments: * Made the test notification title longer by adding the word "Popup" so it is now "Test Popup Notification". * Reduced the Title font size to 20 (from 24). * Moved the content box up to close the gap a bit. * [Core] Exclude the notification popup colour picker button from themes Noticed it was not the colour it should be and thought the setting was always being reset -- turns out the setting was correct, the button colour was not. Here's a quick hack to fix. * [Core] Further notification title tweaking Now "System Discovery Complete" *actually* fits. * [Core] Widen the native pop-up font selector so you can see variants The previous width was a little bit narrow for the family of fonts like Segoe UI Variable... The new width is 2x the original size and fits all fonts on my system and should do the trick for the moment.
164 lines
4.9 KiB
C#
164 lines
4.9 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using System.Windows.Forms;
|
|
|
|
namespace Observatory.UI
|
|
{
|
|
internal class ThemeManager
|
|
{
|
|
public static ThemeManager GetInstance
|
|
{
|
|
get
|
|
{
|
|
return _instance.Value;
|
|
}
|
|
}
|
|
private static HashSet<string> _excludedControlNames = new()
|
|
{
|
|
"ColourButton",
|
|
};
|
|
private static readonly Lazy<ThemeManager> _instance = new(() => new ThemeManager());
|
|
private bool _init;
|
|
|
|
private ThemeManager()
|
|
{
|
|
_init = true;
|
|
controls = new List<Control>();
|
|
Themes = new()
|
|
{
|
|
{ "Dark", DarkTheme },
|
|
{ "Light", LightTheme }
|
|
};
|
|
SelectedTheme = "Dark";
|
|
}
|
|
|
|
private readonly List<Control> controls;
|
|
|
|
public List<string> GetThemes
|
|
{
|
|
get => Themes.Keys.ToList();
|
|
}
|
|
|
|
public string CurrentTheme
|
|
{
|
|
get => SelectedTheme;
|
|
|
|
set
|
|
{
|
|
if (Themes.ContainsKey(value))
|
|
{
|
|
SelectedTheme = value;
|
|
foreach (var control in controls)
|
|
{
|
|
ApplyTheme(control);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public void RegisterControl(Control control)
|
|
{
|
|
// First time registering a control, build the "light" theme based
|
|
// on defaults.
|
|
if (_init)
|
|
{
|
|
SaveTheme(control, LightTheme);
|
|
_init = false;
|
|
}
|
|
|
|
controls.Add(control);
|
|
ApplyTheme(control);
|
|
if (control.HasChildren)
|
|
foreach (Control child in control.Controls)
|
|
{
|
|
if (_excludedControlNames.Contains(child.Name)) continue;
|
|
RegisterControl(child);
|
|
}
|
|
}
|
|
|
|
// This doesn't inherit from Control? Seriously?
|
|
public void RegisterControl(ToolStripMenuItem toolStripMenuItem)
|
|
{
|
|
ApplyTheme(toolStripMenuItem);
|
|
}
|
|
|
|
private void SaveTheme(Control control, Dictionary<string, Color> theme)
|
|
{
|
|
Control rootControl = control;
|
|
while (rootControl.Parent != null)
|
|
{
|
|
rootControl = rootControl.Parent;
|
|
}
|
|
|
|
SaveThemeControl(rootControl, theme);
|
|
var themeJson = System.Text.Json.JsonSerializer.Serialize(DarkTheme);
|
|
}
|
|
|
|
private void SaveThemeControl(Control control, Dictionary<string, Color> theme)
|
|
{
|
|
var properties = control.GetType().GetProperties();
|
|
var colorProperties = properties.Where(p => p.PropertyType == typeof(Color));
|
|
|
|
foreach (var colorProperty in colorProperties)
|
|
{
|
|
string controlKey = control.GetType().Name + "." + colorProperty.Name;
|
|
if (!theme.ContainsKey(controlKey))
|
|
{
|
|
theme.Add(controlKey, (Color)colorProperty.GetValue(control)!);
|
|
}
|
|
}
|
|
|
|
foreach (Control child in control.Controls)
|
|
{
|
|
SaveThemeControl(child, theme);
|
|
}
|
|
}
|
|
|
|
public void DeRegisterControl(Control control)
|
|
{
|
|
if (control.HasChildren)
|
|
foreach (Control child in control.Controls)
|
|
{
|
|
DeRegisterControl(child);
|
|
}
|
|
controls.Remove(control);
|
|
}
|
|
|
|
private void ApplyTheme(Object control)
|
|
{
|
|
var controlType = control.GetType();
|
|
|
|
var theme = Themes.ContainsKey(SelectedTheme)
|
|
? Themes[SelectedTheme] : Themes["Light"];
|
|
|
|
foreach (var property in controlType.GetProperties().Where(p => p.PropertyType == typeof(Color)))
|
|
{
|
|
string themeControl = Themes[SelectedTheme].ContainsKey(controlType.Name + "." + property.Name)
|
|
? controlType.Name
|
|
: "Default";
|
|
|
|
if (Themes[SelectedTheme].ContainsKey(themeControl + "." + property.Name))
|
|
property.SetValue(control, Themes[SelectedTheme][themeControl + "." + property.Name]);
|
|
}
|
|
|
|
}
|
|
|
|
private Dictionary<string, Dictionary<string, Color>> Themes;
|
|
|
|
private string SelectedTheme;
|
|
|
|
private Dictionary<string, Color> LightTheme = new Dictionary<string, Color>();
|
|
|
|
static private Dictionary<string, Color> DarkTheme = new Dictionary<string, Color>
|
|
{
|
|
{"Default.ForeColor", Color.LightGray },
|
|
{"Default.BackColor", Color.Black },
|
|
{"Button.ForeColor", Color.LightGray },
|
|
{"Button.BackColor", Color.DimGray }
|
|
};
|
|
}
|
|
}
|