diff --git a/ObservatoryBotanist/BotanistSettings.cs b/ObservatoryBotanist/BotanistSettings.cs index 5a42479..6540a5d 100644 --- a/ObservatoryBotanist/BotanistSettings.cs +++ b/ObservatoryBotanist/BotanistSettings.cs @@ -7,11 +7,12 @@ using System.Threading.Tasks; namespace Observatory.Botanist { + [SettingSuggestedColumnWidth(450)] class BotanistSettings { [SettingDisplayName("Enable Sampler Status Overlay")] public bool OverlayEnabled { get; set; } - [SettingDisplayName("Sampler Status Overlay is sticky until sampling complete (if enabled)")] + [SettingDisplayName("Status Overlay is sticky until sampling is complete")] public bool OverlayIsSticky { get; set; } } } diff --git a/ObservatoryCore/UI/SettingsForm.cs b/ObservatoryCore/UI/SettingsForm.cs index c13f5ac..db57d39 100644 --- a/ObservatoryCore/UI/SettingsForm.cs +++ b/ObservatoryCore/UI/SettingsForm.cs @@ -26,6 +26,12 @@ namespace Observatory.UI _plugin = plugin; // Filtered to only settings without SettingIgnore attribute + var attrib = _plugin.Settings.GetType().GetCustomAttribute(); + if (attrib != null && attrib.Width > 0) + { + int minScreenWidth = Screen.AllScreens.Min(s => s.Bounds.Width); + _colWidth = Math.Min(attrib.Width, minScreenWidth / 2); + } var settings = PluginManagement.PluginManager.GetSettingDisplayNames(plugin.Settings).Where(s => !Attribute.IsDefined(s.Key, typeof(SettingIgnore))); CreateControls(settings); @@ -51,12 +57,30 @@ namespace Observatory.UI foreach (var setting in settings) { - // Reset the column tracking for checkboxes if this isn't a checkbox + // Reset the column tracking for checkboxes if this isn't a checkbox or explicitly requested + // to start a new grouping of settings. int addedHeight = 35; - if (setting.Key.PropertyType.Name != "Boolean") + var newGroup = Attribute.GetCustomAttribute(setting.Key, typeof(SettingNewGroup)) as SettingNewGroup; + + if (setting.Key.PropertyType.Name != "Boolean" || newGroup != null) { if (recentHalfCol) _colHeight.Add(addedHeight); recentHalfCol = false; + + if (newGroup != null) + { + if (!string.IsNullOrEmpty(newGroup.Label)) + { + var label = CreateGroupLabel(newGroup.Label); + label.Location = GetSettingPosition(); + + Controls.Add(label); + trackBottomEdge(label); + _colHeight.Add(label.Height); + } + else + _colHeight.Add(10); + } } switch (setting.Key.GetValue(_plugin.Settings)) @@ -98,7 +122,7 @@ namespace Observatory.UI break; case int: // We have two options for integer values: - // 1) A slider (explicit by way of the SettingIntegerUseSlider attribute and bounded to 0..100 by default) + // 1) A slider (explicit by way of the SettingNumericUseSlider 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. var intLabel = CreateSettingLabel(setting.Value); @@ -109,7 +133,7 @@ namespace Observatory.UI } else { - intControl = CreateSettingNumericUpDown(setting.Key); + intControl = CreateSettingNumericUpDownForInt(setting.Key); } intLabel.Location = GetSettingPosition(); intControl.Location = GetSettingPosition(true); @@ -122,6 +146,23 @@ namespace Observatory.UI Controls.Add(intControl); trackBottomEdge(intControl); break; + case double: + // We have one options for double values: + // 1) A numeric up/down (default otherwise, and is unbounded by default). + // Bounds can be set via the SettingNumericBounds attribute. + var doubleLabel = CreateSettingLabel(setting.Value); + Control doubleControl = CreateSettingNumericUpDownForDouble(setting.Key); + doubleLabel.Location = GetSettingPosition(); + doubleControl.Location = GetSettingPosition(true); + + addedHeight = doubleControl.Height + 2; + doubleLabel.Height = doubleControl.Height; + doubleLabel.TextAlign = ContentAlignment.MiddleRight; + + Controls.Add(doubleLabel); + Controls.Add(doubleControl); + trackBottomEdge(doubleControl); + break; case Action action: var button = CreateSettingButton(setting.Value, action); @@ -152,8 +193,7 @@ namespace Observatory.UI private Point GetSettingPosition(bool secondCol = false) { - //return new Point(10 + (secondCol ? 200 : 0), -26 + _colHeight.Sum()); - return new Point(10 + (secondCol ? _colWidth : 0), 15 + _colHeight.Sum()); + return new Point(20 + (secondCol ? _colWidth + 20 : 0), 15 + _colHeight.Sum()); } @@ -170,6 +210,20 @@ namespace Observatory.UI return label; } + private Label CreateGroupLabel(string groupLabel) + { + Label label = new() + { + Text = groupLabel, + TextAlign = System.Drawing.ContentAlignment.MiddleLeft, + Width = _colWidth * 2, + ForeColor = Color.LightGray, + }; + label.Font = new Font(label.Font.FontFamily, label.Font.Size + 1, FontStyle.Bold); + label.Height += 10; // Add spacing. + return label; + } + private ComboBox CreateSettingDropdown(PropertyInfo setting, Dictionary dropdownItems) { var backingValueName = (SettingBackingValue?)Attribute.GetCustomAttribute(setting, typeof(SettingBackingValue)); @@ -225,7 +279,7 @@ namespace Observatory.UI private TrackBar CreateSettingTrackbar(PropertyInfo setting) { - SettingNumericBounds? bounds = (SettingNumericBounds?)System.Attribute.GetCustomAttribute(setting, typeof(SettingNumericBounds)); + SettingNumericBounds? bounds = (SettingNumericBounds?)Attribute.GetCustomAttribute(setting, typeof(SettingNumericBounds)); var minBound = Convert.ToInt32(bounds?.Minimum ?? 0); var maxBound = Convert.ToInt32(bounds?.Maximum ?? 100); @@ -246,16 +300,16 @@ namespace Observatory.UI trackBar.ValueChanged += (sender, e) => { - setting.SetValue(_plugin.Settings, Convert.ToInt32(trackBar.Value)); + setting.SetValue(_plugin.Settings, trackBar.Value); SaveSettings(); }; return trackBar; } - private NumericUpDown CreateSettingNumericUpDown(PropertyInfo setting) + private NumericUpDown CreateSettingNumericUpDownForInt(PropertyInfo setting) { - SettingNumericBounds? bounds = (SettingNumericBounds?)System.Attribute.GetCustomAttribute(setting, typeof(SettingNumericBounds)); + SettingNumericBounds? bounds = (SettingNumericBounds?)Attribute.GetCustomAttribute(setting, typeof(SettingNumericBounds)); NumericUpDown numericUpDown = new() { Width = _colWidth, @@ -265,7 +319,6 @@ namespace Observatory.UI }; numericUpDown.Value = (int?)setting.GetValue(_plugin.Settings) ?? 0; - numericUpDown.ValueChanged += (sender, e) => { setting.SetValue(_plugin.Settings, Convert.ToInt32(numericUpDown.Value)); @@ -275,6 +328,28 @@ namespace Observatory.UI return numericUpDown; } + private NumericUpDown CreateSettingNumericUpDownForDouble(PropertyInfo setting) + { + SettingNumericBounds? bounds = (SettingNumericBounds?)Attribute.GetCustomAttribute(setting, typeof(SettingNumericBounds)); + NumericUpDown numericUpDown = new() + { + Width = _colWidth, + Minimum = Convert.ToDecimal(bounds?.Minimum ?? Double.MinValue), + Maximum = Convert.ToDecimal(bounds?.Maximum ?? Double.MaxValue), + Increment = Convert.ToDecimal(bounds?.Increment ?? 1.0), + DecimalPlaces = bounds?.Precision ?? 1, + }; + + numericUpDown.Value = Convert.ToDecimal(setting.GetValue(_plugin.Settings) ?? 0); + numericUpDown.ValueChanged += (sender, e) => + { + setting.SetValue(_plugin.Settings, Convert.ToDouble(numericUpDown.Value)); + SaveSettings(); + }; + + return numericUpDown; + } + private CheckBox CreateBoolSetting(KeyValuePair setting) { CheckBox checkBox = new() diff --git a/ObservatoryExplorer/ExplorerSettings.cs b/ObservatoryExplorer/ExplorerSettings.cs index 2ab2bc5..bf331ac 100644 --- a/ObservatoryExplorer/ExplorerSettings.cs +++ b/ObservatoryExplorer/ExplorerSettings.cs @@ -1,12 +1,14 @@ using Observatory.Framework; using System; using System.Collections.Generic; +using System.ComponentModel; using System.Linq; using System.Text; using System.Threading.Tasks; namespace Observatory.Explorer { + [SettingSuggestedColumnWidth(300)] public class ExplorerSettings { public ExplorerSettings() @@ -14,6 +16,12 @@ namespace Observatory.Explorer CustomCriteriaFile = $"{Environment.GetFolderPath(Environment.SpecialFolder.Personal)}{System.IO.Path.DirectorySeparatorChar}ObservatoryCriteria.lua"; } + [SettingNewGroup("Display")] + [SettingDisplayName("Only Show Current System")] + public bool OnlyShowCurrentSystem { get; set; } + + + [SettingNewGroup("Built-in Criteria")] [SettingDisplayName("Landable & Terraformable")] public bool LandableTerraformable { get; set; } @@ -83,12 +91,9 @@ namespace Observatory.Explorer [SettingDisplayName("High-Value Body")] public bool HighValueMappable { get; set; } + [SettingNewGroup("Custom Criteria")] [SettingDisplayName("Enable Custom Criteria")] public bool EnableCustomCriteria { get; set; } - - [SettingDisplayName("Only Show Current System")] - public bool OnlyShowCurrentSystem { get; set; } - [SettingDisplayName("Custom Criteria File")] [System.Text.Json.Serialization.JsonIgnore] public System.IO.FileInfo CustomCriteria {get => new System.IO.FileInfo(CustomCriteriaFile); set => CustomCriteriaFile = value.FullName;} diff --git a/ObservatoryFramework/Attributes.cs b/ObservatoryFramework/Attributes.cs index 8bfd806..80d410a 100644 --- a/ObservatoryFramework/Attributes.cs +++ b/ObservatoryFramework/Attributes.cs @@ -6,6 +6,32 @@ using System.Threading.Tasks; namespace Observatory.Framework { + + #region Settings class attributes + /// + /// Specifies the width of a settings column in the settings view. There are two columns. + /// + [AttributeUsage(AttributeTargets.Class, AllowMultiple = false)] + public class SettingSuggestedColumnWidth : Attribute + { + /// + /// Specifies the width of a settings column in the settings view. There are two columns. + /// + /// Provides a hint of the width of a settings column. + public SettingSuggestedColumnWidth(int width) + { + Width = width; + } + + /// + /// Provides a hint of the width of a settings column. + /// + public int Width { get; } + } + #endregion + + #region Setting property attributes + /// /// Specifies text to display as the name of the setting in the UI instead of the property name. /// @@ -34,17 +60,24 @@ namespace Observatory.Framework } /// - /// Suggests default column width when building basic UI + /// Starts a new visual group of settings beginning with the current setting with an optional label. /// [AttributeUsage(AttributeTargets.Property | AttributeTargets.Field)] - public class ColumnSuggestedWidth : Attribute - { - public ColumnSuggestedWidth(int width) + public class SettingNewGroup : Attribute + { + /// + /// Starts a new visual group of settings beginning with the current setting with an optional label. + /// + /// An optional label describing the group. + public SettingNewGroup(string label = "") { - Width = width; + Label = label; } - public int Width { get; } + /// + /// An optional label describing the group. + /// + public string Label { get; } } /// @@ -97,6 +130,7 @@ namespace Observatory.Framework private double minimum; private double maximum; private double increment; + private int precision; /// /// Specify bounds for numeric inputs. @@ -104,11 +138,13 @@ namespace Observatory.Framework /// Minimum allowed value. /// Maximum allowed value. /// Increment between allowed values in slider/roller inputs. - public SettingNumericBounds(double minimum, double maximum, double increment = 1.0) + /// The number of digits to display for non integer values. + public SettingNumericBounds(double minimum, double maximum, double increment = 1.0, int precision = 1) { this.minimum = minimum; this.maximum = maximum; this.increment = increment; + this.precision = precision; } /// @@ -121,7 +157,7 @@ namespace Observatory.Framework } /// - /// Maxunyn allowed value. + /// Maximum allowed value. /// public double Maximum { @@ -137,5 +173,38 @@ namespace Observatory.Framework get => increment; set => increment = value; } + + /// + /// The number of digits to display for non integer values. + /// + public int Precision + { + get => precision; + set => precision = value; + } } + #endregion + + #region BasicUI attributes + /// + /// Suggests default column width when building basic plugin grid UI. + /// + [AttributeUsage(AttributeTargets.Property | AttributeTargets.Field)] + public class ColumnSuggestedWidth : Attribute + { + /// + /// Suggests default column width when building basic plugin grid UI. + /// + /// The suggested width of the annotated column. + public ColumnSuggestedWidth(int width) + { + Width = width; + } + + /// + /// The suggested width of the annotated column. + /// + public int Width { get; } + } + #endregion } diff --git a/ObservatoryFramework/ObservatoryFramework.xml b/ObservatoryFramework/ObservatoryFramework.xml index 63ab194..d6172cd 100644 --- a/ObservatoryFramework/ObservatoryFramework.xml +++ b/ObservatoryFramework/ObservatoryFramework.xml @@ -4,6 +4,22 @@ ObservatoryFramework + + + Specifies the width of a settings column in the settings view. There are two columns. + + + + + Specifies the width of a settings column in the settings view. There are two columns. + + Provides a hint of the width of a settings column. + + + + Provides a hint of the width of a settings column. + + Specifies text to display as the name of the setting in the UI instead of the property name. @@ -20,9 +36,20 @@ Accessor to get/set displayed name. - + - Suggests default column width when building basic UI + Starts a new visual group of settings beginning with the current setting with an optional label. + + + + + Starts a new visual group of settings beginning with the current setting with an optional label. + + An optional label describing the group. + + + + An optional label describing the group. @@ -56,13 +83,14 @@ Specify bounds for numeric inputs. - + Specify bounds for numeric inputs. Minimum allowed value. Maximum allowed value. Increment between allowed values in slider/roller inputs. + The number of digits to display for non integer values. @@ -71,7 +99,7 @@ - Maxunyn allowed value. + Maximum allowed value. @@ -79,6 +107,27 @@ Increment between allowed values in slider/roller inputs. + + + The number of digits to display for non integer values. + + + + + Suggests default column width when building basic plugin grid UI. + + + + + Suggests default column width when building basic plugin grid UI. + + The suggested width of the annotated column. + + + + The suggested width of the annotated column. + + Provides data for Elite Dangerous journal events.