Improve loading of widgets
This commit is contained in:
@@ -10,14 +10,13 @@ public class PrimaryWidget :
|
|||||||
{
|
{
|
||||||
args.Id = Guid.Parse("cfdfe07c-d9d6-4174-ae41-988ca24d2e10");
|
args.Id = Guid.Parse("cfdfe07c-d9d6-4174-ae41-988ca24d2e10");
|
||||||
args.Name = "Primary commands";
|
args.Name = "Primary commands";
|
||||||
|
}).ConfigureServices(services =>
|
||||||
}).ConfigureServices(args =>
|
|
||||||
{
|
{
|
||||||
args.AddCache<(Guid ParentId, Guid Id), PrimaryCommandConfiguration>()
|
services.AddCache<(Guid ParentId, Guid Id), PrimaryCommandConfiguration>()
|
||||||
.AddCache<Guid, IWidgetComponentViewModel>()
|
.AddCache<Guid, IWidgetComponentViewModel>()
|
||||||
.AddTransient<IProvider<PrimaryCommandConfiguration, IWidgetComponentViewModel?>, WidgetComponentProvider>()
|
.AddTransient<IProvider<PrimaryCommandConfiguration, IWidgetComponentViewModel?>, WidgetComponentProvider>()
|
||||||
.AddTransient<IFactory<PrimaryCommandConfiguration, IWidgetComponentViewModel?>, WidgetComponentFactory>()
|
.AddTransient<IFactory<PrimaryCommandConfiguration, IWidgetComponentViewModel?>, WidgetComponentFactory>()
|
||||||
.AddTransient<IEnumerator<IWidgetComponentViewModel>, WidgetComponentEnumerator>()
|
.AddTransient<IEnumerator<IWidgetComponentViewModel>, WidgetComponentEnumerationHandler>()
|
||||||
.AddWidgetTemplate<PrimaryWidgetViewModel>()
|
.AddWidgetTemplate<PrimaryWidgetViewModel>()
|
||||||
.AddHandler<PrimaryWidgetConfigurationHandler>();
|
.AddHandler<PrimaryWidgetConfigurationHandler>();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -20,61 +20,66 @@ public class PrimaryWidgetConfigurationHandler(IMediator mediator,
|
|||||||
(Guid currentParentId, List<PrimaryCommandConfiguration> currentConfigurations) = stack.Pop();
|
(Guid currentParentId, List<PrimaryCommandConfiguration> currentConfigurations) = stack.Pop();
|
||||||
foreach (PrimaryCommandConfiguration configuration in currentConfigurations)
|
foreach (PrimaryCommandConfiguration configuration in currentConfigurations)
|
||||||
{
|
{
|
||||||
items.Add(new KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration>((currentParentId, configuration.Id), configuration));
|
var key = (currentParentId, configuration.Id);
|
||||||
if (configuration.Commands is not null && configuration.Commands.Count > 0)
|
items.Add(new KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration>(key, configuration));
|
||||||
|
|
||||||
|
if (configuration.Commands?.Count > 0)
|
||||||
{
|
{
|
||||||
stack.Push((configuration.Id, configuration.Commands));
|
stack.Push((configuration.Id, configuration.Commands));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration> moved in
|
HashSet<Guid> cacheIds = new(cache.Select(x => x.Key.Id));
|
||||||
items.ExceptBy(cache.Select(x => new { x.Value.Order, x.Value.Id }), x => new { x.Value.Order, x.Value.Id }))
|
HashSet<Guid> itemIds = new(items.Select(x => x.Key.Id));
|
||||||
|
|
||||||
|
List<KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration>> movedItems =
|
||||||
|
items.ExceptBy(cache.Select(x => new { x.Value.Order, x.Value.Id }), x =>
|
||||||
|
new { x.Value.Order, x.Value.Id }).ToList();
|
||||||
|
|
||||||
|
foreach (KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration> moved in movedItems)
|
||||||
{
|
{
|
||||||
if (moved.Value is PrimaryCommandConfiguration configuration)
|
if (moved.Value is PrimaryCommandConfiguration configuration &&
|
||||||
|
provider.Get(configuration) is IWidgetComponentViewModel viewModel)
|
||||||
{
|
{
|
||||||
if (provider.Get(configuration) is IWidgetComponentViewModel viewModel)
|
await mediator.PublishAsync(
|
||||||
{
|
new Moved<IWidgetComponentViewModel>(configuration.Order, viewModel),
|
||||||
await mediator.PublishAsync(new Moved<IWidgetComponentViewModel>(configuration.Order, viewModel),
|
|
||||||
moved.Key.ParentId == Guid.Empty ? nameof(PrimaryWidgetViewModel) : moved.Key.ParentId,
|
moved.Key.ParentId == Guid.Empty ? nameof(PrimaryWidgetViewModel) : moved.Key.ParentId,
|
||||||
cancellationToken);
|
cancellationToken
|
||||||
|
);
|
||||||
|
|
||||||
cache.Remove(moved.Key);
|
cache.Remove(moved.Key);
|
||||||
cache.Add(moved.Key, moved.Value);
|
cache.Add(moved.Key, moved.Value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
List<KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration>> addedItems =
|
||||||
|
items.ExceptBy(cacheIds.Select(x => x), x => x.Key.Id).ToList();
|
||||||
|
|
||||||
|
foreach (KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration> added in addedItems)
|
||||||
|
|
||||||
// // cache.Add(added.Key, added.Value);
|
|
||||||
//}
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration> added in
|
|
||||||
items.ExceptBy(cache.Select(x => x.Key.Id), x => x.Key.Id))
|
|
||||||
{
|
{
|
||||||
if (added.Value is PrimaryCommandConfiguration configuration)
|
if (added.Value is PrimaryCommandConfiguration configuration &&
|
||||||
|
factory.Create(configuration) is IWidgetComponentViewModel viewModel)
|
||||||
{
|
{
|
||||||
if (factory.Create(configuration) is IWidgetComponentViewModel viewModel)
|
await mediator.PublishAsync(
|
||||||
{
|
new Inserted<IWidgetComponentViewModel>(configuration.Order, viewModel),
|
||||||
await mediator.PublishAsync(new Inserted<IWidgetComponentViewModel>(configuration.Order, viewModel),
|
|
||||||
added.Key.ParentId == Guid.Empty ? nameof(PrimaryWidgetViewModel) : added.Key.ParentId,
|
added.Key.ParentId == Guid.Empty ? nameof(PrimaryWidgetViewModel) : added.Key.ParentId,
|
||||||
cancellationToken);
|
cancellationToken);
|
||||||
}
|
|
||||||
|
|
||||||
cache.Add(added.Key, added.Value);
|
cache.Add(added.Key, added.Value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration> removed in
|
List<KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration>> removedItems =
|
||||||
cache.ExceptBy(items.Select(x => x.Key.Id), x => x.Key.Id))
|
cache.ExceptBy(itemIds.Select(x => x), x => x.Key.Id).ToList();
|
||||||
|
|
||||||
|
foreach (KeyValuePair<(Guid ParentId, Guid Id), PrimaryCommandConfiguration> removed in removedItems)
|
||||||
{
|
{
|
||||||
if (removed.Value is PrimaryCommandConfiguration configuration)
|
if (removed.Value is PrimaryCommandConfiguration configuration &&
|
||||||
|
provider.Get(configuration) is IWidgetComponentViewModel viewModel)
|
||||||
{
|
{
|
||||||
if (provider.Get(configuration) is IWidgetComponentViewModel viewModel)
|
await mediator.PublishAsync(
|
||||||
{
|
new Removed<IWidgetComponentViewModel>(viewModel),
|
||||||
await mediator.PublishAsync(new Removed<IWidgetComponentViewModel>(viewModel),
|
|
||||||
removed.Key.ParentId == Guid.Empty ? nameof(PrimaryWidgetViewModel) : removed.Key.ParentId,
|
removed.Key.ParentId == Guid.Empty ? nameof(PrimaryWidgetViewModel) : removed.Key.ParentId,
|
||||||
cancellationToken);
|
cancellationToken);
|
||||||
|
|
||||||
@@ -82,5 +87,4 @@ public class PrimaryWidgetConfigurationHandler(IMediator mediator,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
+2
-2
@@ -1,11 +1,11 @@
|
|||||||
namespace Hyperbar.Windows.Primary;
|
namespace Hyperbar.Windows.Primary;
|
||||||
|
|
||||||
public class WidgetComponentEnumerator(PrimaryWidgetConfiguration configuration,
|
public class WidgetComponentEnumerationHandler(PrimaryWidgetConfiguration configuration,
|
||||||
IFactory<PrimaryCommandConfiguration, IWidgetComponentViewModel?> factory,
|
IFactory<PrimaryCommandConfiguration, IWidgetComponentViewModel?> factory,
|
||||||
ICache<(Guid ParentId, Guid Id), PrimaryCommandConfiguration> cache) :
|
ICache<(Guid ParentId, Guid Id), PrimaryCommandConfiguration> cache) :
|
||||||
IEnumerator<IWidgetComponentViewModel>
|
IEnumerator<IWidgetComponentViewModel>
|
||||||
{
|
{
|
||||||
public IEnumerable<IWidgetComponentViewModel?> Next()
|
public IEnumerable<IWidgetComponentViewModel?> Get()
|
||||||
{
|
{
|
||||||
Stack<(Guid, List<PrimaryCommandConfiguration>)> stack = new();
|
Stack<(Guid, List<PrimaryCommandConfiguration>)> stack = new();
|
||||||
stack.Push((Guid.Empty, configuration.Commands));
|
stack.Push((Guid.Empty, configuration.Commands));
|
||||||
@@ -1,10 +1,12 @@
|
|||||||
using Hyperbar.Windows.Controls;
|
using Hyperbar.Windows.Controls;
|
||||||
|
using Hyperbar.Windows.Interop;
|
||||||
using Hyperbar.Windows.UI;
|
using Hyperbar.Windows.UI;
|
||||||
using Microsoft.Extensions.Configuration;
|
using Microsoft.Extensions.Configuration;
|
||||||
using Microsoft.Extensions.Configuration.Json;
|
using Microsoft.Extensions.Configuration.Json;
|
||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Hosting;
|
using Microsoft.Extensions.Hosting;
|
||||||
using Microsoft.UI.Xaml;
|
using Microsoft.UI.Xaml;
|
||||||
|
using System;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
|
|
||||||
namespace Hyperbar.Windows;
|
namespace Hyperbar.Windows;
|
||||||
@@ -27,24 +29,40 @@ public partial class App :
|
|||||||
})
|
})
|
||||||
.ConfigureServices((context, services) =>
|
.ConfigureServices((context, services) =>
|
||||||
{
|
{
|
||||||
services.AddSingleton<IServiceFactory>(provider =>
|
services.AddDefault();
|
||||||
new ServiceFactory((type, parameters) => ActivatorUtilities.CreateInstance(provider, type, parameters!)));
|
|
||||||
|
|
||||||
services.AddSingleton<IMediator, Mediator>();
|
|
||||||
services.AddSingleton<IDisposer, Disposer>();
|
|
||||||
services.AddSingleton<IDispatcher, Dispatcher>();
|
|
||||||
|
|
||||||
services.AddHostedService<AppService>();
|
services.AddHostedService<AppService>();
|
||||||
services.AddConfiguration<AppConfiguration>(args => { args.Placement = DesktopBarPlacemenet.Top; });
|
|
||||||
|
|
||||||
services.AddTransient<IInitializer, AppInitializer>();
|
services.AddSingleton<IDispatcher, Dispatcher>();
|
||||||
services.AddTransient<ITemplateFactory, TemplateFactory>();
|
services.AddTransient<ITemplateFactory, TemplateFactory>();
|
||||||
|
services.AddTransient<IInitializer, AppInitializer>();
|
||||||
services.AddSingleton<DesktopBar>();
|
|
||||||
|
|
||||||
services.AddContentTemplate<WidgetBarViewModel, WidgetBarView>();
|
|
||||||
|
|
||||||
services.AddHandler<AppConfigurationChangedHandler>();
|
services.AddHandler<AppConfigurationChangedHandler>();
|
||||||
|
services.AddConfiguration<AppConfiguration>(args =>
|
||||||
|
{
|
||||||
|
args.Placement = DesktopBarPlacemenet.Top;
|
||||||
|
});
|
||||||
|
|
||||||
|
services.AddSingleton<DesktopBar>();
|
||||||
|
services.AddContentTemplate<WidgetBarViewModel, WidgetBarView>();
|
||||||
|
|
||||||
|
services.AddTransient<IWidgetServiceCollection>(provider =>
|
||||||
|
new WidgetServiceCollection(services =>
|
||||||
|
{
|
||||||
|
services.AddSingleton<IDispatcher, Dispatcher>();
|
||||||
|
services.AddTransient<ITemplateFactory, TemplateFactory>();
|
||||||
|
|
||||||
|
services.AddScoped<IVirtualKeyboard, VirtualKeyboard>();
|
||||||
|
|
||||||
|
services.AddHandler<KeyAcceleratorHandler>();
|
||||||
|
services.AddHandler<StartProcessHandler>();
|
||||||
|
|
||||||
|
services.AddTransient<IWidgetView, WidgetView>();
|
||||||
|
|
||||||
|
services.AddContentTemplate<WidgetContainerViewModel, WidgetContainerView>();
|
||||||
|
services.AddContentTemplate<WidgetButtonViewModel, WidgetButtonView>();
|
||||||
|
services.AddContentTemplate<WidgetSplitButtonViewModel, WidgetSplitButtonView>();
|
||||||
|
}));
|
||||||
|
|
||||||
//services.AddWidget<MediaControllerWidgetBuilder>();
|
//services.AddWidget<MediaControllerWidgetBuilder>();
|
||||||
//services.AddWidget<PrimaryWidget>();
|
//services.AddWidget<PrimaryWidget>();
|
||||||
|
|||||||
@@ -3,59 +3,57 @@ using Hyperbar.Windows.UI;
|
|||||||
using Microsoft.Extensions.Configuration;
|
using Microsoft.Extensions.Configuration;
|
||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Hosting;
|
using Microsoft.Extensions.Hosting;
|
||||||
using Microsoft.UI.Dispatching;
|
|
||||||
|
|
||||||
namespace Hyperbar.Windows
|
namespace Hyperbar.Windows
|
||||||
{
|
{
|
||||||
public static class IServiceCollectionExtensions
|
//public static class IServiceCollectionExtensions
|
||||||
{
|
//{
|
||||||
public static IServiceCollection AddWidget<TWidget>(this IServiceCollection services)
|
// public static IServiceCollection AddWidget<TWidget>(this IServiceCollection services)
|
||||||
where TWidget :
|
// where TWidget :
|
||||||
IWidget,
|
// IWidget,
|
||||||
new()
|
// new()
|
||||||
{
|
// {
|
||||||
IHost? host = new HostBuilder()
|
// IHost? host = new HostBuilder()
|
||||||
.UseContentRoot(AppContext.BaseDirectory)
|
// .UseContentRoot(AppContext.BaseDirectory)
|
||||||
.ConfigureAppConfiguration(config =>
|
// .ConfigureAppConfiguration(config =>
|
||||||
{
|
// {
|
||||||
config.SetBasePath(AppContext.BaseDirectory);
|
// config.SetBasePath(AppContext.BaseDirectory);
|
||||||
config.AddJsonFile("Settings.json", true, true);
|
// config.AddJsonFile("Settings.json", true, true);
|
||||||
|
// })
|
||||||
|
// .ConfigureServices((context, isolatedServices) =>
|
||||||
|
// {
|
||||||
|
// isolatedServices.AddScoped<IServiceFactory>(provider =>
|
||||||
|
// new ServiceFactory((type, parameters) => ActivatorUtilities.CreateInstance(provider, type, parameters!)));
|
||||||
|
|
||||||
config.Build();
|
// isolatedServices.AddHostedService<WidgetService>();
|
||||||
})
|
// isolatedServices.AddTransient<ITemplateFactory, TemplateFactory>();
|
||||||
.ConfigureServices((context, isolatedServices) =>
|
|
||||||
{
|
|
||||||
isolatedServices.AddScoped<IServiceFactory>(provider =>
|
|
||||||
new ServiceFactory((type, parameters) => ActivatorUtilities.CreateInstance(provider, type, parameters!)));
|
|
||||||
|
|
||||||
isolatedServices.AddHostedService<WidgetService>();
|
// isolatedServices.AddScoped<IMediator, Mediator>();
|
||||||
isolatedServices.AddTransient<ITemplateFactory, TemplateFactory>();
|
// isolatedServices.AddScoped<IDisposer, Disposer>();
|
||||||
|
// isolatedServices.AddSingleton<IDispatcher, Dispatcher>();
|
||||||
|
|
||||||
isolatedServices.AddScoped<IMediator, Mediator>();
|
// isolatedServices.AddScoped<IVirtualKeyboard, VirtualKeyboard>();
|
||||||
isolatedServices.AddScoped<IDisposer, Disposer>();
|
|
||||||
isolatedServices.AddSingleton<IDispatcher, Dispatcher>();
|
|
||||||
|
|
||||||
isolatedServices.AddScoped<IVirtualKeyboard, VirtualKeyboard>();
|
// isolatedServices.AddHandler<KeyAcceleratorHandler>();
|
||||||
|
// isolatedServices.AddHandler<StartProcessHandler>();
|
||||||
|
|
||||||
isolatedServices.AddHandler<KeyAcceleratorHandler>();
|
// isolatedServices.AddTransient<IWidgetView, WidgetView>();
|
||||||
isolatedServices.AddHandler<StartProcessHandler>();
|
|
||||||
|
|
||||||
isolatedServices.AddTransient<IWidgetView, WidgetView>();
|
// isolatedServices.AddContentTemplate<WidgetContainerViewModel, WidgetContainerView>();
|
||||||
|
// isolatedServices.AddContentTemplate<WidgetButtonViewModel, WidgetButtonView>();
|
||||||
|
// isolatedServices.AddContentTemplate<WidgetSplitButtonViewModel, WidgetSplitButtonView>();
|
||||||
|
|
||||||
isolatedServices.AddContentTemplate<WidgetContainerViewModel, WidgetContainerView>();
|
// TWidget widget = new();
|
||||||
isolatedServices.AddContentTemplate<WidgetButtonViewModel, WidgetButtonView>();
|
// IWidgetBuilder builder = widget.Create();
|
||||||
isolatedServices.AddContentTemplate<WidgetSplitButtonViewModel, WidgetSplitButtonView>();
|
|
||||||
|
|
||||||
TWidget widget = new();
|
// isolatedServices.AddRange(builder.Services);
|
||||||
IWidgetBuilder builder = widget.Create();
|
|
||||||
isolatedServices.AddRange(builder.Services);
|
|
||||||
|
|
||||||
}).Build();
|
// }).Build();
|
||||||
|
|
||||||
services.AddTransient(provider => new WidgetContext(host.Services));
|
// services.AddTransient(provider => new WidgetContext(host.Services));
|
||||||
|
|
||||||
host.Start();
|
// host.Start();
|
||||||
return services;
|
// return services;
|
||||||
}
|
// }
|
||||||
}
|
//}
|
||||||
}
|
}
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
public class Configuration<TConfiguration>(IConfigurationReader<TConfiguration> reader) :
|
public class Configuration<TConfiguration>(IConfigurationReader<TConfiguration> reader) :
|
||||||
IConfiguration<TConfiguration>
|
IConfiguration<TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
public TConfiguration Value => reader.Read();
|
public TConfiguration Value => reader.Read();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,9 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class ConfigurationFactory<TConfiguration>(Func<TConfiguration> factory) :
|
||||||
|
IConfigurationFactory<TConfiguration>
|
||||||
|
where TConfiguration :
|
||||||
|
class
|
||||||
|
{
|
||||||
|
public TConfiguration Create() => factory.Invoke();
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
using Microsoft.Extensions.FileProviders;
|
||||||
|
|
||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class ConfigurationFile<TConfiguration>(IFileInfo fileInfo) :
|
||||||
|
IConfigurationFile<TConfiguration>
|
||||||
|
where TConfiguration :
|
||||||
|
class
|
||||||
|
{
|
||||||
|
public IFileInfo FileInfo => fileInfo;
|
||||||
|
}
|
||||||
@@ -1,23 +1,13 @@
|
|||||||
namespace Hyperbar;
|
namespace Hyperbar;
|
||||||
|
|
||||||
public class ConfigurationInitializer<TConfiguration>(DefaultConfiguration<TConfiguration> defaults,
|
public class ConfigurationInitializer<TConfiguration>(IConfigurationReader<TConfiguration> reader) :
|
||||||
IConfigurationReader<TConfiguration> reader,
|
|
||||||
IConfigurationWriter<TConfiguration> writer) :
|
|
||||||
IInitializer
|
IInitializer
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class,
|
class
|
||||||
new()
|
|
||||||
{
|
{
|
||||||
public Task InitializeAsync()
|
public Task InitializeAsync()
|
||||||
{
|
{
|
||||||
if (!reader.TryRead(out TConfiguration? _))
|
reader.Read();
|
||||||
{
|
|
||||||
if (defaults.Configuration is not null)
|
|
||||||
{
|
|
||||||
writer.Write(defaults.Configuration);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Task.CompletedTask;
|
return Task.CompletedTask;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ public class ConfigurationMonitor<TConfiguration>(IConfigurationFile<TConfigurat
|
|||||||
IConfigurationReader<TConfiguration> reader,
|
IConfigurationReader<TConfiguration> reader,
|
||||||
IMediator mediator) : IInitializer
|
IMediator mediator) : IInitializer
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
private FileSystemWatcher? watcher;
|
private FileSystemWatcher? watcher;
|
||||||
|
|
||||||
|
|||||||
@@ -1,19 +1,19 @@
|
|||||||
namespace Hyperbar;
|
namespace Hyperbar;
|
||||||
|
|
||||||
public class ConfigurationReader<TConfiguration>(IConfigurationSource<TConfiguration> source) :
|
public class ConfigurationReader<TConfiguration>(IConfigurationSource<TConfiguration> source,
|
||||||
|
IConfigurationFactory<TConfiguration> factory) :
|
||||||
IConfigurationReader<TConfiguration>
|
IConfigurationReader<TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
public TConfiguration Read()
|
public TConfiguration Read()
|
||||||
{
|
{
|
||||||
if ((source.TryGet(out TConfiguration? value) ? value :
|
if ((source.TryGet(out TConfiguration? value) ? value : factory.Create()) is TConfiguration configuration)
|
||||||
new TConfiguration()) is TConfiguration configuration)
|
|
||||||
{
|
{
|
||||||
return configuration;
|
return configuration;
|
||||||
}
|
}
|
||||||
|
|
||||||
return new TConfiguration();
|
return factory.Create();
|
||||||
}
|
}
|
||||||
|
|
||||||
public bool TryRead(out TConfiguration? configuration)
|
public bool TryRead(out TConfiguration? configuration)
|
||||||
|
|||||||
@@ -1,13 +1,15 @@
|
|||||||
namespace Hyperbar;
|
namespace Hyperbar;
|
||||||
|
|
||||||
public class ConfigurationWriter<TConfiguration>(IConfigurationSource<TConfiguration> source) :
|
public class ConfigurationWriter<TConfiguration>(IConfigurationSource<TConfiguration> source,
|
||||||
|
IConfigurationFactory<TConfiguration> factory) :
|
||||||
IConfigurationWriter<TConfiguration>
|
IConfigurationWriter<TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
public void Write(Action<TConfiguration> updateDelegate)
|
public void Write(Action<TConfiguration> updateDelegate)
|
||||||
{
|
{
|
||||||
if ((source.TryGet(out TConfiguration? value) ? value : new TConfiguration()) is TConfiguration updatedValue)
|
if ((source.TryGet(out TConfiguration? value) ? value :
|
||||||
|
factory.Create()) is TConfiguration updatedValue)
|
||||||
{
|
{
|
||||||
updateDelegate?.Invoke(updatedValue);
|
updateDelegate?.Invoke(updatedValue);
|
||||||
Write(updatedValue);
|
Write(updatedValue);
|
||||||
|
|||||||
@@ -1,8 +0,0 @@
|
|||||||
namespace Hyperbar;
|
|
||||||
|
|
||||||
public class DefaultConfiguration<TConfiguration>(TConfiguration? configuration = null)
|
|
||||||
where TConfiguration :
|
|
||||||
class
|
|
||||||
{
|
|
||||||
public TConfiguration? Configuration => configuration;
|
|
||||||
}
|
|
||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
public interface IConfiguration<out TConfiguration>
|
public interface IConfiguration<out TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
TConfiguration Value { get; }
|
TConfiguration Value { get; }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
namespace Hyperbar
|
||||||
|
{
|
||||||
|
public interface IConfigurationFactory<TConfiguration> where TConfiguration : class
|
||||||
|
{
|
||||||
|
TConfiguration Create();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
using Microsoft.Extensions.FileProviders;
|
||||||
|
|
||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public interface IConfigurationFile<TConfiguration>
|
||||||
|
where TConfiguration :
|
||||||
|
class
|
||||||
|
{
|
||||||
|
IFileInfo FileInfo { get; }
|
||||||
|
}
|
||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
public interface IConfigurationReader<TConfiguration>
|
public interface IConfigurationReader<TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
bool TryRead(out TConfiguration? configuration);
|
bool TryRead(out TConfiguration? configuration);
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
using Microsoft.Extensions.FileProviders;
|
namespace Hyperbar;
|
||||||
|
|
||||||
namespace Hyperbar;
|
|
||||||
|
|
||||||
public interface IConfigurationSource<TConfiguration>
|
public interface IConfigurationSource<TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
public interface IConfigurationWriter<TConfiguration>
|
public interface IConfigurationWriter<TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
void Write(Action<TConfiguration> updateDelegate);
|
void Write(Action<TConfiguration> updateDelegate);
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
public interface IWritableConfiguration<out TConfiguration>
|
public interface IWritableConfiguration<out TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
void Write(Action<TConfiguration> updateDelegate);
|
void Write(Action<TConfiguration> updateDelegate);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
public class WritableConfiguration<TConfiguration>(IConfigurationWriter<TConfiguration> writer) :
|
public class WritableConfiguration<TConfiguration>(IConfigurationWriter<TConfiguration> writer) :
|
||||||
IWritableConfiguration<TConfiguration>
|
IWritableConfiguration<TConfiguration>
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
public void Write(Action<TConfiguration> updateDelegate) => writer.Write(updateDelegate);
|
public void Write(Action<TConfiguration> updateDelegate) => writer.Write(updateDelegate);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,23 +7,26 @@ using System.Text.Json;
|
|||||||
|
|
||||||
namespace Hyperbar;
|
namespace Hyperbar;
|
||||||
|
|
||||||
public interface IConfigurationFile<TConfiguration>
|
|
||||||
where TConfiguration :
|
|
||||||
class
|
|
||||||
{
|
|
||||||
IFileInfo FileInfo { get; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public class ConfigurationFile<TConfiguration>(IFileInfo fileInfo) :
|
|
||||||
IConfigurationFile<TConfiguration>
|
|
||||||
where TConfiguration :
|
|
||||||
class
|
|
||||||
{
|
|
||||||
public IFileInfo FileInfo => fileInfo;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class IServiceCollectionExtensions
|
public static class IServiceCollectionExtensions
|
||||||
{
|
{
|
||||||
|
public static IServiceCollection AddDefault(this IServiceCollection services)
|
||||||
|
{
|
||||||
|
services.AddSingleton<IServiceFactory>(provider =>
|
||||||
|
new ServiceFactory((type, parameters) => ActivatorUtilities.CreateInstance(provider, type, parameters!)));
|
||||||
|
|
||||||
|
services.AddSingleton<IMediator, Mediator>();
|
||||||
|
services.AddSingleton<IDisposer, Disposer>();
|
||||||
|
|
||||||
|
services.AddTransient<IInitializer, WidgetInitializer>();
|
||||||
|
services.AddTransient<IFactory<Type, IWidget>, WidgetFactory>();
|
||||||
|
|
||||||
|
services.AddHandler<WidgetEnumerationHandler>();
|
||||||
|
services.AddHandler<WidgetAssemblyHandler>();
|
||||||
|
services.AddHandler<WidgetHandler>();
|
||||||
|
|
||||||
|
return services;
|
||||||
|
}
|
||||||
|
|
||||||
public static IServiceCollection AddCache<TKey, TValue>(this IServiceCollection services)
|
public static IServiceCollection AddCache<TKey, TValue>(this IServiceCollection services)
|
||||||
where TKey :
|
where TKey :
|
||||||
notnull
|
notnull
|
||||||
@@ -46,38 +49,45 @@ public static class IServiceCollectionExtensions
|
|||||||
|
|
||||||
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services)
|
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services)
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class => services.AddConfiguration<TConfiguration>(typeof(TConfiguration).Name,
|
||||||
{
|
"Settings.json",
|
||||||
return services.AddConfiguration<TConfiguration>(typeof(TConfiguration).Name, "Settings.json", null);
|
null);
|
||||||
}
|
|
||||||
|
|
||||||
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services,
|
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services,
|
||||||
Action<TConfiguration> configurationDelegate)
|
Action<TConfiguration> configurationDelegate)
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class,
|
||||||
|
new()
|
||||||
{
|
{
|
||||||
TConfiguration configuration = new();
|
TConfiguration configuration = new();
|
||||||
configurationDelegate.Invoke(configuration);
|
configurationDelegate.Invoke(configuration);
|
||||||
|
|
||||||
return services.AddConfiguration<TConfiguration>(typeof(TConfiguration).Name, "Settings.json",
|
return services.AddConfiguration<TConfiguration>(typeof(TConfiguration).Name,
|
||||||
|
"Settings.json",
|
||||||
configuration);
|
configuration);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services,
|
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services,
|
||||||
TConfiguration? defaults = null)
|
TConfiguration configuration)
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class => services.AddConfiguration<TConfiguration>(configuration.GetType().Name,
|
||||||
{
|
"Settings.json",
|
||||||
return services.AddConfiguration(typeof(TConfiguration).Name, "Settings.json", defaults);
|
configuration);
|
||||||
}
|
|
||||||
|
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services,
|
||||||
|
object configuration)
|
||||||
|
where TConfiguration :
|
||||||
|
class => services.AddConfiguration<TConfiguration>(configuration.GetType().Name,
|
||||||
|
"Settings.json",
|
||||||
|
(TConfiguration?)configuration);
|
||||||
|
|
||||||
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services,
|
public static IServiceCollection AddConfiguration<TConfiguration>(this IServiceCollection services,
|
||||||
string section,
|
string section,
|
||||||
string path = "Settings.json",
|
string path = "Settings.json",
|
||||||
TConfiguration? defaults = null,
|
object? configuration = null,
|
||||||
Action<JsonSerializerOptions>? serializerDelegate = null)
|
Action<JsonSerializerOptions>? serializerDelegate = null)
|
||||||
where TConfiguration :
|
where TConfiguration :
|
||||||
class, new()
|
class
|
||||||
{
|
{
|
||||||
services.AddSingleton<IConfigurationSource<TConfiguration>>(provider =>
|
services.AddSingleton<IConfigurationSource<TConfiguration>>(provider =>
|
||||||
{
|
{
|
||||||
@@ -109,7 +119,17 @@ public static class IServiceCollectionExtensions
|
|||||||
services.AddSingleton<IConfigurationReader<TConfiguration>, ConfigurationReader<TConfiguration>>();
|
services.AddSingleton<IConfigurationReader<TConfiguration>, ConfigurationReader<TConfiguration>>();
|
||||||
services.AddSingleton<IConfigurationWriter<TConfiguration>, ConfigurationWriter<TConfiguration>>();
|
services.AddSingleton<IConfigurationWriter<TConfiguration>, ConfigurationWriter<TConfiguration>>();
|
||||||
|
|
||||||
services.AddTransient(provider => new DefaultConfiguration<TConfiguration>(defaults));
|
if (configuration is not null)
|
||||||
|
{
|
||||||
|
services.AddTransient(typeof(TConfiguration), provider => configuration);
|
||||||
|
}
|
||||||
|
|
||||||
|
services.AddTransient<IConfigurationFactory<TConfiguration>>(provider => new ConfigurationFactory<TConfiguration>(() =>
|
||||||
|
{
|
||||||
|
var fo = configuration ?? provider.GetRequiredService<TConfiguration>();
|
||||||
|
|
||||||
|
return (TConfiguration)fo;
|
||||||
|
}));
|
||||||
services.AddTransient<IInitializer, ConfigurationInitializer<TConfiguration>>();
|
services.AddTransient<IInitializer, ConfigurationInitializer<TConfiguration>>();
|
||||||
|
|
||||||
services.AddTransient<IWritableConfiguration<TConfiguration>, WritableConfiguration<TConfiguration>>();
|
services.AddTransient<IWritableConfiguration<TConfiguration>, WritableConfiguration<TConfiguration>>();
|
||||||
@@ -134,7 +154,12 @@ public static class IServiceCollectionExtensions
|
|||||||
services.AddKeyedTransient(contentType, key);
|
services.AddKeyedTransient(contentType, key);
|
||||||
services.AddKeyedTransient(templateType, key);
|
services.AddKeyedTransient(templateType, key);
|
||||||
|
|
||||||
services.AddTransient<IContentTemplateDescriptor>(provider => new ContentTemplateDescriptor { ContentType = contentType, TemplateType = templateType, Key = key });
|
services.AddTransient<IContentTemplateDescriptor>(provider => new ContentTemplateDescriptor
|
||||||
|
{
|
||||||
|
ContentType = contentType,
|
||||||
|
TemplateType = templateType,
|
||||||
|
Key = key
|
||||||
|
});
|
||||||
|
|
||||||
return services;
|
return services;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
namespace Hyperbar;
|
namespace Hyperbar;
|
||||||
|
|
||||||
public record Changed<TValue>(TValue? Value = default) : INotification;
|
public record Changed<TValue>(TValue? Value = default) : INotification;
|
||||||
|
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public record Created<TValue>(TValue Value, object? Target = null) : INotification;
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public record Enumerate<TValue> : INotification;
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
namespace Hyperbar;
|
|
||||||
|
|
||||||
public record Created<TValue>(TValue Value, object Target) : INotification
|
|
||||||
{
|
|
||||||
public static Created<TValue> For<TTarget>(TValue value)
|
|
||||||
{
|
|
||||||
return new Created<TValue>(value, typeof(TTarget).Name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -2,5 +2,5 @@
|
|||||||
|
|
||||||
public interface IEnumerator<TItem>
|
public interface IEnumerator<TItem>
|
||||||
{
|
{
|
||||||
IEnumerable<TItem?> Next();
|
IEnumerable<TItem?> Get();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -50,7 +50,7 @@ public partial class ObservableCollectionViewModel<TItem> :
|
|||||||
|
|
||||||
if (enumerator is not null)
|
if (enumerator is not null)
|
||||||
{
|
{
|
||||||
foreach (TItem? item in enumerator.Next())
|
foreach (TItem? item in enumerator.Get())
|
||||||
{
|
{
|
||||||
if (item is not null)
|
if (item is not null)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
namespace Hyperbar
|
|
||||||
{
|
|
||||||
public interface IWidgetAssemblyLoader
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -4,7 +4,7 @@ namespace Hyperbar;
|
|||||||
|
|
||||||
public interface IWidgetBuilder
|
public interface IWidgetBuilder
|
||||||
{
|
{
|
||||||
WidgetConfiguration Configuration { get; }
|
IWidgetBuilder ConfigureServices(Action<IServiceCollection> configureDelegate);
|
||||||
|
|
||||||
IServiceCollection Services { get; }
|
IWidgetHost Build();
|
||||||
}
|
}
|
||||||
@@ -1,35 +1,13 @@
|
|||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using System.Reflection;
|
|
||||||
using System.Runtime.Loader;
|
|
||||||
|
|
||||||
namespace Hyperbar;
|
namespace Hyperbar;
|
||||||
|
|
||||||
public class AssemblyProvider
|
|
||||||
{
|
|
||||||
public static void Load()
|
|
||||||
{
|
|
||||||
string extensionsPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Extensions");
|
|
||||||
Directory.CreateDirectory(extensionsPath);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static IEnumerable<Assembly> Get(string path)
|
|
||||||
{
|
|
||||||
string extensionsPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Extensions");
|
|
||||||
|
|
||||||
Directory.CreateDirectory(extensionsPath);
|
|
||||||
foreach (string assemblyPath in Directory.GetFiles(extensionsPath, "*.dll", SearchOption.AllDirectories))
|
|
||||||
{
|
|
||||||
yield return AssemblyLoadContext.Default.LoadFromAssemblyPath(assemblyPath);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class IWidgetBuilderExtensions
|
public static class IWidgetBuilderExtensions
|
||||||
{
|
{
|
||||||
public static IWidgetBuilder ConfigureServices(this IWidgetBuilder builder,
|
//public static IWidgetBuilder ConfigureServices(this IWidgetBuilder builder,
|
||||||
Action<IServiceCollection> servicesDelegate)
|
// Action<IServiceCollection> servicesDelegate)
|
||||||
{
|
//{
|
||||||
servicesDelegate(builder.Services);
|
// servicesDelegate(builder.Services);
|
||||||
return builder;
|
// return builder;
|
||||||
}
|
//}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,6 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public interface IWidgetHost
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public interface IWidgetServiceBuilder
|
||||||
|
{
|
||||||
|
void ConfigureWidgetServices(IWidgetServiceCollection widgetServices);
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
|
||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public interface IWidgetServiceCollection
|
||||||
|
{
|
||||||
|
IServiceCollection Services { get; }
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
using System.Reflection;
|
||||||
|
|
||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public record WidgetAssembly(Assembly? Assembly = default) :
|
||||||
|
INotification;
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
using System.Reflection;
|
||||||
|
|
||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class WidgetAssemblyHandler(IMediator mediator,
|
||||||
|
IFactory<Type, IWidget> factory) :
|
||||||
|
INotificationHandler<Created<Assembly>>
|
||||||
|
{
|
||||||
|
public Task Handle(Created<Assembly> notification,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
if (notification.Value?.GetTypes().FirstOrDefault(x => typeof(IWidget).IsAssignableFrom(x)) is Type widgetType)
|
||||||
|
{
|
||||||
|
if (factory.Create(widgetType) is IWidget widget)
|
||||||
|
{
|
||||||
|
mediator.PublishAsync(new Created<IWidget>(widget),
|
||||||
|
cancellationToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,17 +1,42 @@
|
|||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.Configuration;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
using Microsoft.Extensions.Hosting;
|
||||||
|
using System.Reflection;
|
||||||
|
|
||||||
namespace Hyperbar;
|
namespace Hyperbar;
|
||||||
|
|
||||||
public class WidgetBuilder(WidgetConfiguration configuration) :
|
public class WidgetBuilder :
|
||||||
IWidgetBuilder
|
IWidgetBuilder,
|
||||||
|
IWidgetServiceBuilder
|
||||||
{
|
{
|
||||||
public IServiceCollection Services => new ServiceCollection();
|
private readonly IHostBuilder hostBuilder;
|
||||||
|
|
||||||
public WidgetConfiguration Configuration => configuration;
|
public WidgetBuilder(IWidgetConfiguration configuration)
|
||||||
|
{
|
||||||
|
hostBuilder = new HostBuilder()
|
||||||
|
.UseContentRoot(Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData),
|
||||||
|
Assembly.GetEntryAssembly()?.GetName().Name!), true)
|
||||||
|
.ConfigureAppConfiguration(config =>
|
||||||
|
{
|
||||||
|
config.AddJsonFile("Settings.json", true, true);
|
||||||
|
})
|
||||||
|
.ConfigureServices((context, services) =>
|
||||||
|
{
|
||||||
|
services.AddScoped<IServiceFactory>(provider =>
|
||||||
|
new ServiceFactory((type, parameters) =>
|
||||||
|
ActivatorUtilities.CreateInstance(provider, type, parameters!)));
|
||||||
|
|
||||||
|
services.AddHostedService<WidgetService>();
|
||||||
|
|
||||||
|
services.AddScoped<IMediator, Mediator>();
|
||||||
|
services.AddScoped<IDisposer, Disposer>();
|
||||||
|
services.AddConfiguration<IWidgetConfiguration>(configuration);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
public static IWidgetBuilder Configure<TWidgetConfiguration>(Action<TWidgetConfiguration> configurationDelegate)
|
public static IWidgetBuilder Configure<TWidgetConfiguration>(Action<TWidgetConfiguration> configurationDelegate)
|
||||||
where TWidgetConfiguration :
|
where TWidgetConfiguration :
|
||||||
WidgetConfiguration,
|
IWidgetConfiguration,
|
||||||
new()
|
new()
|
||||||
{
|
{
|
||||||
TWidgetConfiguration configuration = new();
|
TWidgetConfiguration configuration = new();
|
||||||
@@ -19,4 +44,19 @@ public class WidgetBuilder(WidgetConfiguration configuration) :
|
|||||||
|
|
||||||
return new WidgetBuilder(configuration);
|
return new WidgetBuilder(configuration);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public IWidgetHost Build()
|
||||||
|
{
|
||||||
|
IHost host = hostBuilder.Build();
|
||||||
|
return new WidgetHost(host);
|
||||||
|
}
|
||||||
|
|
||||||
|
public IWidgetBuilder ConfigureServices(Action<IServiceCollection> configureDelegate)
|
||||||
|
{
|
||||||
|
hostBuilder.ConfigureServices(configureDelegate.Invoke);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ConfigureWidgetServices(IWidgetServiceCollection widgetServices) =>
|
||||||
|
hostBuilder.ConfigureServices(services => services.AddRange(widgetServices.Services));
|
||||||
}
|
}
|
||||||
@@ -1,6 +1,16 @@
|
|||||||
namespace Hyperbar;
|
namespace Hyperbar;
|
||||||
|
|
||||||
public abstract class WidgetConfiguration
|
public interface IWidgetConfiguration
|
||||||
|
{
|
||||||
|
Guid Id { get; set; }
|
||||||
|
|
||||||
|
string? Name { get; set; }
|
||||||
|
|
||||||
|
string? Description { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract class WidgetConfiguration :
|
||||||
|
IWidgetConfiguration
|
||||||
{
|
{
|
||||||
public Guid Id { get; set; } = Guid.NewGuid();
|
public Guid Id { get; set; } = Guid.NewGuid();
|
||||||
|
|
||||||
|
|||||||
@@ -1,30 +1,21 @@
|
|||||||
namespace Hyperbar;
|
using Microsoft.Extensions.Hosting;
|
||||||
|
|
||||||
public class WidgetContext(IServiceProvider serviceProvider) :
|
namespace Hyperbar;
|
||||||
IInitializer
|
public class WidgetHost :
|
||||||
|
IWidgetHost
|
||||||
{
|
{
|
||||||
public IServiceProvider ServiceProvider => serviceProvider;
|
public WidgetHost(IHost host)
|
||||||
|
|
||||||
public Task InitializeAsync()
|
|
||||||
{
|
{
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
public void Start()
|
||||||
public class WidgetMonitor :
|
{
|
||||||
IInitializer
|
|
||||||
{
|
}
|
||||||
public Task InitializeAsync()
|
|
||||||
{
|
public void Stop()
|
||||||
throw new NotImplementedException();
|
{
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class WidgetManager :
|
|
||||||
IInitializer
|
|
||||||
{
|
|
||||||
public Task InitializeAsync()
|
|
||||||
{
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
using Microsoft.Extensions.Hosting;
|
||||||
|
using System.Reflection;
|
||||||
|
using System.Runtime.Loader;
|
||||||
|
|
||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class WidgetEnumerationHandler(IHostEnvironment hostEnvironment,
|
||||||
|
IMediator mediator) :
|
||||||
|
INotificationHandler<Enumerate<IWidget>>
|
||||||
|
{
|
||||||
|
public Task Handle(Enumerate<IWidget> notification,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
string extensionsDirectory = Path.Combine(hostEnvironment.ContentRootPath, "Extensions");
|
||||||
|
if (Directory.Exists(extensionsDirectory))
|
||||||
|
{
|
||||||
|
List<string> assemblyPaths =
|
||||||
|
[
|
||||||
|
.. Directory.GetDirectories(extensionsDirectory)
|
||||||
|
.AsParallel()
|
||||||
|
.SelectMany(assemblyDirectory => Directory.GetFiles(assemblyDirectory, "*.dll"))
|
||||||
|
];
|
||||||
|
|
||||||
|
Parallel.ForEach(assemblyPaths, (string assemblyPath) =>
|
||||||
|
{
|
||||||
|
Assembly assembly = AssemblyLoadContext.Default.LoadFromAssemblyPath(assemblyPath);
|
||||||
|
mediator.PublishAsync(new Created<Assembly>(assembly));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
|
|
||||||
namespace Hyperbar;
|
|
||||||
|
|
||||||
public class WidgetEnumerator() :
|
|
||||||
IEnumerator<IWidgetComponentViewModel>
|
|
||||||
{
|
|
||||||
public IEnumerable<IWidgetComponentViewModel?> Next()
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class WidgetFactory :
|
||||||
|
IFactory<Type, IWidget>
|
||||||
|
{
|
||||||
|
public IWidget? Create(Type value)
|
||||||
|
{
|
||||||
|
if (Activator.CreateInstance(value) is IWidget widget)
|
||||||
|
{
|
||||||
|
return widget;
|
||||||
|
}
|
||||||
|
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class WidgetHandler(IWidgetServiceCollection serviceCollection,
|
||||||
|
IMediator mediator) :
|
||||||
|
INotificationHandler<Created<IWidget>>
|
||||||
|
{
|
||||||
|
public Task Handle(Created<IWidget> notification,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
if(notification.Value is IWidget widget)
|
||||||
|
{
|
||||||
|
IWidgetBuilder widgetBuilder = widget.Create();
|
||||||
|
if (widgetBuilder is IWidgetServiceBuilder serviceBuilder)
|
||||||
|
{
|
||||||
|
serviceBuilder.ConfigureWidgetServices(serviceCollection);
|
||||||
|
}
|
||||||
|
|
||||||
|
widgetBuilder.Build();
|
||||||
|
}
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class WidgetInitializer(IMediator mediator) :
|
||||||
|
IInitializer
|
||||||
|
{
|
||||||
|
public Task InitializeAsync()
|
||||||
|
{
|
||||||
|
mediator.PublishAsync<Enumerate<IWidget>>();
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class WidgetMonitor :
|
||||||
|
IInitializer
|
||||||
|
{
|
||||||
|
public Task InitializeAsync()
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
|
||||||
|
namespace Hyperbar;
|
||||||
|
|
||||||
|
public class WidgetServiceCollection :
|
||||||
|
IWidgetServiceCollection
|
||||||
|
{
|
||||||
|
public WidgetServiceCollection(Action<IServiceCollection> configureDelegate)
|
||||||
|
{
|
||||||
|
Services = new ServiceCollection();
|
||||||
|
configureDelegate.Invoke(Services);
|
||||||
|
}
|
||||||
|
|
||||||
|
public IServiceCollection Services { get; private set; }
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user