From 3c3aca8bfdba5c15a81058fc7e711a79aced1ebb Mon Sep 17 00:00:00 2001 From: Fred Kuipers Date: Sat, 14 Aug 2021 00:14:13 -0400 Subject: [PATCH 1/3] Enclose pre-read with Read-all; de-dupe plugin table and show types, etc A few changes in preparation for the upcoming Announcer plugin. 1) De-dupe entries in the plugin table in the core settings UI when a plugin implements both Worker and Notifier interfaces. 2) Also add a "Types" column to the plugin table indicating the set of implemented interfaces found. 3) Wrap the pre-reading logic in "Read-all" scope to properly suppress a stampede of notifications (particularly audio ones) when starting the log monitor. 4) Fix the int setting UI. It rendered incorrectly, it didn't get the value from the settings class, nor wrote it back. Also implemented optional value bounding using the new setting attribute (see the related Framework PR). 5) Applied similar fixes to string setting (addressed rendering, write-back although untested). 6) Minor visual fixes (height/vertical alignment) for non-boolean settings to address cut-off/overlapping inputs. Sorry for the larger PR, but these *should* be all the loose ends before we could consider including Basic Announcer... --- ObservatoryCore/LogMonitor.cs | 2 + .../UI/ViewModels/CoreViewModel.cs | 30 +++++-- ObservatoryCore/UI/Views/BasicUIView.axaml.cs | 82 +++++++++++++++---- 3 files changed, 90 insertions(+), 24 deletions(-) diff --git a/ObservatoryCore/LogMonitor.cs b/ObservatoryCore/LogMonitor.cs index fd57421..7282314 100644 --- a/ObservatoryCore/LogMonitor.cs +++ b/ObservatoryCore/LogMonitor.cs @@ -146,7 +146,9 @@ namespace Observatory // We found an FSD jump, buffered the lines for that system (possibly including startup logs // over a file boundary). Pump these through the plugins. + readall = true; ReportErrors(ProcessLines(lastSystemLines, "Pre-read")); + readall = false; } #endregion diff --git a/ObservatoryCore/UI/ViewModels/CoreViewModel.cs b/ObservatoryCore/UI/ViewModels/CoreViewModel.cs index 3b8f166..62458e0 100644 --- a/ObservatoryCore/UI/ViewModels/CoreViewModel.cs +++ b/ObservatoryCore/UI/ViewModels/CoreViewModel.cs @@ -55,21 +55,15 @@ namespace Observatory.UI.ViewModels public void ReadAll() { - foreach (var worker in workers) - { - worker.ReadAllStarted(); - } + SetWorkerReadAllState(true); LogMonitor.GetInstance.ReadAllJournals(); - foreach (var worker in workers) - { - worker.ReadAllFinished(); - } + SetWorkerReadAllState(false); } public void ToggleMonitor() { var logMonitor = LogMonitor.GetInstance; - + if (logMonitor.IsMonitoring()) { logMonitor.Stop(); @@ -77,7 +71,10 @@ namespace Observatory.UI.ViewModels } else { + // HACK: Find a better way of suppressing notifications when pre-reading. + SetWorkerReadAllState(true); logMonitor.Start(); + SetWorkerReadAllState(false); ToggleButtonText = "Stop Monitor"; } } @@ -123,5 +120,20 @@ namespace Observatory.UI.ViewModels { get { return tabs; } } + private void SetWorkerReadAllState(bool isReadingAll) + { + foreach (var worker in workers) + { + if (isReadingAll) + { + worker.ReadAllStarted(); + } + else + { + worker.ReadAllFinished(); + } + } + } + } } diff --git a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs index 7d407b0..05378c8 100644 --- a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs +++ b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs @@ -10,6 +10,7 @@ using Avalonia.Interactivity; using Avalonia.VisualTree; using System; using System.Collections.ObjectModel; +using System.Collections.Generic; namespace Observatory.UI.Views { @@ -252,6 +253,12 @@ namespace Observatory.UI.Views Binding = new Binding("Name") }); + pluginList.Columns.Add(new DataGridTextColumn() + { + Header = "Types", + Binding = new Binding("TypesString") + }); + pluginList.Columns.Add(new DataGridTextColumn() { Header = "Version", @@ -264,19 +271,29 @@ namespace Observatory.UI.Views Binding = new Binding("Status") }); - System.Collections.Generic.List allPlugins = new(); - + Dictionary uniquePlugins = new(); foreach(var (plugin, signed) in pluginManager.workerPlugins) { - allPlugins.Add(new PluginView() { Name = plugin.Name, Version = plugin.Version, Status = GetStatusText(signed) }); + if (!uniquePlugins.ContainsKey(plugin.Name)) + { + uniquePlugins.Add(plugin.Name, + new PluginView() { Name = plugin.Name, Types = new() { PluginType.Worker }, Version = plugin.Version, Status = GetStatusText(signed) }); + } } foreach (var (plugin, signed) in pluginManager.notifyPlugins) { - allPlugins.Add(new PluginView() { Name = plugin.Name, Version = plugin.Version, Status = GetStatusText(signed) }); + if (!uniquePlugins.ContainsKey(plugin.Name)) + { + uniquePlugins.Add(plugin.Name, + new PluginView() { Name = plugin.Name, Types = new() { PluginType.Notifier }, Version = plugin.Version, Status = GetStatusText(signed) }); + } else + { + uniquePlugins[plugin.Name].Types.Add(PluginType.Notifier); + } } - pluginList.Items = allPlugins; + pluginList.Items = uniquePlugins.Values; corePanel.AddControl(pluginList, SettingRowTracker.PLUGIN_LIST_ROW_INDEX, 0, 2); #endregion @@ -324,10 +341,13 @@ namespace Observatory.UI.Views { if (setting.Key.PropertyType != typeof(bool) || settingsGrid.Children.Count % 2 == 0) { - settingsGrid.RowDefinitions.Add(new RowDefinition() { Height = new GridLength(21) }); + settingsGrid.RowDefinitions.Add(new RowDefinition() + { + Height = new GridLength(setting.Key.PropertyType != typeof(bool) ? 32 : 25), + }); } - TextBlock label = new() { Text = setting.Value }; + TextBlock label = new() { Text = setting.Value, VerticalAlignment = Avalonia.Layout.VerticalAlignment.Center }; switch (setting.Key.GetValue(plugin.Settings)) { @@ -351,13 +371,30 @@ namespace Observatory.UI.Views break; case string stringSetting: TextBox textBox = new() { Text = stringSetting }; - settingsGrid.Children.Add(label); - settingsGrid.Children.Add(textBox); + settingsGrid.AddControl(label, settingsGrid.RowDefinitions.Count - 1, 0); + settingsGrid.AddControl(textBox, settingsGrid.RowDefinitions.Count - 1, 1); + textBox.TextInput += (object sender, Avalonia.Input.TextInputEventArgs e) => + { + setting.Key.SetValue(plugin.Settings, e.Text); + PluginManagement.PluginManager.GetInstance.SaveSettings(plugin, plugin.Settings); + }; break; case int intSetting: - NumericUpDown numericUpDown = new() { Text = intSetting.ToString(), AllowSpin = true }; - settingsGrid.Children.Add(label); - settingsGrid.Children.Add(numericUpDown); + NumericUpDown numericUpDown = new() { Value = intSetting, AllowSpin = true }; + SettingNumericBounds attr = (SettingNumericBounds)System.Attribute.GetCustomAttribute(setting.Key, typeof(SettingNumericBounds)); + if (attr != null) + { + numericUpDown.Minimum = attr.Minimum; + numericUpDown.Maximum = attr.Maximum; + numericUpDown.Increment = attr.Increment; + } + settingsGrid.AddControl(label, settingsGrid.RowDefinitions.Count - 1, 0); + settingsGrid.AddControl(numericUpDown, settingsGrid.RowDefinitions.Count - 1, 1); + numericUpDown.ValueChanged += (object sender, NumericUpDownValueChangedEventArgs e) => + { + setting.Key.SetValue(plugin.Settings, Convert.ToInt32(e.NewValue)); + PluginManagement.PluginManager.GetInstance.SaveSettings(plugin, plugin.Settings); + }; break; case System.IO.FileInfo fileSetting: label.Text += ": "; @@ -404,7 +441,7 @@ namespace Observatory.UI.Views stackPanel.Children.Add(label); stackPanel.Children.Add(settingPath); - settingsGrid.RowDefinitions.Add(new RowDefinition() { Height = new GridLength(21) }); + settingsGrid.RowDefinitions.Add(new RowDefinition() { Height = new GridLength(32) }); settingsGrid.AddControl(stackPanel, settingsGrid.RowDefinitions.Count - 1, 0, 2); settingsGrid.AddControl(settingBrowse, settingsGrid.RowDefinitions.Count - 1, 2); @@ -415,6 +452,11 @@ namespace Observatory.UI.Views } } + private void TextBox_TextInput(object sender, Avalonia.Input.TextInputEventArgs e) + { + throw new NotImplementedException(); + } + private string GetStatusText(PluginManagement.PluginManager.PluginStatus status) { string statusText; @@ -462,12 +504,22 @@ namespace Observatory.UI.Views internal class PluginView { public string Name { get; set; } + public HashSet Types { get; set; } + public string TypesString { + get + { + return string.Join(", ", Types); + } + set { } } public string Version { get; set; } public string Status { get; set; } } - - + enum PluginType + { + Worker, + Notifier, + } internal static class GridExtention { From 73c9b94b0f2d4ab0fc6c17eedf1a1a2ad2f56668 Mon Sep 17 00:00:00 2001 From: Fred Kuipers Date: Sat, 14 Aug 2021 00:19:02 -0400 Subject: [PATCH 2/3] Remove auto-generated method --- ObservatoryCore/UI/Views/BasicUIView.axaml.cs | 5 ----- 1 file changed, 5 deletions(-) diff --git a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs index 05378c8..739d2d7 100644 --- a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs +++ b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs @@ -452,11 +452,6 @@ namespace Observatory.UI.Views } } - private void TextBox_TextInput(object sender, Avalonia.Input.TextInputEventArgs e) - { - throw new NotImplementedException(); - } - private string GetStatusText(PluginManagement.PluginManager.PluginStatus status) { string statusText; From e523dddfe3f23515c8f5efbbac6c56018f87261a Mon Sep 17 00:00:00 2001 From: Fred Kuipers Date: Sun, 15 Aug 2021 17:02:05 -0400 Subject: [PATCH 3/3] Revert LogMonitor/CoreViewModel changes, implement Slider int setting Also revert the PluginType enum. Fix more whitespace --- ObservatoryCore/LogMonitor.cs | 2 - .../UI/ViewModels/CoreViewModel.cs | 30 ++----- ObservatoryCore/UI/Views/BasicUIView.axaml.cs | 87 ++++++++++++------- 3 files changed, 66 insertions(+), 53 deletions(-) diff --git a/ObservatoryCore/LogMonitor.cs b/ObservatoryCore/LogMonitor.cs index 7282314..fd57421 100644 --- a/ObservatoryCore/LogMonitor.cs +++ b/ObservatoryCore/LogMonitor.cs @@ -146,9 +146,7 @@ namespace Observatory // We found an FSD jump, buffered the lines for that system (possibly including startup logs // over a file boundary). Pump these through the plugins. - readall = true; ReportErrors(ProcessLines(lastSystemLines, "Pre-read")); - readall = false; } #endregion diff --git a/ObservatoryCore/UI/ViewModels/CoreViewModel.cs b/ObservatoryCore/UI/ViewModels/CoreViewModel.cs index 62458e0..3b8f166 100644 --- a/ObservatoryCore/UI/ViewModels/CoreViewModel.cs +++ b/ObservatoryCore/UI/ViewModels/CoreViewModel.cs @@ -55,15 +55,21 @@ namespace Observatory.UI.ViewModels public void ReadAll() { - SetWorkerReadAllState(true); + foreach (var worker in workers) + { + worker.ReadAllStarted(); + } LogMonitor.GetInstance.ReadAllJournals(); - SetWorkerReadAllState(false); + foreach (var worker in workers) + { + worker.ReadAllFinished(); + } } public void ToggleMonitor() { var logMonitor = LogMonitor.GetInstance; - + if (logMonitor.IsMonitoring()) { logMonitor.Stop(); @@ -71,10 +77,7 @@ namespace Observatory.UI.ViewModels } else { - // HACK: Find a better way of suppressing notifications when pre-reading. - SetWorkerReadAllState(true); logMonitor.Start(); - SetWorkerReadAllState(false); ToggleButtonText = "Stop Monitor"; } } @@ -120,20 +123,5 @@ namespace Observatory.UI.ViewModels { get { return tabs; } } - private void SetWorkerReadAllState(bool isReadingAll) - { - foreach (var worker in workers) - { - if (isReadingAll) - { - worker.ReadAllStarted(); - } - else - { - worker.ReadAllFinished(); - } - } - } - } } diff --git a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs index 739d2d7..def362b 100644 --- a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs +++ b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs @@ -277,7 +277,7 @@ namespace Observatory.UI.Views if (!uniquePlugins.ContainsKey(plugin.Name)) { uniquePlugins.Add(plugin.Name, - new PluginView() { Name = plugin.Name, Types = new() { PluginType.Worker }, Version = plugin.Version, Status = GetStatusText(signed) }); + new PluginView() { Name = plugin.Name, Types = new() { typeof(IObservatoryWorker).Name }, Version = plugin.Version, Status = GetStatusText(signed) }); } } @@ -286,10 +286,10 @@ namespace Observatory.UI.Views if (!uniquePlugins.ContainsKey(plugin.Name)) { uniquePlugins.Add(plugin.Name, - new PluginView() { Name = plugin.Name, Types = new() { PluginType.Notifier }, Version = plugin.Version, Status = GetStatusText(signed) }); + new PluginView() { Name = plugin.Name, Types = new() { typeof(IObservatoryNotifier).Name }, Version = plugin.Version, Status = GetStatusText(signed) }); } else { - uniquePlugins[plugin.Name].Types.Add(PluginType.Notifier); + uniquePlugins[plugin.Name].Types.Add(typeof(IObservatoryNotifier).Name); } } @@ -343,7 +343,7 @@ namespace Observatory.UI.Views { settingsGrid.RowDefinitions.Add(new RowDefinition() { - Height = new GridLength(setting.Key.PropertyType != typeof(bool) ? 32 : 25), + Height = new GridLength(setting.Key.PropertyType != typeof(bool) ? 40 : 25), }); } @@ -380,21 +380,55 @@ namespace Observatory.UI.Views }; break; case int intSetting: - NumericUpDown numericUpDown = new() { Value = intSetting, AllowSpin = true }; - SettingNumericBounds attr = (SettingNumericBounds)System.Attribute.GetCustomAttribute(setting.Key, typeof(SettingNumericBounds)); - if (attr != null) + // We have two options for integer values: + // 1) A slider (explicit by way of the SettingIntegerUseSlider attribute and bounded to 0..100 by default) + // 2) A numeric up/down (default otherwise, and is unbounded by default). + // Bounds for both can be set via the SettingNumericBounds attribute, only the up/down uses Increment. + Control intControl; + SettingNumericBounds bounds = (SettingNumericBounds)System.Attribute.GetCustomAttribute(setting.Key, typeof(SettingNumericBounds)); + if (System.Attribute.IsDefined(setting.Key, typeof(SettingNumericUseSlider))) { - numericUpDown.Minimum = attr.Minimum; - numericUpDown.Maximum = attr.Maximum; - numericUpDown.Increment = attr.Increment; + // TODO: Suss the contents of this block into a function to share with non-integral numeric types as well? + Slider slider = new() + { + Value = intSetting, + Height = 40, + Width = 300, + }; + if (bounds != null) + { + slider.Minimum = bounds.Minimum; + slider.Maximum = bounds.Maximum; + }; + slider.PropertyChanged += (object sender, AvaloniaPropertyChangedEventArgs e) => + { + if (e.Property == Slider.ValueProperty) + { + setting.Key.SetValue(plugin.Settings, Convert.ToInt32(e.NewValue)); + PluginManagement.PluginManager.GetInstance.SaveSettings(plugin, plugin.Settings); + } + }; + intControl = slider; } - settingsGrid.AddControl(label, settingsGrid.RowDefinitions.Count - 1, 0); - settingsGrid.AddControl(numericUpDown, settingsGrid.RowDefinitions.Count - 1, 1); - numericUpDown.ValueChanged += (object sender, NumericUpDownValueChangedEventArgs e) => + else // Use a Numeric Up/Down { - setting.Key.SetValue(plugin.Settings, Convert.ToInt32(e.NewValue)); - PluginManagement.PluginManager.GetInstance.SaveSettings(plugin, plugin.Settings); - }; + NumericUpDown numericUpDown = new() { Value = intSetting, AllowSpin = true }; + if (bounds != null) + { + numericUpDown.Minimum = bounds.Minimum; + numericUpDown.Maximum = bounds.Maximum; + numericUpDown.Increment = bounds.Increment; + } + numericUpDown.ValueChanged += (object sender, NumericUpDownValueChangedEventArgs e) => + { + setting.Key.SetValue(plugin.Settings, Convert.ToInt32(e.NewValue)); + PluginManagement.PluginManager.GetInstance.SaveSettings(plugin, plugin.Settings); + }; + intControl = numericUpDown; + } + + settingsGrid.AddControl(label, settingsGrid.RowDefinitions.Count - 1, 0); + settingsGrid.AddControl(intControl, settingsGrid.RowDefinitions.Count - 1, 1); break; case System.IO.FileInfo fileSetting: label.Text += ": "; @@ -402,7 +436,7 @@ namespace Observatory.UI.Views TextBox settingPath = new() { Text = fileSetting.FullName, - Width = 250, + Width = 300, HorizontalAlignment = Avalonia.Layout.HorizontalAlignment.Right }; @@ -499,23 +533,16 @@ namespace Observatory.UI.Views internal class PluginView { public string Name { get; set; } - public HashSet Types { get; set; } - public string TypesString { - get - { - return string.Join(", ", Types); - } - set { } } + public HashSet Types { get; set; } + public string TypesString + { + get => string.Join(", ", Types); + set { } + } public string Version { get; set; } public string Status { get; set; } } - enum PluginType - { - Worker, - Notifier, - } - internal static class GridExtention { public static void AddControl(this Grid grid, Control control, int row, int column, int span = 1)