diff --git a/Source/Installer/Product.wxs b/Source/Installer/Product.wxs index 1d8f0d86..c7507bdb 100644 --- a/Source/Installer/Product.wxs +++ b/Source/Installer/Product.wxs @@ -1,6 +1,6 @@  - GetTask(string name, Action action) => return $@"{name,-14} -> {elapsed.ToString($@"{(elapsed.Minutes > 0 ? @"m\:" : string.Empty)}s\.fff")}"; }); } - - private string GetSystem() - { - using var @class = new ManagementClass("Win32_ComputerSystem"); - using var instance = @class.GetInstances().Cast().FirstOrDefault(); - return $"Manufacturer: {instance?["Manufacturer"]}, Model: {instance?["Model"]}, OS: {Environment.OSVersion.Version}"; - } } #endregion diff --git a/Source/Monitorian.Core/Models/SystemInfo.cs b/Source/Monitorian.Core/Models/SystemInfo.cs new file mode 100644 index 00000000..0d122282 --- /dev/null +++ b/Source/Monitorian.Core/Models/SystemInfo.cs @@ -0,0 +1,52 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Linq; +using System.Management; +using System.Text; +using System.Threading.Tasks; +using Microsoft.Win32; + +namespace Monitorian.Core.Models +{ + /// + /// System information + /// + public static class SystemInfo + { + private class SystemInfoInternal + { + public string Manufacturer { get; } + public string Model { get; } + + public SystemInfoInternal() + { + try + { + using var @class = new ManagementClass("Win32_ComputerSystem"); + using var instance = @class.GetInstances().Cast().FirstOrDefault(); + + Manufacturer = instance?["Manufacturer"] as string; + Model = instance?["Model"] as string; + return; + } + catch (ManagementException) + { + Debug.WriteLine($"Failed to get instances by Win32_ComputerSystem"); + } + + const string keyName = @"SYSTEM\CurrentControlSet\Control\SystemInformation"; // HKLM + + using var key = Registry.LocalMachine.OpenSubKey(keyName); + + Manufacturer = key?.GetValue("SystemManufacturer") as string; + Model = key?.GetValue("SystemProductName") as string; + } + } + + private static readonly Lazy _instance = new(() => new()); + + public static string Manufacturer => _instance.Value.Manufacturer; + public static string Model => _instance.Value.Model; + } +} \ No newline at end of file diff --git a/Source/Monitorian.Core/Monitorian.Core.csproj b/Source/Monitorian.Core/Monitorian.Core.csproj index 6f777e6a..3628a0cd 100644 --- a/Source/Monitorian.Core/Monitorian.Core.csproj +++ b/Source/Monitorian.Core/Monitorian.Core.csproj @@ -93,6 +93,7 @@ + True diff --git a/Source/Monitorian.Core/Properties/AssemblyInfo.cs b/Source/Monitorian.Core/Properties/AssemblyInfo.cs index c1e4f7ab..d0a916ea 100644 --- a/Source/Monitorian.Core/Properties/AssemblyInfo.cs +++ b/Source/Monitorian.Core/Properties/AssemblyInfo.cs @@ -33,8 +33,8 @@ // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: // [assembly: AssemblyVersion("1.0.*")] -[assembly: AssemblyVersion("3.13.0.0")] -[assembly: AssemblyFileVersion("3.13.0.0")] +[assembly: AssemblyVersion("3.14.0.0")] +[assembly: AssemblyFileVersion("3.14.0.0")] [assembly: NeutralResourcesLanguage("en-US")] // For unit test diff --git a/Source/Monitorian.Core/ViewModels/MainWindowViewModel.cs b/Source/Monitorian.Core/ViewModels/MainWindowViewModel.cs index d63474b1..9ed06440 100644 --- a/Source/Monitorian.Core/ViewModels/MainWindowViewModel.cs +++ b/Source/Monitorian.Core/ViewModels/MainWindowViewModel.cs @@ -32,7 +32,9 @@ public ListCollectionView MonitorsView _monitorsView = new ListCollectionView(_controller.Monitors); if (Settings.OrdersArrangement) { - _monitorsView.SortDescriptions.Add(new SortDescription(nameof(MonitorViewModel.MonitorTop), ListSortDirection.Ascending)); + _monitorsView.SortDescriptions.Add(new SortDescription(nameof(MonitorViewModel.MonitorTopLeft), ListSortDirection.Ascending)); + _monitorsView.IsLiveSorting = true; + _monitorsView.LiveSortingProperties.Add(nameof(MonitorViewModel.MonitorTopLeft)); } _monitorsView.SortDescriptions.Add(new SortDescription(nameof(MonitorViewModel.DisplayIndex), ListSortDirection.Ascending)); _monitorsView.SortDescriptions.Add(new SortDescription(nameof(MonitorViewModel.MonitorIndex), ListSortDirection.Ascending)); @@ -88,21 +90,22 @@ private void OnSettingsChanged(object sender, PropertyChangedEventArgs e) switch (e.PropertyName) { case nameof(Settings.OrdersArrangement): - var orderDescription = MonitorsView.SortDescriptions.FirstOrDefault(x => x.PropertyName == nameof(MonitorViewModel.MonitorTop)); + var description = new SortDescription(nameof(MonitorViewModel.MonitorTopLeft), ListSortDirection.Ascending); + int index = MonitorsView.SortDescriptions.IndexOf(description); - if (Settings.OrdersArrangement) + switch (Settings.OrdersArrangement, index) { - if (orderDescription != default) - return; - - MonitorsView.SortDescriptions.Insert(0, new SortDescription(nameof(MonitorViewModel.MonitorTop), ListSortDirection.Ascending)); - } - else - { - if (orderDescription == default) - return; - - MonitorsView.SortDescriptions.Remove(orderDescription); + case (true, < 0): + MonitorsView.SortDescriptions.Insert(0, description); + MonitorsView.IsLiveSorting = true; + MonitorsView.LiveSortingProperties.Add(description.PropertyName); + break; + + case (false, >= 0): + MonitorsView.SortDescriptions.RemoveAt(index); + MonitorsView.IsLiveSorting = false; + MonitorsView.LiveSortingProperties.Remove(description.PropertyName); + break; } MonitorsView.Refresh(); diff --git a/Source/Monitorian.Core/ViewModels/MonitorViewModel.cs b/Source/Monitorian.Core/ViewModels/MonitorViewModel.cs index e3817923..c0308a82 100644 --- a/Source/Monitorian.Core/ViewModels/MonitorViewModel.cs +++ b/Source/Monitorian.Core/ViewModels/MonitorViewModel.cs @@ -23,6 +23,7 @@ public MonitorViewModel(AppControllerCore controller, IMonitor monitor) { this._controller = controller ?? throw new ArgumentNullException(nameof(controller)); this._monitor = monitor ?? throw new ArgumentNullException(nameof(monitor)); + SetTopLeft(); LoadCustomization(); } @@ -37,6 +38,7 @@ internal void Replace(IMonitor monitor) { this._monitor.Dispose(); this._monitor = monitor; + SetTopLeft(); } } else @@ -50,7 +52,6 @@ internal void Replace(IMonitor monitor) public byte DisplayIndex => _monitor.DisplayIndex; public byte MonitorIndex => _monitor.MonitorIndex; public Rect MonitorRect => _monitor.MonitorRect; - public double MonitorTop => _monitor.MonitorRect.Top; #region Customization @@ -463,6 +464,29 @@ public bool IsSelected #endregion + #region Arrangement + + public ulong MonitorTopLeft + { + get => _monitorTopLeft; + private set => SetProperty(ref _monitorTopLeft, value); + } + private ulong _monitorTopLeft; + + private void SetTopLeft() + { + MonitorTopLeft = GetTopLeft(_monitor.MonitorRect.Location); + + static ulong GetTopLeft(Point location) + { + var x = (long)Math.Round(location.X, MidpointRounding.AwayFromZero) + int.MaxValue; + var y = (long)Math.Round(location.Y, MidpointRounding.AwayFromZero) + int.MaxValue; + return (ulong)x | ((ulong)y << 32); + } + } + + #endregion + public bool IsTarget { get => _isTarget; diff --git a/Source/Monitorian/Properties/AssemblyInfo.cs b/Source/Monitorian/Properties/AssemblyInfo.cs index 021f44f2..17d3164d 100644 --- a/Source/Monitorian/Properties/AssemblyInfo.cs +++ b/Source/Monitorian/Properties/AssemblyInfo.cs @@ -51,7 +51,7 @@ // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: // [assembly: AssemblyVersion("1.0.*")] -[assembly: AssemblyVersion("3.13.0.0")] -[assembly: AssemblyFileVersion("3.13.0.0")] +[assembly: AssemblyVersion("3.14.0.0")] +[assembly: AssemblyFileVersion("3.14.0.0")] [assembly: Guid("a4cc5362-9b08-465b-ad64-5cfabc72a4c7")] [assembly: NeutralResourcesLanguage("en-US")]