diff --git a/ObservatoryCore/PluginManagement/PluginCore.cs b/ObservatoryCore/PluginManagement/PluginCore.cs index 3fa0637..0bb0a33 100644 --- a/ObservatoryCore/PluginManagement/PluginCore.cs +++ b/ObservatoryCore/PluginManagement/PluginCore.cs @@ -3,6 +3,7 @@ using Observatory.Framework.Files; using Observatory.Framework.Interfaces; using Observatory.NativeNotification; using System; +using System.Collections.ObjectModel; using System.IO; namespace Observatory.PluginManagement @@ -96,47 +97,39 @@ namespace Observatory.PluginManagement /// /// /// - public void AddGridItem(IObservatoryWorker worker, object item) + public void AddGridItem(IObservatoryWorker worker, List item) { Avalonia.Threading.Dispatcher.UIThread.InvokeAsync(() => { - worker.PluginUI.DataGrid.Add(item); + ObservableCollection newRow = new(item); + worker.PluginUI.BasicGrid.Items.Add(newRow); //Hacky removal of original empty object if one was used to populate columns - if (worker.PluginUI.DataGrid.Count == 2) + if (worker.PluginUI.BasicGrid.Items.Count == 2) { - if (FirstRowIsAllNull(worker)) - worker.PluginUI.DataGrid.RemoveAt(0); + bool allNull = true; + + foreach (var cell in worker.PluginUI.BasicGrid.Items[0]) + { + if (cell != null) + { + allNull = false; + break; + } + } + + if (allNull) + worker.PluginUI.BasicGrid.Items.RemoveAt(0); } + }); } - /// - /// Adds multiple items to the datagrid on UI thread to ensure visual update. - /// - /// - /// - public void AddGridItems(IObservatoryWorker worker, IEnumerable items) + public void ClearGrid(IObservatoryWorker worker) { Avalonia.Threading.Dispatcher.UIThread.InvokeAsync(() => { - var cleanEmptyRow = worker.PluginUI.DataGrid.Count == 1 && FirstRowIsAllNull(worker) && items.Count() > 0; - foreach (var item in items) - { - worker.PluginUI.DataGrid.Add(item); - } - if (cleanEmptyRow) - worker.PluginUI.DataGrid.RemoveAt(0); - }); - } - - public void ClearGrid(IObservatoryWorker worker, object templateItem) - { - Avalonia.Threading.Dispatcher.UIThread.InvokeAsync(() => - { - worker.PluginUI.DataGrid.Add(templateItem); - while (worker.PluginUI.DataGrid.Count > 1) - worker.PluginUI.DataGrid.RemoveAt(0); + worker.PluginUI.BasicGrid.Items.Clear(); }); } diff --git a/ObservatoryCore/UI/ViewModels/BasicUIViewModel.cs b/ObservatoryCore/UI/ViewModels/BasicUIViewModel.cs index a2fd62c..e7f472f 100644 --- a/ObservatoryCore/UI/ViewModels/BasicUIViewModel.cs +++ b/ObservatoryCore/UI/ViewModels/BasicUIViewModel.cs @@ -61,11 +61,11 @@ namespace Observatory.UI.ViewModels } } - - public BasicUIViewModel(ObservableCollection headers, ObservableCollection formats) + public BasicUIViewModel(BasicGrid basicGrid) { - Headers = headers; - Formats = formats; + Headers = basicGrid.Headers; + Formats = basicGrid.Formats; + Items = basicGrid.Items; } private PluginUI.UIType _uiType; diff --git a/ObservatoryCore/UI/ViewModels/CoreViewModel.cs b/ObservatoryCore/UI/ViewModels/CoreViewModel.cs index dfd8557..bccdc54 100644 --- a/ObservatoryCore/UI/ViewModels/CoreViewModel.cs +++ b/ObservatoryCore/UI/ViewModels/CoreViewModel.cs @@ -36,7 +36,7 @@ namespace Observatory.UI.ViewModels { CoreModel coreModel = new(); coreModel.Name = worker.ShortName; - coreModel.UI = new BasicUIViewModel(worker.PluginUI.DataGrid) + coreModel.UI = new BasicUIViewModel(worker.PluginUI.BasicGrid) { UIType = worker.PluginUI.PluginUIType }; @@ -55,7 +55,7 @@ namespace Observatory.UI.ViewModels } - tabs.Add(new CoreModel() { Name = "Core", UI = new BasicUIViewModel(new ObservableCollection()) { UIType = Framework.PluginUI.UIType.Core } }); + tabs.Add(new CoreModel() { Name = "Core", UI = new BasicUIViewModel(new Framework.BasicGrid()) { UIType = Framework.PluginUI.UIType.Core } }); if (Properties.Core.Default.StartMonitor) ToggleMonitor(); @@ -112,67 +112,95 @@ namespace Observatory.UI.ViewModels try { var exportFolder = Properties.Core.Default.ExportFolder; - if (string.IsNullOrEmpty(exportFolder) || !Directory.Exists(exportFolder)) + + if (string.IsNullOrEmpty(exportFolder)) { exportFolder = System.Environment.GetFolderPath(Environment.SpecialFolder.MyDocuments); - - OpenFolderDialog openFolderDialog = new() - { - Directory = exportFolder - }; - - var application = (IClassicDesktopStyleApplicationLifetime)Avalonia.Application.Current.ApplicationLifetime; - - var selectedFolder = await openFolderDialog.ShowAsync(application.MainWindow); - - if (!string.IsNullOrEmpty(selectedFolder)) - { - Properties.Core.Default.ExportFolder = selectedFolder; - Properties.Core.Default.Save(); - exportFolder = selectedFolder; - } } - var exportStyle = Properties.Core.Default.ExportStyle; - if (string.IsNullOrEmpty(exportStyle)) + OpenFolderDialog openFolderDialog = new() { - exportStyle = "Fixed width"; - Properties.Core.Default.ExportStyle = exportStyle; + Directory = exportFolder + }; + + var application = (IClassicDesktopStyleApplicationLifetime)Avalonia.Application.Current.ApplicationLifetime; + + var selectedFolder = await openFolderDialog.ShowAsync(application.MainWindow); + + if (!string.IsNullOrEmpty(selectedFolder)) + { + Properties.Core.Default.ExportFolder = selectedFolder; Properties.Core.Default.Save(); - } + exportFolder = selectedFolder; - foreach (var tab in tabs.Where(t => t.Name != "Core")) - { - var ui = (BasicUIViewModel)tab.UI; - List selectedData; - bool specificallySelected = ui.SelectedItems?.Count > 1; - - if (specificallySelected) + foreach (var tab in tabs.Where(t => t.Name != "Core")) { - selectedData = new(); + var ui = (BasicUIViewModel)tab.UI; + List selectedData; + bool specificallySelected = ui.SelectedItems?.Count > 1; - foreach (var item in ui.SelectedItems) - selectedData.Add(item); + if (specificallySelected) + { + selectedData = new(); + + foreach (var item in ui.SelectedItems) + selectedData.Add(item); + } + else + { + selectedData = new(); // TODO: Make this work in new UI + } + + var columns = selectedData[0].GetType().GetProperties(); + Dictionary colSize = new(); + Dictionary> colContent = new(); + + foreach (var column in columns) + { + colSize.Add(column.Name, 0); + colContent.Add(column.Name, new()); + } + + foreach (var line in selectedData) + { + var lineType = line.GetType(); // some plugins have different line types, so don't move this out of loop + foreach (var column in colContent) + { + var cellValue = lineType.GetProperty(column.Key)?.GetValue(line)?.ToString() ?? string.Empty; + column.Value.Add(cellValue); + if (colSize[column.Key] < cellValue.Length) + colSize[column.Key] = cellValue.Length; + } + } + + System.Text.StringBuilder exportData = new(); + + + foreach (var colTitle in colContent.Keys) + { + if (colSize[colTitle] < colTitle.Length) + colSize[colTitle] = colTitle.Length; + + exportData.Append(colTitle.PadRight(colSize[colTitle]) + " "); + } + exportData.AppendLine(); + + for (int i = 0; i < colContent.First().Value.Count; i++) + { + foreach (var column in colContent) + { + if (column.Value[i].Length > 0 && !char.IsNumber(column.Value[i][0]) && column.Value[i].Count(char.IsLetter) / (float)column.Value[i].Length > 0.25) + exportData.Append(column.Value[i].PadRight(colSize[column.Key]) + " "); + else + exportData.Append(column.Value[i].PadLeft(colSize[column.Key]) + " "); + } + exportData.AppendLine(); + } + + string exportPath = $"{exportFolder}{System.IO.Path.DirectorySeparatorChar}Observatory Export - {DateTime.UtcNow:yyyyMMdd-HHmmss} - {tab.Name}.txt"; + + System.IO.File.WriteAllText(exportPath, exportData.ToString()); } - else - { - selectedData = ui.BasicUIGrid.ToList(); - } - - System.Text.StringBuilder exportData; - switch (exportStyle) - { - case "Tab separated": - exportData = ExportTabSeparated(selectedData); - break; - default: // Fixed width. - exportData = ExportFixedWidth(selectedData); - break; - } - - string exportPath = $"{exportFolder}{System.IO.Path.DirectorySeparatorChar}Observatory Export - {DateTime.UtcNow:yyyyMMdd-HHmmss} - {tab.Name}.txt"; - - System.IO.File.WriteAllText(exportPath, exportData.ToString()); } } catch (Exception e) @@ -182,78 +210,6 @@ namespace Observatory.UI.ViewModels new List<(string, string)> { ("An error occurred while exporting; output may be missing or incomplete." + Environment.NewLine + "Please check the error log (found in your Documents folder) for more details and visit our discord to report it.", e.Message) }); } - - static System.Text.StringBuilder ExportTabSeparated(List selectedData) - { - System.Text.StringBuilder exportData = new(); - - var columnNames = selectedData[0].GetType().GetProperties().Select(c => c.Name).ToList(); - exportData.AppendJoin('\t', columnNames).AppendLine(); - - var lastColumn = columnNames.Last(); - foreach (var line in selectedData) - { - var lineType = line.GetType(); // some plugins have different line types, so don't move this out of loop - foreach (var columnName in columnNames) - { - var cellValue = lineType.GetProperty(columnName)?.GetValue(line)?.ToString() ?? string.Empty; - exportData.Append(cellValue).Append('\t'); - } - exportData.AppendLine(); - } - return exportData; - } - - static System.Text.StringBuilder ExportFixedWidth(List selectedData) - { - Dictionary colSize = new(); - Dictionary> colContent = new(); - - var columns = selectedData[0].GetType().GetProperties(); - foreach (var column in columns) - { - colSize.Add(column.Name, column.Name.Length); - colContent.Add(column.Name, new()); - } - - foreach (var line in selectedData) - { - var lineType = line.GetType(); // some plugins have different line types, so don't move this out of loop - foreach (var column in colContent) - { - var cellValue = lineType.GetProperty(column.Key)?.GetValue(line)?.ToString() ?? string.Empty; - column.Value.Add(cellValue); - if (colSize[column.Key] < cellValue.Length) - colSize[column.Key] = cellValue.Length; - } - } - - System.Text.StringBuilder exportData = new(); - - - foreach (var colTitle in colContent.Keys) - { - if (colSize[colTitle] < colTitle.Length) - colSize[colTitle] = colTitle.Length; - - exportData.Append(colTitle.PadRight(colSize[colTitle]) + " "); - } - exportData.AppendLine(); - - for (int i = 0; i < colContent.First().Value.Count; i++) - { - foreach (var column in colContent) - { - if (column.Value[i].Length > 0 && !char.IsNumber(column.Value[i][0]) && column.Value[i].Count(char.IsLetter) / (float)column.Value[i].Length > 0.25) - exportData.Append(column.Value[i].PadRight(colSize[column.Key]) + " "); - else - exportData.Append(column.Value[i].PadLeft(colSize[column.Key]) + " "); - } - exportData.AppendLine(); - } - - return exportData; - } } public void ClearGrid() @@ -262,15 +218,7 @@ namespace Observatory.UI.ViewModels { var ui = (BasicUIViewModel)tab.UI; - var rowTemplate = ui.BasicUIGrid.First(); - - foreach (var property in rowTemplate.GetType().GetProperties()) - { - property.SetValue(rowTemplate, null); - } - - ui.BasicUIGrid.Clear(); - ui.BasicUIGrid.Add(rowTemplate); + ui.Items.Clear(); // For some reason UIType's change event will properly // redraw the grid, not BasicUIGrid's. diff --git a/ObservatoryFramework/BasicGrid.cs b/ObservatoryFramework/BasicGrid.cs new file mode 100644 index 0000000..b36105e --- /dev/null +++ b/ObservatoryFramework/BasicGrid.cs @@ -0,0 +1,23 @@ +using System; +using System.Collections.Generic; +using System.Collections.ObjectModel; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace Observatory.Framework +{ + public class BasicGrid + { + public BasicGrid() + { + Headers = new(); + Formats = new(); + Items = new(); + } + + public readonly ObservableCollection Headers; + public readonly ObservableCollection Formats; + public readonly ObservableCollection> Items; + } +} diff --git a/ObservatoryFramework/Interfaces.cs b/ObservatoryFramework/Interfaces.cs index 3a1a73b..52ad4b2 100644 --- a/ObservatoryFramework/Interfaces.cs +++ b/ObservatoryFramework/Interfaces.cs @@ -154,7 +154,7 @@ namespace Observatory.Framework.Interfaces /// /// Reference to the calling plugin's worker interface. /// Grid item to be added. Object type should match original template item used to create the grid. - public void AddGridItem(IObservatoryWorker worker, object item); + public void AddGridItem(IObservatoryWorker worker, List item); /// /// Add multiple items to the bottom of the basic UI grid. @@ -167,8 +167,7 @@ namespace Observatory.Framework.Interfaces /// Clears basic UI grid, removing all items. /// /// Reference to the calling plugin's worker interface. - /// Template item used to re-initialise the grid. - public void ClearGrid(IObservatoryWorker worker, object templateItem); + public void ClearGrid(IObservatoryWorker worker); /// /// Requests current Elite Dangerous status.json content. diff --git a/ObservatoryFramework/ObservatoryFramework.xml b/ObservatoryFramework/ObservatoryFramework.xml index 8d53c9a..1c9f796 100644 --- a/ObservatoryFramework/ObservatoryFramework.xml +++ b/ObservatoryFramework/ObservatoryFramework.xml @@ -1439,26 +1439,18 @@ Guid of notification to be updated. NotificationArgs object specifying updated notification content and behaviour. - + Add an item to the bottom of the basic UI grid. Reference to the calling plugin's worker interface. Grid item to be added. Object type should match original template item used to create the grid. - - - Add multiple items to the bottom of the basic UI grid. - - Reference to the calling plugin's worker interface. - Grid items to be added. Object types should match original template item used to create the grid. - - + Clears basic UI grid, removing all items. Reference to the calling plugin's worker interface. - Template item used to re-initialise the grid. @@ -1520,13 +1512,12 @@ (Untested/not implemented) - + - Collection bound to DataGrid used byu plugins with UIType.Basic. - Objects in collection should be of a class defined within the plugin consisting of string properties.
Each object is a single row, and the property names are used as column headers.
+ >Two-dimensional collection of items to display in UI grid for UIType.Basic
- + Instantiate PluginUI of UIType.Basic. diff --git a/ObservatoryFramework/PluginUI.cs b/ObservatoryFramework/PluginUI.cs index 0a8e718..d0a3403 100644 --- a/ObservatoryFramework/PluginUI.cs +++ b/ObservatoryFramework/PluginUI.cs @@ -23,25 +23,9 @@ namespace Observatory.Framework public object UI; /// - /// Collection bound to DataGrid used by plugins with UIType.Basic. - /// Objects in collection should be of a class defined within the plugin consisting of string properties.
Each object is a single row, and the property names are used as column headers.
+ /// >Two-dimensional collection of items to display in UI grid for UIType.Basic ///
- public ObservableCollection DataGrid; - - /// - /// Collection bound to DataGrid headers used by plugins with UIType.Basic. - /// - public ObservableCollection Headers; - - /// - /// Collection used to specify formatting of items in respective columns. - /// - public ObservableCollection Formats; - - /// - /// Two-dimensional collection of items to display in UI grid. - /// - public ObservableCollection> Items; + public BasicGrid BasicGrid; /// /// Instantiate PluginUI of UIType.Basic. @@ -50,10 +34,10 @@ namespace Observatory.Framework /// Collection bound to DataGrid used byu plugins with UIType.Basic. /// Objects in collection should be of a class defined within the plugin consisting of string properties.
Each object is a single row, and the property names are used as column headers.
/// - public PluginUI(ObservableCollection DataGrid) + public PluginUI(BasicGrid basicGrid) { PluginUIType = UIType.Basic; - this.DataGrid = DataGrid; + BasicGrid = basicGrid; } ///