add support for REST-ful API building
- allow parameters within url - allow different methods other then just POST -> still needs to be tested!
This commit is contained in:
@@ -16,42 +16,62 @@ public class ApiClient(string apiUrl) : IApiClient
|
|||||||
|
|
||||||
public virtual Task<ApiResponse> SendRequest(string route)
|
public virtual Task<ApiResponse> SendRequest(string route)
|
||||||
{
|
{
|
||||||
return SendRequest(route, null);
|
return SendRequest(route, HttpMethod.Post);
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual Task<ApiResponse> SendRequest(string route, HttpMethod method)
|
||||||
|
{
|
||||||
|
return SendRequest(route, method, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
public virtual Task<ApiResponse> SendRequest(string route, ApiMessage? message)
|
public virtual Task<ApiResponse> SendRequest(string route, ApiMessage? message)
|
||||||
{
|
{
|
||||||
return SendRequest(route, message, null);
|
return SendRequest(route, HttpMethod.Post, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
public virtual async Task<ApiResponse> SendRequest(string route, ApiMessage? message, IApiMessageSerializer? serializer)
|
public virtual Task<ApiResponse> SendRequest(string route, HttpMethod method, ApiMessage? message)
|
||||||
|
{
|
||||||
|
return SendRequest(route, method, message, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual async Task<ApiResponse> SendRequest(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer? serializer)
|
||||||
{
|
{
|
||||||
serializer ??= Serializer;
|
serializer ??= Serializer;
|
||||||
|
|
||||||
Log.InfoFormat("Send message to {0}", route);
|
Log.InfoFormat("Send message to {0}", route);
|
||||||
|
|
||||||
var res = await Send(route, message, serializer);
|
var res = await Send(route, method, message, serializer);
|
||||||
|
|
||||||
return new(res.StatusCode);
|
return new(res.StatusCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route) where TResponse : ApiMessage
|
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route) where TResponse : ApiMessage
|
||||||
{
|
{
|
||||||
return SendRequest<TResponse>(route, null);
|
return SendRequest<TResponse>(route, HttpMethod.Post);
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method) where TResponse : ApiMessage
|
||||||
|
{
|
||||||
|
return SendRequest<TResponse>(route, method, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message) where TResponse : ApiMessage
|
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message) where TResponse : ApiMessage
|
||||||
{
|
{
|
||||||
return SendRequest<TResponse>(route, message, null);
|
return SendRequest<TResponse>(route, HttpMethod.Post, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
public virtual async Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message, IApiMessageSerializer? serializer) where TResponse : ApiMessage
|
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method, ApiMessage? message) where TResponse : ApiMessage
|
||||||
|
{
|
||||||
|
return SendRequest<TResponse>(route, method, message, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual async Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer? serializer) where TResponse : ApiMessage
|
||||||
{
|
{
|
||||||
serializer ??= Serializer;
|
serializer ??= Serializer;
|
||||||
|
|
||||||
Log.InfoFormat("Send request to {0}", route);
|
Log.InfoFormat("Send request to {0}", route);
|
||||||
|
|
||||||
var res = await Send(route, message, serializer);
|
var res = await Send(route, method, message, serializer);
|
||||||
TResponse? result = null;
|
TResponse? result = null;
|
||||||
|
|
||||||
if (res.IsSuccessStatusCode)
|
if (res.IsSuccessStatusCode)
|
||||||
@@ -60,7 +80,7 @@ public class ApiClient(string apiUrl) : IApiClient
|
|||||||
return new(res.StatusCode, result);
|
return new(res.StatusCode, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual async Task<HttpResponseMessage> Send(string route, ApiMessage? message, IApiMessageSerializer serializer)
|
protected virtual async Task<HttpResponseMessage> Send(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer serializer)
|
||||||
{
|
{
|
||||||
var url = ApiUrl + route;
|
var url = ApiUrl + route;
|
||||||
HttpContent content;
|
HttpContent content;
|
||||||
@@ -80,7 +100,12 @@ public class ApiClient(string apiUrl) : IApiClient
|
|||||||
|
|
||||||
Log.Debug("Sending request");
|
Log.Debug("Sending request");
|
||||||
|
|
||||||
return await httpClient.PostAsync(url, content);
|
var httpmsg = new HttpRequestMessage(method, url)
|
||||||
|
{
|
||||||
|
Content = content
|
||||||
|
};
|
||||||
|
|
||||||
|
return await httpClient.SendAsync(httpmsg);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual string? EncodeAuthKey()
|
protected virtual string? EncodeAuthKey()
|
||||||
|
|||||||
@@ -1,9 +1,21 @@
|
|||||||
namespace Pilz.Net.Api;
|
namespace Pilz.Net.Api;
|
||||||
|
|
||||||
[AttributeUsage(AttributeTargets.Method)]
|
[AttributeUsage(AttributeTargets.Method)]
|
||||||
public class ApiMessageHandlerAttribute(string route) : Attribute
|
public class ApiMessageHandlerAttribute(string route, params string[] methods) : Attribute
|
||||||
{
|
{
|
||||||
public string Route { get; set; } = route;
|
public ApiMessageHandlerAttribute(string route, params HttpMethod[] methods)
|
||||||
|
: this(route, methods.Select(m => m.Method).ToArray())
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public ApiMessageHandlerAttribute(string route)
|
||||||
|
: this(route, HttpMethod.Post)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Route { get; } = route;
|
||||||
|
public string[] Methods { get; } = methods;
|
||||||
public Type? Serializer { get; set; }
|
public Type? Serializer { get; set; }
|
||||||
public bool RequiesAuth { get; set; }
|
public bool RequiesAuth { get; set; }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,4 +6,6 @@ public record class ApiRequestInfo(
|
|||||||
ApiMessage? Message,
|
ApiMessage? Message,
|
||||||
[property: MemberNotNullWhen(true, "AuthKey")]
|
[property: MemberNotNullWhen(true, "AuthKey")]
|
||||||
bool IsAuthenticated,
|
bool IsAuthenticated,
|
||||||
string? AuthKey);
|
string? AuthKey,
|
||||||
|
string Url,
|
||||||
|
string Method);
|
||||||
|
|||||||
@@ -1,14 +1,20 @@
|
|||||||
using Castle.Core.Logging;
|
using Castle.Core.Logging;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
using Pilz.Extensions.Reflection;
|
using Pilz.Extensions.Reflection;
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
using System.Linq;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
|
using System.Text.Encodings.Web;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using System.Web;
|
||||||
using static Pilz.Net.Api.IApiServer;
|
using static Pilz.Net.Api.IApiServer;
|
||||||
|
|
||||||
namespace Pilz.Net.Api;
|
namespace Pilz.Net.Api;
|
||||||
|
|
||||||
public class ApiServer(string apiUrl) : IApiServer
|
public class ApiServer(string apiUrl) : IApiServer
|
||||||
{
|
{
|
||||||
protected readonly Dictionary<string, Delegate> handlers = [];
|
protected readonly List<PrivateMessageHandler> handlers = [];
|
||||||
protected readonly Dictionary<Type, IApiMessageSerializer> serializers = [];
|
protected readonly Dictionary<Type, IApiMessageSerializer> serializers = [];
|
||||||
protected readonly HttpListener httpListener = new();
|
protected readonly HttpListener httpListener = new();
|
||||||
|
|
||||||
@@ -16,6 +22,10 @@ public class ApiServer(string apiUrl) : IApiServer
|
|||||||
public event OnCheckContextEventHandler? OnCheckContext;
|
public event OnCheckContextEventHandler? OnCheckContext;
|
||||||
public event OnCheckContextEventHandler? OnCheckContextCompleted;
|
public event OnCheckContextEventHandler? OnCheckContextCompleted;
|
||||||
|
|
||||||
|
protected record PrivateParameterInfo(string Name, int Index);
|
||||||
|
|
||||||
|
protected record PrivateMessageHandler(string Url, bool UseRegEx, Delegate Handler, PrivateParameterInfo[] Parameters, ApiMessageHandlerAttribute Attribute);
|
||||||
|
|
||||||
protected record PrivateApiResult(ApiResult Original, string? ResultJson);
|
protected record PrivateApiResult(ApiResult Original, string? ResultJson);
|
||||||
|
|
||||||
public string ApiUrl { get; } = apiUrl;
|
public string ApiUrl { get; } = apiUrl;
|
||||||
@@ -72,10 +82,33 @@ public class ApiServer(string apiUrl) : IApiServer
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Resolves parameters
|
||||||
|
var url = attribute.Route;
|
||||||
|
var useRegEx = false;
|
||||||
|
var nextBreacket = url.IndexOf('{');
|
||||||
|
var parameters = new List<PrivateParameterInfo>();
|
||||||
|
while (nextBreacket != -1)
|
||||||
|
{
|
||||||
|
var endBreacket = url.IndexOf('}', nextBreacket + 1);
|
||||||
|
if (endBreacket == -1)
|
||||||
|
{
|
||||||
|
var name = url.Substring(nextBreacket + 1, endBreacket - nextBreacket - 1);
|
||||||
|
const string regex = "[A-Za-z0-9%]+";
|
||||||
|
url = url.Replace(url.Substring(nextBreacket, endBreacket + 1), regex);
|
||||||
|
|
||||||
|
var index = url.Substring(0, nextBreacket + 1).Split('/').Length;
|
||||||
|
parameters.Add(new(name, index));
|
||||||
|
|
||||||
|
useRegEx = true;
|
||||||
|
nextBreacket = url.IndexOf('{', endBreacket + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (useRegEx)
|
||||||
|
url = url.Replace(".", "\\."); // Escape special characters
|
||||||
|
|
||||||
// Add handler
|
// Add handler
|
||||||
var fullUrl = attribute.Route;
|
Log.InfoFormat("Added handler for {0}", attribute.Route);
|
||||||
Log.InfoFormat("Added handler for {0}", fullUrl);
|
handlers.Add(new(url, useRegEx, handler, [.. parameters], attribute));
|
||||||
handlers.Add(fullUrl, handler);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void Receive()
|
protected void Receive()
|
||||||
@@ -143,16 +176,6 @@ public class ApiServer(string apiUrl) : IApiServer
|
|||||||
context.Response.OutputStream.Close();
|
context.Response.OutputStream.Close();
|
||||||
}
|
}
|
||||||
|
|
||||||
Log.Debug("Sanity checks");
|
|
||||||
if (context.Request.HttpMethod != HttpMethod.Post.Method
|
|
||||||
|| context.Request.ContentType is not string contentType
|
|
||||||
|| !contentType.Contains("application/json"))
|
|
||||||
{
|
|
||||||
Log.Info("Request has no json content");
|
|
||||||
close();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse url
|
// Parse url
|
||||||
Log.Debug("Parse url");
|
Log.Debug("Parse url");
|
||||||
var path = context.Request.Url?.PathAndQuery.Replace(ApiUrl, string.Empty);
|
var path = context.Request.Url?.PathAndQuery.Replace(ApiUrl, string.Empty);
|
||||||
@@ -163,25 +186,51 @@ public class ApiServer(string apiUrl) : IApiServer
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read input content
|
// Find handler
|
||||||
Log.Debug("Read input content");
|
Log.Debug("Find handler");
|
||||||
string? contentJson;
|
if (!TryGetHandler(path, context.Request.HttpMethod, out var handler))
|
||||||
if (context.Request.ContentLength64 > 0)
|
|
||||||
{
|
{
|
||||||
using StreamReader input = new(context.Request.InputStream);
|
Log.Info("Request handler couldn't be found");
|
||||||
contentJson = input.ReadToEnd();
|
close();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
contentJson = null;
|
|
||||||
|
|
||||||
// Get auth key
|
// Get auth key
|
||||||
Log.Debug("Get auth key");
|
Log.Debug("Get auth key");
|
||||||
if (context.Request.Headers.Get("API-AUTH-KEY") is not string authKey)
|
if (context.Request.Headers.Get("API-AUTH-KEY") is not string authKey)
|
||||||
authKey = null!;
|
authKey = null!;
|
||||||
|
|
||||||
|
// Read input content
|
||||||
|
Log.Debug("Read input content");
|
||||||
|
string? contentJson;
|
||||||
|
if (context.Request.ContentType is string contentType
|
||||||
|
&& contentType.Contains("application/json")
|
||||||
|
&& context.Request.ContentLength64 > 0)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using StreamReader input = new(context.Request.InputStream);
|
||||||
|
contentJson = input.ReadToEnd();
|
||||||
|
}
|
||||||
|
catch (OutOfMemoryException)
|
||||||
|
{
|
||||||
|
Log.Error("Error reading remote data due to missing memory");
|
||||||
|
close();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Log.Error("Error reading remote data", ex);
|
||||||
|
close();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
contentJson = null;
|
||||||
|
|
||||||
// Handle message
|
// Handle message
|
||||||
Log.Debug("Handle mssage");
|
Log.Debug("Handle mssage");
|
||||||
if (HandleMessage(path, contentJson, authKey) is not PrivateApiResult result)
|
if (HandleMessage(path, context.Request.HttpMethod, handler, contentJson, authKey) is not PrivateApiResult result)
|
||||||
{
|
{
|
||||||
Log.Warn("Request couldn't be handled");
|
Log.Warn("Request couldn't be handled");
|
||||||
close();
|
close();
|
||||||
@@ -206,28 +255,22 @@ public class ApiServer(string apiUrl) : IApiServer
|
|||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual PrivateApiResult? HandleMessage(string url, string? json, string? authKey)
|
protected virtual PrivateApiResult? HandleMessage(string url, string method, PrivateMessageHandler handler, string? json, string? authKey)
|
||||||
{
|
{
|
||||||
// Get handler
|
|
||||||
Log.Debug("Find handler");
|
|
||||||
if (!handlers.TryGetValue(url, out var handler)
|
|
||||||
|| handler.Method.GetCustomAttribute<ApiMessageHandlerAttribute>() is not ApiMessageHandlerAttribute attribute)
|
|
||||||
return null;
|
|
||||||
|
|
||||||
// Check authentication
|
// Check authentication
|
||||||
Log.Debug("Check authentication");
|
Log.Debug("Check authentication");
|
||||||
var isAuthenticated = false;
|
var isAuthenticated = false;
|
||||||
if (!string.IsNullOrWhiteSpace(authKey) && DecodeAuthKey(authKey) is string authKeyDecoded)
|
if (!string.IsNullOrWhiteSpace(authKey) && DecodeAuthKey(authKey) is string authKeyDecoded)
|
||||||
isAuthenticated = CheckAuthentication(authKeyDecoded, handler);
|
isAuthenticated = CheckAuthentication(authKeyDecoded, handler.Handler);
|
||||||
else
|
else
|
||||||
authKeyDecoded = null!;
|
authKeyDecoded = null!;
|
||||||
if (attribute.RequiesAuth && !isAuthenticated)
|
if (handler.Attribute.RequiesAuth && !isAuthenticated)
|
||||||
return new(ApiResult.Unauthorized(), null);
|
return new(ApiResult.Unauthorized(), null);
|
||||||
|
|
||||||
// Get required infos
|
// Get required infos
|
||||||
Log.Debug("Identify message parameter type and serializer");
|
Log.Debug("Identify message parameter type and serializer");
|
||||||
var targetType = handler.Method.GetParameters().FirstOrDefault(p => p.ParameterType.IsAssignableTo(typeof(ApiMessage)))?.ParameterType;
|
var targetType = handler.Handler.Method.GetParameters().FirstOrDefault(p => p.ParameterType.IsAssignableTo(typeof(ApiMessage)))?.ParameterType;
|
||||||
var serializer = GetSerializer(attribute.Serializer);
|
var serializer = GetSerializer(handler.Attribute.Serializer);
|
||||||
|
|
||||||
// Deserialize
|
// Deserialize
|
||||||
Log.Debug("Deserialize message");
|
Log.Debug("Deserialize message");
|
||||||
@@ -239,8 +282,8 @@ public class ApiServer(string apiUrl) : IApiServer
|
|||||||
|
|
||||||
// Invoke handler
|
// Invoke handler
|
||||||
Log.Debug("Invoke handler");
|
Log.Debug("Invoke handler");
|
||||||
var parameters = BuildParameters(handler, () => message, () => new(message, isAuthenticated, authKeyDecoded));
|
var parameters = BuildParameters(url, handler, () => message, () => new(message, isAuthenticated, authKeyDecoded, url, method));
|
||||||
if (handler.DynamicInvoke(parameters) is not ApiResult result)
|
if (handler.Handler.DynamicInvoke(parameters) is not ApiResult result)
|
||||||
return new(ApiResult.InternalServerError(), null);
|
return new(ApiResult.InternalServerError(), null);
|
||||||
|
|
||||||
// Return result without message
|
// Return result without message
|
||||||
@@ -258,9 +301,25 @@ public class ApiServer(string apiUrl) : IApiServer
|
|||||||
return new(result, resultStr);
|
return new(result, resultStr);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual object?[]? BuildParameters(Delegate handler, Func<ApiMessage?> getMessage, Func<ApiRequestInfo> getRequestInfo)
|
protected virtual bool TryGetHandler(string url, string method, [NotNullWhen(true)] out PrivateMessageHandler? handler)
|
||||||
{
|
{
|
||||||
var infos = handler.Method.GetParameters();
|
handler = handlers.FirstOrDefault(handler =>
|
||||||
|
{
|
||||||
|
if (!handler.Attribute.Methods.Contains(method))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (handler.UseRegEx)
|
||||||
|
return Regex.IsMatch(url, handler.Url, RegexOptions.IgnoreCase);
|
||||||
|
|
||||||
|
return handler.Url.Equals(url, StringComparison.InvariantCultureIgnoreCase);
|
||||||
|
});
|
||||||
|
|
||||||
|
return handler != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected virtual object?[]? BuildParameters(string url, PrivateMessageHandler handler, Func<ApiMessage?> getMessage, Func<ApiRequestInfo> getRequestInfo)
|
||||||
|
{
|
||||||
|
var infos = handler.Handler.Method.GetParameters();
|
||||||
var objs = new List<object?>();
|
var objs = new List<object?>();
|
||||||
|
|
||||||
foreach (var info in infos)
|
foreach (var info in infos)
|
||||||
@@ -269,6 +328,9 @@ public class ApiServer(string apiUrl) : IApiServer
|
|||||||
objs.Add(getMessage());
|
objs.Add(getMessage());
|
||||||
else if (info.ParameterType.IsAssignableTo(typeof(ApiRequestInfo)))
|
else if (info.ParameterType.IsAssignableTo(typeof(ApiRequestInfo)))
|
||||||
objs.Add(getRequestInfo());
|
objs.Add(getRequestInfo());
|
||||||
|
else if (handler.Parameters.FirstOrDefault(p => p.Name.Equals(info.Name, StringComparison.InvariantCultureIgnoreCase)) is PrivateParameterInfo parameterInfo
|
||||||
|
&& url.Split('/').ElementAtOrDefault(parameterInfo.Index) is string parameterValue)
|
||||||
|
objs.Add(Convert.ChangeType(HttpUtility.UrlDecode(parameterValue), info.ParameterType)); // or Uri.UnescapeDataString(); maybe run this line twice?
|
||||||
else
|
else
|
||||||
objs.Add(null);
|
objs.Add(null);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,13 +14,21 @@ public interface IApiClient
|
|||||||
|
|
||||||
Task<ApiResponse> SendRequest(string route);
|
Task<ApiResponse> SendRequest(string route);
|
||||||
|
|
||||||
|
Task<ApiResponse> SendRequest(string route, HttpMethod method);
|
||||||
|
|
||||||
Task<ApiResponse> SendRequest(string route, ApiMessage? message);
|
Task<ApiResponse> SendRequest(string route, ApiMessage? message);
|
||||||
|
|
||||||
Task<ApiResponse> SendRequest(string route, ApiMessage? message, IApiMessageSerializer? serializer);
|
Task<ApiResponse> SendRequest(string route, HttpMethod method, ApiMessage? message);
|
||||||
|
|
||||||
|
Task<ApiResponse> SendRequest(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer? serializer);
|
||||||
|
|
||||||
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route) where TResponse : ApiMessage;
|
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route) where TResponse : ApiMessage;
|
||||||
|
|
||||||
|
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method) where TResponse : ApiMessage;
|
||||||
|
|
||||||
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message) where TResponse : ApiMessage;
|
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message) where TResponse : ApiMessage;
|
||||||
|
|
||||||
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message, IApiMessageSerializer? serializer) where TResponse : ApiMessage;
|
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method, ApiMessage? message) where TResponse : ApiMessage;
|
||||||
|
|
||||||
|
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer? serializer) where TResponse : ApiMessage;
|
||||||
}
|
}
|
||||||
|
|||||||
10
Pilz.UI/Controls/ConfigurationManager/ConfigurationEntry.cs
Normal file
10
Pilz.UI/Controls/ConfigurationManager/ConfigurationEntry.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
namespace Pilz.UI.Controls.ConfigurationManager;
|
||||||
|
|
||||||
|
public class ConfigurationEntry(string name, string title, ConfigurationValueListener Listener)
|
||||||
|
{
|
||||||
|
public event EventHandler? OnValueChanged;
|
||||||
|
|
||||||
|
public string Name { get; } = name;
|
||||||
|
public string Title { get; } = title;
|
||||||
|
public ConfigurationValueListener Listener { get; } = Listener;
|
||||||
|
}
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
namespace Pilz.UI.Controls.ConfigurationManager;
|
||||||
|
|
||||||
|
public class ConfigurationManager
|
||||||
|
{
|
||||||
|
private readonly List<ConfigurationPanel> panels = [];
|
||||||
|
|
||||||
|
public void Register(ConfigurationPanel panel)
|
||||||
|
{
|
||||||
|
panels.Add(panel);
|
||||||
|
}
|
||||||
|
|
||||||
|
public IEnumerable<Control> Build()
|
||||||
|
{
|
||||||
|
foreach (var panel in panels)
|
||||||
|
{
|
||||||
|
panel.Build(this);
|
||||||
|
yield return panel;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
32
Pilz.UI/Controls/ConfigurationManager/ConfigurationPanel.cs
Normal file
32
Pilz.UI/Controls/ConfigurationManager/ConfigurationPanel.cs
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
namespace Pilz.UI.Controls.ConfigurationManager;
|
||||||
|
|
||||||
|
public class ConfigurationPanel : TableLayoutPanel
|
||||||
|
{
|
||||||
|
private readonly List<ConfigurationEntry> entries = [];
|
||||||
|
|
||||||
|
public ConfigurationEntry CreateEntry(string name, string title)
|
||||||
|
{
|
||||||
|
return CreateEntry(name, title, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ConfigurationEntry CreateEntry(string name, string title, Action? create)
|
||||||
|
{
|
||||||
|
return CreateEntry(new(name, title, ));
|
||||||
|
}
|
||||||
|
|
||||||
|
public ConfigurationEntry CreateEntry(ConfigurationEntry entry)
|
||||||
|
{
|
||||||
|
entries.Add(entry);
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
internal protected void Build(ConfigurationManager manager)
|
||||||
|
{
|
||||||
|
foreach (var entry in entries)
|
||||||
|
{
|
||||||
|
var control = ;
|
||||||
|
entry.Listener.Initialize();
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
namespace Pilz.UI.Controls.ConfigurationManager;
|
||||||
|
|
||||||
|
public abstract class ConfigurationValueListener : IDisposable
|
||||||
|
{
|
||||||
|
public event EventHandler? ValueChanged;
|
||||||
|
|
||||||
|
protected virtual void OnValueChanged(object sender, EventArgs e)
|
||||||
|
{
|
||||||
|
ValueChanged?.Invoke(sender, e);
|
||||||
|
}
|
||||||
|
|
||||||
|
internal protected abstract void Initialize(Control control);
|
||||||
|
|
||||||
|
public abstract void Dispose();
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user