From 71266dec45d43324a56fea4491495bae4ab108af Mon Sep 17 00:00:00 2001 From: Xjph Date: Mon, 13 Sep 2021 11:24:44 -0230 Subject: [PATCH 1/5] Update README.md Add custom criteria documentation link --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 9c5f319..bc9280c 100644 --- a/README.md +++ b/README.md @@ -23,6 +23,8 @@ For specifics on what each plugin does, please refer to their respective reposit * [Explorer](https://github.com/Xjph/ObservatoryExplorer) * [Botanist](https://github.com/Xjph/ObservatoryBotanist) +If you're interested in Custom Criteria for Explorer in particular you can find [the documentation for writing them](https://github.com/Xjph/ObservatoryExplorer/wiki/Lua-Custom-Criteria) in the Explorer project wiki. + For information on how to create a plugin, refer to the repository for [ObservatoryFramework](https://github.com/Xjph/ObservatoryFramework). ## Prerequisites for use From 51595006442faadb96184403f899c719bb7aff93 Mon Sep 17 00:00:00 2001 From: Xjph Date: Tue, 12 Oct 2021 20:45:00 -0230 Subject: [PATCH 2/5] Notification overhaul and update to avaloniaui 0.10.7 --- .../NativeNotification/NativePopup.cs | 52 ++++++++ .../NativeNotification/NativeVoice.cs | 74 +++++++++++ ObservatoryCore/ObservatoryCore.csproj | 12 +- .../PluginManagement/PlaceholderPlugin.cs | 2 +- .../PluginManagement/PluginCore.cs | 50 +++++--- .../PluginManagement/PluginEventHandler.cs | 4 +- .../UI/Models/NotificationModel.cs | 3 + .../UI/ViewModels/NotificationViewModel.cs | 10 +- ObservatoryCore/UI/Views/BasicUIView.axaml.cs | 31 +++-- .../UI/Views/NotificationView.axaml.cs | 116 +++++++++++++----- 10 files changed, 279 insertions(+), 75 deletions(-) create mode 100644 ObservatoryCore/NativeNotification/NativePopup.cs create mode 100644 ObservatoryCore/NativeNotification/NativeVoice.cs diff --git a/ObservatoryCore/NativeNotification/NativePopup.cs b/ObservatoryCore/NativeNotification/NativePopup.cs new file mode 100644 index 0000000..4a4f2ba --- /dev/null +++ b/ObservatoryCore/NativeNotification/NativePopup.cs @@ -0,0 +1,52 @@ +using Observatory.Framework; +using System; +using System.Collections.Generic; +using Observatory.UI.Views; +using Observatory.UI.ViewModels; + +namespace Observatory.NativeNotification +{ + public class NativePopup + { + private Dictionary notifications; + + public NativePopup() + { + notifications = new(); + } + + public Guid InvokeNativeNotification(NotificationArgs notificationArgs) + { + var notificationGuid = Guid.NewGuid(); + Avalonia.Threading.Dispatcher.UIThread.InvokeAsync(() => + { + var notifyWindow = new NotificationView(notificationGuid) { DataContext = new NotificationViewModel(notificationArgs) }; + notifyWindow.Closed += NotifyWindow_Closed; + notifications.Add(notificationGuid, notifyWindow); + notifyWindow.Show(); + }); + + return notificationGuid; + } + + private void NotifyWindow_Closed(object sender, EventArgs e) + { + var notification = (NotificationView)sender; + + if (notifications.ContainsKey(notification.Guid)) + notifications.Remove(notification.Guid); + } + + public void CloseNotification(Guid guid) + { + if (notifications.ContainsKey(guid)) + notifications[guid].Close(); + } + + public void UpdateNotification(Guid guid, NotificationArgs notificationArgs) + { + if (notifications.ContainsKey(guid)) + notifications[guid].DataContext = new NotificationViewModel(notificationArgs); + } + } +} diff --git a/ObservatoryCore/NativeNotification/NativeVoice.cs b/ObservatoryCore/NativeNotification/NativeVoice.cs new file mode 100644 index 0000000..41e6838 --- /dev/null +++ b/ObservatoryCore/NativeNotification/NativeVoice.cs @@ -0,0 +1,74 @@ +using Observatory.Framework; +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; +using System.Speech.Synthesis; +using System.Runtime.InteropServices; + +namespace Observatory.NativeNotification +{ + public class NativeVoice + { + private Queue notificationEvents; + private bool processing; + + public NativeVoice() + { + notificationEvents = new(); + processing = false; + } + + public void EnqueueAndAnnounce(NotificationArgs eventArgs) + { + notificationEvents.Enqueue(eventArgs); + + if (!processing) + { + processing = true; + ProcessQueueAsync(); + } + } + + private async void ProcessQueueAsync() + { + await Task.Factory.StartNew(ProcessQueue); + } + + private void ProcessQueue() + { + if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) + { + var speech = new SpeechSynthesizer() + { + Volume = Properties.Core.Default.VoiceVolume, + Rate = Properties.Core.Default.VoiceRate + }; + speech.SelectVoice(Properties.Core.Default.VoiceSelected); + + while (notificationEvents.Any()) + { + var notification = notificationEvents.Dequeue(); + + if (notification.TitleSsml?.Length > 0) + { + speech.SpeakSsml(notification.TitleSsml); + } + else + { + speech.Speak(notification.Title); + } + + if (notification.DetailSsml?.Length > 0) + { + speech.SpeakSsml(notification.DetailSsml); + } + else + { + speech.Speak(notification.Detail); + } + } + } + processing = false; + } + } +} diff --git a/ObservatoryCore/ObservatoryCore.csproj b/ObservatoryCore/ObservatoryCore.csproj index 475837a..aa4efa2 100644 --- a/ObservatoryCore/ObservatoryCore.csproj +++ b/ObservatoryCore/ObservatoryCore.csproj @@ -22,12 +22,12 @@ - - - - - - + + + + + + diff --git a/ObservatoryCore/PluginManagement/PlaceholderPlugin.cs b/ObservatoryCore/PluginManagement/PlaceholderPlugin.cs index 1e8f442..b357153 100644 --- a/ObservatoryCore/PluginManagement/PlaceholderPlugin.cs +++ b/ObservatoryCore/PluginManagement/PlaceholderPlugin.cs @@ -32,7 +32,7 @@ namespace Observatory.PluginManagement public void Load(IObservatoryCore observatoryCore) { } - public void OnNotificationEvent(string title, string text) + public void OnNotificationEvent(NotificationArgs notificationArgs) { } } } diff --git a/ObservatoryCore/PluginManagement/PluginCore.cs b/ObservatoryCore/PluginManagement/PluginCore.cs index 71b8891..cc4104e 100644 --- a/ObservatoryCore/PluginManagement/PluginCore.cs +++ b/ObservatoryCore/PluginManagement/PluginCore.cs @@ -1,6 +1,7 @@ using Observatory.Framework; using Observatory.Framework.Files; using Observatory.Framework.Interfaces; +using Observatory.NativeNotification; using System; using System.Runtime.InteropServices; @@ -9,6 +10,15 @@ namespace Observatory.PluginManagement public class PluginCore : IObservatoryCore { + private readonly NativeVoice NativeVoice; + private readonly NativePopup NativePopup; + + public PluginCore() + { + NativeVoice = new(); + NativePopup = new(); + } + public string Version => System.Reflection.Assembly.GetEntryAssembly().GetName().Version.ToString(); public Status GetStatus() @@ -16,38 +26,42 @@ namespace Observatory.PluginManagement throw new NotImplementedException(); } - public void SendNotification(string title, string text) + public Guid SendNotification(string title, string text) { + return SendNotification(new NotificationArgs() { Title = title, Detail = text }); + } + + public Guid SendNotification(NotificationArgs notificationArgs) + { + var guid = Guid.Empty; + if (!LogMonitor.GetInstance.ReadAllInProgress()) { var handler = Notification; - handler?.Invoke(this, new NotificationEventArgs() { Title = title, Detail = text }); + handler?.Invoke(this, notificationArgs); if (Properties.Core.Default.NativeNotify) { - InvokeNativeNotification(title, text); + guid = NativePopup.InvokeNativeNotification(notificationArgs); } - if (Properties.Core.Default.VoiceNotify && RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) + if (Properties.Core.Default.VoiceNotify) { - var speech = new System.Speech.Synthesis.SpeechSynthesizer() - { - Volume = Properties.Core.Default.VoiceVolume, - Rate = Properties.Core.Default.VoiceRate - }; - speech.SelectVoice(Properties.Core.Default.VoiceSelected); - speech.SpeakAsync(title + ": " + text); + NativeVoice.EnqueueAndAnnounce(notificationArgs); } } + + return guid; } - private void InvokeNativeNotification(string title, string text) + public void CancelNotification(Guid id) { - Avalonia.Threading.Dispatcher.UIThread.InvokeAsync(() => - { - var notifyWindow = new UI.Views.NotificationView() { DataContext = new UI.ViewModels.NotificationViewModel(title, text) }; - notifyWindow.Show(); - }); + + } + + public void UpdateNotification(Guid id, NotificationArgs notificationArgs) + { + } /// @@ -97,6 +111,6 @@ namespace Observatory.PluginManagement Avalonia.Threading.Dispatcher.UIThread.InvokeAsync(action); } - public event EventHandler Notification; + public event EventHandler Notification; } } diff --git a/ObservatoryCore/PluginManagement/PluginEventHandler.cs b/ObservatoryCore/PluginManagement/PluginEventHandler.cs index da9ff84..3507e48 100644 --- a/ObservatoryCore/PluginManagement/PluginEventHandler.cs +++ b/ObservatoryCore/PluginManagement/PluginEventHandler.cs @@ -37,11 +37,11 @@ namespace Observatory.PluginManagement } } - public void OnNotificationEvent(object source, NotificationEventArgs notificationEventArgs) + public void OnNotificationEvent(object source, NotificationArgs notificationArgs) { foreach (var notifier in observatoryNotifiers) { - notifier.OnNotificationEvent(notificationEventArgs.Title, notificationEventArgs.Detail); + notifier.OnNotificationEvent(notificationArgs); } } } diff --git a/ObservatoryCore/UI/Models/NotificationModel.cs b/ObservatoryCore/UI/Models/NotificationModel.cs index 8519da8..0881c81 100644 --- a/ObservatoryCore/UI/Models/NotificationModel.cs +++ b/ObservatoryCore/UI/Models/NotificationModel.cs @@ -11,5 +11,8 @@ namespace Observatory.UI.Models public string Title { get; set; } public string Detail { get; set; } public string Colour { get; set; } + public int Timeout { get; set; } + public double XPos { get; set; } + public double YPos { get; set; } } } diff --git a/ObservatoryCore/UI/ViewModels/NotificationViewModel.cs b/ObservatoryCore/UI/ViewModels/NotificationViewModel.cs index 4225214..a84a085 100644 --- a/ObservatoryCore/UI/ViewModels/NotificationViewModel.cs +++ b/ObservatoryCore/UI/ViewModels/NotificationViewModel.cs @@ -3,18 +3,22 @@ using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; +using Observatory.Framework; namespace Observatory.UI.ViewModels { public class NotificationViewModel : ViewModelBase { - public NotificationViewModel(string title, string detail) + public NotificationViewModel(NotificationArgs notificationArgs) { Notification = new() { - Title = title, - Detail = detail, + Title = notificationArgs.Title, + Detail = notificationArgs.Detail, + Timeout = notificationArgs.Timeout, + XPos = notificationArgs.XPos, + YPos = notificationArgs.YPos, Colour = Avalonia.Media.Color.FromUInt32(Properties.Core.Default.NativeNotifyColour).ToString() }; diff --git a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs index f50ce07..3a01bfc 100644 --- a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs +++ b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs @@ -63,7 +63,6 @@ namespace Observatory.UI.Views e.Column.CanUserResize = true; e.Column.CanUserSort = true; } - private void UITypeChange() { var uiPanel = this.Find("UIPanel"); @@ -109,6 +108,7 @@ namespace Observatory.UI.Views { var dataContext = ((ViewModels.BasicUIViewModel)dataGrid.DataContext).BasicUIGrid; dataContext.CollectionChanged += ScrollToLast; + } } @@ -200,9 +200,7 @@ namespace Observatory.UI.Views { Header = "Popup Notifications", DataContext = Properties.Core.Default, - Margin = new Thickness(0, 0), - Background = this.Background, - BorderThickness = new Thickness(0) + Margin = new Thickness(0, 0) }; Grid notificationGrid = new() { Margin = new Thickness(10, 10) }; @@ -244,8 +242,13 @@ namespace Observatory.UI.Views { Avalonia.Threading.Dispatcher.UIThread.InvokeAsync(() => { - var notifyWindow = new UI.Views.NotificationView() { DataContext = new UI.ViewModels.NotificationViewModel("Test Notification", "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Cras suscipit hendrerit libero ac scelerisque.") }; - notifyWindow.Show(); + var notificationArgs = new NotificationArgs() + { + Title = "Test Notification", + Detail = "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Cras suscipit hendrerit libero ac scelerisque." + }; + + new NativeNotification.NativePopup().InvokeNativeNotification(notificationArgs); }); }; @@ -448,9 +451,7 @@ namespace Observatory.UI.Views { Header = "Voice Notifications", DataContext = Properties.Core.Default, - Margin = new Thickness(0, 0), - Background = this.Background, - BorderThickness = new Thickness(0) + Margin = new Thickness(0, 0) }; Grid voiceGrid = new() { Margin = new Thickness(10, 10) }; @@ -489,13 +490,6 @@ namespace Observatory.UI.Views { if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows) && Properties.Core.Default.VoiceSelected.Length > 0) { - var speech = new System.Speech.Synthesis.SpeechSynthesizer() - { - Volume = Properties.Core.Default.VoiceVolume, - Rate = Properties.Core.Default.VoiceRate - }; - speech.SelectVoice(Properties.Core.Default.VoiceSelected); - List harvardSentences = new() { "Oak is strong and also gives shade.", @@ -510,7 +504,10 @@ namespace Observatory.UI.Views "Move the vat over the hot fire." }; - speech.SpeakAsync("Speech Synthesis Test: " + harvardSentences.OrderBy(s => new Random().NextDouble()).First()); + NotificationArgs args = new() { Title = "Speech Synthesis Test", Detail = harvardSentences.OrderBy(s => new Random().NextDouble()).First() }; + + new NativeNotification.NativeVoice().EnqueueAndAnnounce(args); + } }; diff --git a/ObservatoryCore/UI/Views/NotificationView.axaml.cs b/ObservatoryCore/UI/Views/NotificationView.axaml.cs index 8d0d785..4a343d4 100644 --- a/ObservatoryCore/UI/Views/NotificationView.axaml.cs +++ b/ObservatoryCore/UI/Views/NotificationView.axaml.cs @@ -2,25 +2,77 @@ using Avalonia; using Avalonia.Controls; using Avalonia.Layout; using Avalonia.Markup.Xaml; +using Observatory.UI.ViewModels; using System; +using System.Timers; using System.Runtime.InteropServices; namespace Observatory.UI.Views { public partial class NotificationView : Window { - public NotificationView() + private readonly double scale; + private readonly Timer timer; + private readonly Guid guid; + + public NotificationView() : this(default) + { } + + public NotificationView(Guid guid) { + this.guid = guid; InitializeComponent(); SystemDecorations = SystemDecorations.None; ShowInTaskbar = false; MakeClickThrough(); //Platform specific, currently windows only. - - int screen = Properties.Core.Default.NativeNotifyScreen; - int corner = Properties.Core.Default.NativeNotifyCorner; - string font = Properties.Core.Default.NativeNotifyFont; - double scale = Properties.Core.Default.NativeNotifyScale / 100.0; + + this.DataContextChanged += NotificationView_DataContextChanged; + scale = Properties.Core.Default.NativeNotifyScale / 100.0; + + AdjustText(); + + AdjustPanel(); + + AdjustPosition(); + + timer = new(); + timer.Elapsed += CloseNotification; + timer.Interval = Properties.Core.Default.NativeNotifyTimeout; + timer.Start(); + +#if DEBUG + this.AttachDevTools(); +#endif + } + + public Guid Guid { get => guid; } + + private void NotificationView_DataContextChanged(object sender, EventArgs e) + { + var notification = ((NotificationViewModel)DataContext).Notification; + + AdjustText(); + + AdjustPanel(); + + AdjustPosition(notification.XPos / 100, notification.YPos / 100); + + if (notification.Timeout > 0) + { + timer.Stop(); + timer.Interval = notification.Timeout; + timer.Start(); + } + else if (notification.Timeout == 0) + { + timer.Stop(); + } + } + + private void AdjustText() + { + string font = Properties.Core.Default.NativeNotifyFont; var titleText = this.Find("Title"); var detailText = this.Find("Detail"); @@ -34,7 +86,10 @@ namespace Observatory.UI.Views titleText.FontSize *= scale; detailText.FontSize *= scale; + } + private void AdjustPanel() + { var textPanel = this.Find("TextPanel"); Width *= scale; Height *= scale; @@ -43,8 +98,13 @@ namespace Observatory.UI.Views var textBorder = this.Find("TextBorder"); textBorder.BorderThickness *= scale; - + } + + private void AdjustPosition(double xOverride = -1.0, double yOverride = -1.0) + { PixelRect screenBounds; + int screen = Properties.Core.Default.NativeNotifyScreen; + int corner = Properties.Core.Default.NativeNotifyCorner; if (screen == -1 || screen > Screens.All.Count) screenBounds = Screens.Primary.Bounds; @@ -55,29 +115,29 @@ namespace Observatory.UI.Views double scaleWidth = Width * displayScale; double scaleHeight = Height * displayScale; - switch (corner) + if (xOverride >= 0 && yOverride >= 0) { - default: case 0: - Position = screenBounds.BottomRight - new PixelPoint((int)scaleWidth + 50, (int)scaleHeight + 50); - break; - case 1: - Position = screenBounds.BottomLeft - new PixelPoint(-50, (int)scaleHeight + 50); - break; - case 2: - Position = screenBounds.TopRight - new PixelPoint((int)scaleWidth + 50, -50); - break; - case 3: - Position = screenBounds.TopLeft + new PixelPoint(50, 50); - break; + Position = screenBounds.TopLeft + new PixelPoint(Convert.ToInt32(screenBounds.Width * xOverride), Convert.ToInt32(screenBounds.Height * yOverride)); + } + else + { + switch (corner) + { + default: + case 0: + Position = screenBounds.BottomRight - new PixelPoint(Convert.ToInt32(scaleWidth) + 50, Convert.ToInt32(scaleHeight) + 50); + break; + case 1: + Position = screenBounds.BottomLeft - new PixelPoint(-50, Convert.ToInt32(scaleHeight) + 50); + break; + case 2: + Position = screenBounds.TopRight - new PixelPoint(Convert.ToInt32(scaleWidth) + 50, -50); + break; + case 3: + Position = screenBounds.TopLeft + new PixelPoint(50, 50); + break; + } } - - var timer = new System.Timers.Timer(); - timer.Elapsed += CloseNotification; - timer.Interval = Properties.Core.Default.NativeNotifyTimeout; - timer.Start(); -#if DEBUG - this.AttachDevTools(); -#endif } private void CloseNotification(object sender, System.Timers.ElapsedEventArgs e) From 330f30c5e7d93906d2be04070c8260b982cf8938 Mon Sep 17 00:00:00 2001 From: Xjph Date: Sat, 16 Oct 2021 11:59:42 -0230 Subject: [PATCH 3/5] fix: string settings not saving --- ObservatoryCore/UI/Views/BasicUIView.axaml.cs | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs index 3a01bfc..410f699 100644 --- a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs +++ b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs @@ -720,7 +720,9 @@ namespace Observatory.UI.Views gridPanel.RowDefinitions.Add(new RowDefinition() { Height = new GridLength(0, GridUnitType.Auto) }); gridPanel.AddControl(expander, nextRow, 0, 3); - foreach (var setting in displayedSettings.Where(s => !System.Attribute.IsDefined(s.Key, typeof(SettingIgnore)))) + var nonIgnoredSettings = displayedSettings.Where(s => !Attribute.IsDefined(s.Key, typeof(SettingIgnore))); + + foreach (var setting in nonIgnoredSettings) { if (setting.Key.PropertyType != typeof(bool) || settingsGrid.Children.Count % 2 == 0) { @@ -756,11 +758,13 @@ namespace Observatory.UI.Views TextBox textBox = new() { Text = stringSetting }; settingsGrid.AddControl(label, settingsGrid.RowDefinitions.Count - 1, 0); settingsGrid.AddControl(textBox, settingsGrid.RowDefinitions.Count - 1, 1); - textBox.TextInput += (object sender, Avalonia.Input.TextInputEventArgs e) => + + textBox.LostFocus += (object sender, RoutedEventArgs e) => { - setting.Key.SetValue(plugin.Settings, e.Text); + setting.Key.SetValue(plugin.Settings, ((TextBox)sender).Text); PluginManagement.PluginManager.GetInstance.SaveSettings(plugin, plugin.Settings); }; + break; case int intSetting: // We have two options for integer values: From 5edb32e8c7d8cc3c656c7336bb8b3de6e109252d Mon Sep 17 00:00:00 2001 From: Xjph Date: Sat, 16 Oct 2021 12:31:12 -0230 Subject: [PATCH 4/5] feat: save text fields on focus loss --- ObservatoryCore/UI/Views/BasicUIView.axaml.cs | 28 +++++++++++++++++-- 1 file changed, 26 insertions(+), 2 deletions(-) diff --git a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs index 410f699..6ec24b0 100644 --- a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs +++ b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs @@ -667,8 +667,15 @@ namespace Observatory.UI.Views Properties.Core.Default.Save(); } }); - - + }; + + journalPath.LostFocus += (object sender, RoutedEventArgs e) => + { + if (System.IO.Directory.Exists(journalPath.Text)) + { + Properties.Core.Default.JournalFolder = journalPath.Text; + Properties.Core.Default.Save(); + } }; @@ -858,6 +865,23 @@ namespace Observatory.UI.Views }; + settingPath.LostFocus += (object sender, RoutedEventArgs e) => + { + string fullPath; + + try + { + fullPath = System.IO.Path.GetFullPath(settingPath.Text); + } + catch + { + fullPath = string.Empty; + } + + setting.Key.SetValue(plugin.Settings, new System.IO.FileInfo(fullPath)); + PluginManagement.PluginManager.GetInstance.SaveSettings(plugin, plugin.Settings); + }; + StackPanel stackPanel = new() { Orientation = Avalonia.Layout.Orientation.Horizontal }; stackPanel.Children.Add(label); stackPanel.Children.Add(settingPath); From a2261a5c6ced4348b76d8135157ab0bb2daa0e3e Mon Sep 17 00:00:00 2001 From: Xjph Date: Mon, 18 Oct 2021 10:33:42 -0230 Subject: [PATCH 5/5] feat: add plugin folder button --- ObservatoryCore/UI/Views/BasicUIView.axaml.cs | 29 +++++++++++++++++-- 1 file changed, 27 insertions(+), 2 deletions(-) diff --git a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs index 6ec24b0..b23164f 100644 --- a/ObservatoryCore/UI/Views/BasicUIView.axaml.cs +++ b/ObservatoryCore/UI/Views/BasicUIView.axaml.cs @@ -14,6 +14,7 @@ using System.Collections.Generic; using Avalonia.Media; using Avalonia.Controls.ApplicationLifetimes; using System.Runtime.InteropServices; +using System.IO; namespace Observatory.UI.Views { @@ -140,7 +141,7 @@ namespace Observatory.UI.Views #region Native Settings #region Plugin List - DataGrid pluginList = new() { Margin = new Thickness(0, 20) }; + DataGrid pluginList = new() { Margin = new Thickness(0, 20, 0, 0) }; pluginList.Columns.Add(new DataGridTextColumn() { @@ -192,6 +193,31 @@ namespace Observatory.UI.Views pluginList.Items = uniquePlugins.Values; gridManager.AddSetting(pluginList); + Button pluginFolderButton = new() + { + Content = "Open Plugin Folder", + Height = 30, + Width = 150, + HorizontalAlignment = Avalonia.Layout.HorizontalAlignment.Right, + HorizontalContentAlignment = Avalonia.Layout.HorizontalAlignment.Center, + Margin = new Thickness(0, 0, 0, 20) + }; + + pluginFolderButton.Click += (object sender, RoutedEventArgs e) => + { + string pluginDir = AppDomain.CurrentDomain.BaseDirectory + "plugins"; + + if (!Directory.Exists(pluginDir)) + { + Directory.CreateDirectory(pluginDir); + } + + var fileExplorerInfo = new System.Diagnostics.ProcessStartInfo() { FileName = pluginDir, UseShellExecute = true }; + System.Diagnostics.Process.Start(fileExplorerInfo); + }; + + gridManager.AddSetting(pluginFolderButton); + #endregion #region Popup Notification settings @@ -678,7 +704,6 @@ namespace Observatory.UI.Views } }; - #endregion