امروز میخواستم برای یکی از پروژههایم، قابلیتی را پیاده سازی کنم که هماهنگ با تم ویندوز، تم برنامه را عوض کند (تیره/روشن). به این منظور که وقتی تم ویندوز Dark میشد، تم برنامهی من هم Dark بشود و برعکس. سادهترین کار این بود که از کدهای WinRT که توسط بستهی نیوگت SDK Contract ارائه میشود استفاده کرد. در این صورت کافیست فقط از کلاس ThemeManager استفاده کنیم و بدون کوچکترین خونریزی، برنامه را به این ویژگی مجهز کنیم😁 اما خب، هرچیزی هزینهی خودش را دارد و من به شخصه علاقهای به استفاده از 25 مگابایت، فقط برای شناسایی وضعیت تم ویندوز را ندارم! پس خودم دست به کار شدم تا یک Listener برای این منظور بنویسم.
در دات نت یکسری رخداد وجود دارند که مربوط به سیستم عامل میشوند و از کلاس SystemEvents قابل دسترسی هستند. در اینجا ایونتی داریم به اسم UserPreferenceChanged که شامل مواردی میشود که کاربر، تنظیمات ویندوز را تغییر میدهد. هر تغییری که در تنظیمات ویندوز اعمال بشود، درون یکی از Categoryها صدا زده میشود. پس اگر ما این ایونت را رجیستر کنیم، هر موقع تغییری در تنظیمات ویندوز اعمال بشود، برنامهی ما نیز متوجه میشود.
برای این منظور یک کلاس را ایجاد میکنیم و در متد سازندهی آن، این رخداد را ثبت میکنیم:
همینطور که میبینید، این دسته بندی شامل موارد مختلفی میشود که به بخشهای مختلف تنظیمات ویندوز مربوط است. تنظیمات مربوط به تم، درون General صدا زده میشود. پس کدهای ما قرار است وارد این قسمت بشود. متاسفانه این رخداد اطلاعات کاملتری را به ما نمیدهد و فقط اطلاع میدهد که تغییری رخ داده (همینطور که قبلا گفتم، هرچیزی هزینهای دارد). پس ما باید مشخصات تم فعلی را دریافت کنیم؛ اما از کجا؟ معلوم است رجیستری! همه چیز در رجیستری ثبت میشود و بهراحتی قابل دسترسی هست. پس یک متد مینویسیم که کلید رجیستری مربوطه را بخواند و آن را بصورت یک مدل (Light یا Dark) برگرداند:
حالا ما نیاز به یک رخداد داریم که کاربر بتواند در برنامهی خودش آنرا ثبت کند و نیازی به پیاده سازی این کدها نداشته باشد. پس یک EventHandler را به اسم WindowsThemeChanged ایجاد میکنیم:
من میخواهم که کاربر، مقدار تم فعلی و رنگ Accent فعلی را نیز بتواند از طریق این رخداد، دریافت کند. پس ما باید یک EventArgs را ایجاد کنیم که پراپرتیهای دلخواهی را داشته باشد. برای همین کلاس FunctionEventArgs را ایجاد میکنیم:
این کلاس از RoutedEventArgs ارث بری کرده و بصورت جنریک پیاده سازی شدهاست. به این معنا که ما میتوانیم هر نوع دلخواهی را که خواستیم، به عنوان arg استفاده کنیم. اگر دقت کنید من یک مدل دلخواه را به عنوان arg مشخص کردهام:
میتوانستیم از همان UITheme هم استفاده کنیم؛ ولی نمیتوانستیم مقدار Accent را به کاربر برگردانیم. برای همین، مدلی را به اسم UIWindowTheme ایجاد میکنیم:
کار تمام است. حالا باید کدهای داخل متد SystemEvents_UserPreferenceChanged را بنویسیم (دقت کنید که کد، باید داخل بخش General نوشته شود):
یک مدل را ایجاد کرده و مقدار AccentBrush را برابر با WindowGlassBrush قرار میدهیم. این پراپرتی هم مانند ایونتی که اول معرفی کردیم، مربوط به سیستم عامل بوده و رنگ فعلی Accent را بر میگرداند. برای مقدار CurrentTheme نیز متدی را که بالاتر برای دریافت تم فعلی از رجیستری نوشتیم، صدا میزنیم و در پایان این مدل را به ایونت، پاس میدهیم. در پایان میتوانیم به این صورت ایونت خود را پیاده سازی کنیم:
در دات نت یکسری رخداد وجود دارند که مربوط به سیستم عامل میشوند و از کلاس SystemEvents قابل دسترسی هستند. در اینجا ایونتی داریم به اسم UserPreferenceChanged که شامل مواردی میشود که کاربر، تنظیمات ویندوز را تغییر میدهد. هر تغییری که در تنظیمات ویندوز اعمال بشود، درون یکی از Categoryها صدا زده میشود. پس اگر ما این ایونت را رجیستر کنیم، هر موقع تغییری در تنظیمات ویندوز اعمال بشود، برنامهی ما نیز متوجه میشود.
برای این منظور یک کلاس را ایجاد میکنیم و در متد سازندهی آن، این رخداد را ثبت میکنیم:
pubic class ThemeHelper { public ThemeHelper() { SystemEvents.UserPreferenceChanged += SystemEvents_UserPreferenceChanged; } private void SystemEvents_UserPreferenceChanged(object sender, UserPreferenceChangedEventArgs e) { switch (e.Category) { case UserPreferenceCategory.Accessibility: break; case UserPreferenceCategory.Color: break; case UserPreferenceCategory.Desktop: break; case UserPreferenceCategory.General: break; case UserPreferenceCategory.Icon: break; case UserPreferenceCategory.Keyboard: break; case UserPreferenceCategory.Menu: break; case UserPreferenceCategory.Mouse: break; case UserPreferenceCategory.Policy: break; case UserPreferenceCategory.Power: break; case UserPreferenceCategory.Screensaver: break; case UserPreferenceCategory.Window: break; case UserPreferenceCategory.Locale: break; case UserPreferenceCategory.VisualStyle: break; } } }
private const string RegistryKeyPathTheme = @"Software\Microsoft\Windows\CurrentVersion\Themes\Personalize"; private const string RegSysMode = "SystemUsesLightTheme"; public static UITheme GetWindowsTheme() { return GetThemeFromRegistry(RegSysMode); } private static UITheme GetThemeFromRegistry(string registryKey) { using var key = Registry.CurrentUser.OpenSubKey(RegistryKeyPathTheme); var themeValue = key?.GetValue(registryKey) as int?; return themeValue != 0 ? UITheme.Light : UITheme.Dark; } public enum UITheme { Light, Dark }
public event EventHandler<FunctionEventArgs<UIWindowTheme>> WindowsThemeChanged; protected virtual void OnWindowsThemeChanged(UIWindowTheme theme) { EventHandler<FunctionEventArgs<UIWindowTheme>> handler = WindowsThemeChanged; handler?.Invoke(this, new FunctionEventArgs<UIWindowTheme>(theme)); }
public class FunctionEventArgs<T> : RoutedEventArgs { public FunctionEventArgs(T theme) { Theme = theme; } public FunctionEventArgs(RoutedEvent routedEvent, object source) : base(routedEvent, source) { } public T Theme { get; set; } }
FunctionEventArgs<UIWindowTheme>
public class UIWindowTheme { public Brush AccentBrush { get; set; } public UITheme CurrentTheme { get; set; } }
case UserPreferenceCategory.General: var changedTheme = new UIWindowTheme() { AccentBrush = SystemParameters.WindowGlassBrush, CurrentTheme = GetWindowsTheme() }; OnWindowsThemeChanged(changedTheme); break;
ThemeHelper tm = new ThemeHelper(); tm.WindowsThemeChanged +=OnWindowsThemeChanged; private void OnWindowsThemeChanged(object? sender, FunctionEventArgs<RegistryThemeHelper.UIWindowTheme> e) { rec.Fill = e.Theme.AccentBrush; if (e.Theme.CurrentTheme == ThemeHelper.UITheme.Light) { Background = Brushes.White; } else { Background = Brushes.Black; } }