feat(adminui): editable AbCip device + tag lists via CollectionEditor
This commit is contained in:
+158
-31
@@ -146,27 +146,65 @@ else
|
|||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
@* Devices — read-only JSON view *@
|
@* Devices *@
|
||||||
<section class="panel rise mt-3" style="animation-delay:.12s">
|
<CollectionEditor TRow="AbCipDeviceRow" Items="_devices" Title="Devices" ItemNoun="device"
|
||||||
<div class="panel-head">Devices</div>
|
AnimationDelay=".12s"
|
||||||
<div style="padding:1rem">
|
NewRow="@(() => new AbCipDeviceRow())" Clone="@(r => r.Clone())"
|
||||||
<p class="form-text mb-2">
|
Validate="AbCipDeviceRow.ValidateRow">
|
||||||
Device list (host addresses, PLC family, packing overrides) — full list-editor coming in a follow-up phase. Each entry: <code>{ "hostAddress": "ab://gateway/1,0", "plcFamily": "ControlLogix" }</code>.
|
<HeaderTemplate>
|
||||||
</p>
|
<tr><th>Host address</th><th>PLC family</th><th>Device name</th><th></th></tr>
|
||||||
<pre class="form-control form-control-sm mono" style="min-height:4rem;overflow:auto;white-space:pre-wrap;">@_devicesJson</pre>
|
</HeaderTemplate>
|
||||||
</div>
|
<RowTemplate Context="d">
|
||||||
</section>
|
<td class="mono">@d.HostAddress</td><td>@d.PlcFamily</td>
|
||||||
|
<td>@(string.IsNullOrWhiteSpace(d.DeviceName) ? "—" : d.DeviceName)</td>
|
||||||
|
</RowTemplate>
|
||||||
|
<EditTemplate Context="d">
|
||||||
|
<div class="row g-3">
|
||||||
|
<div class="col-md-6"><label class="form-label">Host address</label>
|
||||||
|
<input class="form-control form-control-sm mono" @bind="d.HostAddress"
|
||||||
|
placeholder="ab://gateway/1,0" /></div>
|
||||||
|
<div class="col-md-3"><label class="form-label">PLC family</label>
|
||||||
|
<select class="form-select form-select-sm" @bind="d.PlcFamily">
|
||||||
|
@foreach (var e in Enum.GetValues<AbCipPlcFamily>()) { <option value="@e">@e</option> }
|
||||||
|
</select></div>
|
||||||
|
<div class="col-md-3"><label class="form-label">Device name</label>
|
||||||
|
<input class="form-control form-control-sm" @bind="d.DeviceName" /></div>
|
||||||
|
</div>
|
||||||
|
</EditTemplate>
|
||||||
|
</CollectionEditor>
|
||||||
|
|
||||||
@* Tags — read-only JSON view *@
|
@* Tags *@
|
||||||
<section class="panel rise mt-3" style="animation-delay:.14s">
|
<CollectionEditor TRow="AbCipTagRow" Items="_tags" Title="Tags" ItemNoun="tag"
|
||||||
<div class="panel-head">Tags</div>
|
AnimationDelay=".14s"
|
||||||
<div style="padding:1rem">
|
NewRow="@(() => new AbCipTagRow())" Clone="@(r => r.Clone())"
|
||||||
<p class="form-text mb-2">
|
Validate="AbCipTagRow.ValidateRow">
|
||||||
Tag list — full list-editor coming in a follow-up phase. Edit via the Tag editor pages or export/import the driver config JSON.
|
<HeaderTemplate>
|
||||||
</p>
|
<tr><th>Name</th><th>Device</th><th>Tag path</th><th>Type</th><th>Writable</th><th></th></tr>
|
||||||
<pre class="form-control form-control-sm mono" style="min-height:4rem;overflow:auto;white-space:pre-wrap;">@_tagsJson</pre>
|
</HeaderTemplate>
|
||||||
</div>
|
<RowTemplate Context="t">
|
||||||
</section>
|
<td class="mono">@t.Name</td><td class="mono">@t.DeviceHostAddress</td>
|
||||||
|
<td class="mono">@t.TagPath</td><td>@t.DataType</td><td>@(t.Writable ? "yes" : "no")</td>
|
||||||
|
</RowTemplate>
|
||||||
|
<EditTemplate Context="t">
|
||||||
|
<div class="row g-3">
|
||||||
|
<div class="col-md-6"><label class="form-label">Name</label>
|
||||||
|
<input class="form-control form-control-sm" @bind="t.Name" /></div>
|
||||||
|
<div class="col-md-6"><label class="form-label">Device host address</label>
|
||||||
|
<input class="form-control form-control-sm mono" @bind="t.DeviceHostAddress"
|
||||||
|
placeholder="ab://gateway/1,0" /></div>
|
||||||
|
<div class="col-md-6"><label class="form-label">Tag path</label>
|
||||||
|
<input class="form-control form-control-sm mono" @bind="t.TagPath"
|
||||||
|
placeholder="e.g. Program:Main.SomeTag" /></div>
|
||||||
|
<div class="col-md-3"><label class="form-label">Data type</label>
|
||||||
|
<select class="form-select form-select-sm" @bind="t.DataType">
|
||||||
|
@foreach (var e in Enum.GetValues<AbCipDataType>()) { <option value="@e">@e</option> }
|
||||||
|
</select></div>
|
||||||
|
<div class="col-md-3"><div class="form-check form-switch mt-4">
|
||||||
|
<input type="checkbox" class="form-check-input" @bind="t.Writable" id="tagWritable" />
|
||||||
|
<label class="form-check-label" for="tagWritable">Writable</label></div></div>
|
||||||
|
</div>
|
||||||
|
</EditTemplate>
|
||||||
|
</CollectionEditor>
|
||||||
|
|
||||||
<DriverResilienceSection @bind-ResilienceConfig="_form.ResilienceConfig" />
|
<DriverResilienceSection @bind-ResilienceConfig="_form.ResilienceConfig" />
|
||||||
</DriverFormShell>
|
</DriverFormShell>
|
||||||
@@ -202,11 +240,9 @@ else
|
|||||||
|
|
||||||
private void OnAddressPicked(string address) => _pickedAddress = address;
|
private void OnAddressPicked(string address) => _pickedAddress = address;
|
||||||
|
|
||||||
// Collections are preserved through round-trip and shown as read-only JSON.
|
// Held separately because Devices/Tags are collections — edited via the CollectionEditor modal.
|
||||||
private IReadOnlyList<AbCipDeviceOptions> _devices = [];
|
private List<AbCipDeviceRow> _devices = [];
|
||||||
private IReadOnlyList<AbCipTagDefinition> _tags = [];
|
private List<AbCipTagRow> _tags = [];
|
||||||
private string _devicesJson = "[]";
|
|
||||||
private string _tagsJson = "[]";
|
|
||||||
|
|
||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
{
|
{
|
||||||
@@ -246,12 +282,10 @@ else
|
|||||||
_form = FormModel.FromOptions(opts);
|
_form = FormModel.FromOptions(opts);
|
||||||
_form.ResilienceConfig = _existing.ResilienceConfig;
|
_form.ResilienceConfig = _existing.ResilienceConfig;
|
||||||
_form.RowVersion = _existing.RowVersion;
|
_form.RowVersion = _existing.RowVersion;
|
||||||
_devices = opts.Devices;
|
_devices = opts.Devices.Select(AbCipDeviceRow.FromDefinition).ToList();
|
||||||
_tags = opts.Tags;
|
_tags = opts.Tags.Select(AbCipTagRow.FromDefinition).ToList();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_devicesJson = System.Text.Json.JsonSerializer.Serialize(_devices, _jsonOpts);
|
|
||||||
_tagsJson = System.Text.Json.JsonSerializer.Serialize(_tags, _jsonOpts);
|
|
||||||
_loaded = true;
|
_loaded = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -260,7 +294,11 @@ else
|
|||||||
_busy = true; _error = null;
|
_busy = true; _error = null;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var configJson = System.Text.Json.JsonSerializer.Serialize(_form.ToOptions(_devices, _tags), _jsonOpts);
|
var configJson = System.Text.Json.JsonSerializer.Serialize(
|
||||||
|
_form.ToOptions(
|
||||||
|
_devices.Select(r => r.ToDefinition()).ToList(),
|
||||||
|
_tags.Select(r => r.ToDefinition()).ToList()),
|
||||||
|
_jsonOpts);
|
||||||
await using var db = await DbFactory.CreateDbContextAsync();
|
await using var db = await DbFactory.CreateDbContextAsync();
|
||||||
if (IsNew)
|
if (IsNew)
|
||||||
{
|
{
|
||||||
@@ -331,7 +369,11 @@ else
|
|||||||
}
|
}
|
||||||
|
|
||||||
private string SerializeCurrentConfig()
|
private string SerializeCurrentConfig()
|
||||||
=> System.Text.Json.JsonSerializer.Serialize(_form.ToOptions(_devices, _tags), _jsonOpts);
|
=> System.Text.Json.JsonSerializer.Serialize(
|
||||||
|
_form.ToOptions(
|
||||||
|
_devices.Select(r => r.ToDefinition()).ToList(),
|
||||||
|
_tags.Select(r => r.ToDefinition()).ToList()),
|
||||||
|
_jsonOpts);
|
||||||
|
|
||||||
private static AbCipDriverOptions? TryDeserialize(string json)
|
private static AbCipDriverOptions? TryDeserialize(string json)
|
||||||
{
|
{
|
||||||
@@ -339,6 +381,91 @@ else
|
|||||||
catch { return null; }
|
catch { return null; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Mutable VM for the modal editor — AbCipDeviceOptions is an immutable record.
|
||||||
|
public sealed class AbCipDeviceRow
|
||||||
|
{
|
||||||
|
public string HostAddress { get; set; } = "";
|
||||||
|
public AbCipPlcFamily PlcFamily { get; set; } = AbCipPlcFamily.ControlLogix;
|
||||||
|
public string? DeviceName { get; set; }
|
||||||
|
|
||||||
|
// Original record (null for newly-added rows). Preserves fields the editor doesn't expose
|
||||||
|
// (AllowPacking, ConnectionSize) across a load→save.
|
||||||
|
private AbCipDeviceOptions? _source;
|
||||||
|
|
||||||
|
public AbCipDeviceRow Clone() => (AbCipDeviceRow)MemberwiseClone(); // _source is an immutable record ref — safe to share
|
||||||
|
|
||||||
|
public static AbCipDeviceRow FromDefinition(AbCipDeviceOptions d) => new()
|
||||||
|
{
|
||||||
|
HostAddress = d.HostAddress, PlcFamily = d.PlcFamily, DeviceName = d.DeviceName,
|
||||||
|
_source = d,
|
||||||
|
};
|
||||||
|
|
||||||
|
public AbCipDeviceOptions ToDefinition()
|
||||||
|
{
|
||||||
|
var baseDef = _source ?? new AbCipDeviceOptions(HostAddress.Trim(), PlcFamily);
|
||||||
|
return baseDef with
|
||||||
|
{
|
||||||
|
HostAddress = HostAddress.Trim(),
|
||||||
|
PlcFamily = PlcFamily,
|
||||||
|
DeviceName = string.IsNullOrWhiteSpace(DeviceName) ? null : DeviceName.Trim(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string? ValidateRow(AbCipDeviceRow row, IReadOnlyList<AbCipDeviceRow> all, int? editIndex)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(row.HostAddress)) return "Host address is required.";
|
||||||
|
for (var i = 0; i < all.Count; i++)
|
||||||
|
if (i != editIndex && string.Equals(all[i].HostAddress, row.HostAddress, StringComparison.OrdinalIgnoreCase))
|
||||||
|
return $"Duplicate device host address '{row.HostAddress}'.";
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mutable VM for the modal editor — AbCipTagDefinition is an immutable record.
|
||||||
|
public sealed class AbCipTagRow
|
||||||
|
{
|
||||||
|
public string Name { get; set; } = "";
|
||||||
|
public string DeviceHostAddress { get; set; } = "";
|
||||||
|
public string TagPath { get; set; } = "";
|
||||||
|
public AbCipDataType DataType { get; set; } = AbCipDataType.DInt;
|
||||||
|
public bool Writable { get; set; } = true;
|
||||||
|
|
||||||
|
// Original record (null for newly-added rows). Preserves fields the editor doesn't expose
|
||||||
|
// (WriteIdempotent, Members, SafetyTag) across a load→save.
|
||||||
|
private AbCipTagDefinition? _source;
|
||||||
|
|
||||||
|
public AbCipTagRow Clone() => (AbCipTagRow)MemberwiseClone(); // _source is an immutable record ref — safe to share
|
||||||
|
|
||||||
|
public static AbCipTagRow FromDefinition(AbCipTagDefinition d) => new()
|
||||||
|
{
|
||||||
|
Name = d.Name, DeviceHostAddress = d.DeviceHostAddress, TagPath = d.TagPath,
|
||||||
|
DataType = d.DataType, Writable = d.Writable,
|
||||||
|
_source = d,
|
||||||
|
};
|
||||||
|
|
||||||
|
public AbCipTagDefinition ToDefinition()
|
||||||
|
{
|
||||||
|
var baseDef = _source ?? new AbCipTagDefinition(Name.Trim(), DeviceHostAddress.Trim(), TagPath.Trim(), DataType);
|
||||||
|
return baseDef with
|
||||||
|
{
|
||||||
|
Name = Name.Trim(),
|
||||||
|
DeviceHostAddress = DeviceHostAddress.Trim(),
|
||||||
|
TagPath = TagPath.Trim(),
|
||||||
|
DataType = DataType,
|
||||||
|
Writable = Writable,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string? ValidateRow(AbCipTagRow row, IReadOnlyList<AbCipTagRow> all, int? editIndex)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(row.Name)) return "Name is required.";
|
||||||
|
for (var i = 0; i < all.Count; i++)
|
||||||
|
if (i != editIndex && string.Equals(all[i].Name, row.Name, StringComparison.OrdinalIgnoreCase))
|
||||||
|
return $"Duplicate tag name '{row.Name}'.";
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Flat mutable model — all scalar properties settable for Blazor @bind-Value.
|
// Flat mutable model — all scalar properties settable for Blazor @bind-Value.
|
||||||
// Collections (Devices, Tags) are kept on the component and passed in on ToOptions().
|
// Collections (Devices, Tags) are kept on the component and passed in on ToOptions().
|
||||||
public sealed class FormModel
|
public sealed class FormModel
|
||||||
|
|||||||
+113
@@ -2,6 +2,7 @@ using System.Text.Json;
|
|||||||
using System.Text.Json.Serialization;
|
using System.Text.Json.Serialization;
|
||||||
using Shouldly;
|
using Shouldly;
|
||||||
using Xunit;
|
using Xunit;
|
||||||
|
using ZB.MOM.WW.OtOpcUa.AdminUI.Components.Pages.Clusters.Drivers;
|
||||||
using ZB.MOM.WW.OtOpcUa.Driver.AbCip;
|
using ZB.MOM.WW.OtOpcUa.Driver.AbCip;
|
||||||
|
|
||||||
namespace ZB.MOM.WW.OtOpcUa.AdminUI.Tests;
|
namespace ZB.MOM.WW.OtOpcUa.AdminUI.Tests;
|
||||||
@@ -14,6 +15,12 @@ public sealed class AbCipDriverPageFormSerializationTests
|
|||||||
WriteIndented = false,
|
WriteIndented = false,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
private static readonly JsonSerializerOptions TestJsonOpts = new()
|
||||||
|
{
|
||||||
|
PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
|
||||||
|
UnmappedMemberHandling = JsonUnmappedMemberHandling.Skip,
|
||||||
|
};
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
public void RoundTrip_PreservesKnownFields()
|
public void RoundTrip_PreservesKnownFields()
|
||||||
{
|
{
|
||||||
@@ -78,4 +85,110 @@ public sealed class AbCipDriverPageFormSerializationTests
|
|||||||
back.ShouldNotBeNull();
|
back.ShouldNotBeNull();
|
||||||
back.ProbeTimeoutSeconds.ShouldBe(10);
|
back.ProbeTimeoutSeconds.ShouldBe(10);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void DeviceRow_round_trips_through_definition()
|
||||||
|
{
|
||||||
|
var row = new AbCipDriverPage.AbCipDeviceRow
|
||||||
|
{
|
||||||
|
HostAddress = "ab://10.0.0.1/1,0", PlcFamily = AbCipPlcFamily.CompactLogix, DeviceName = "PLC-A",
|
||||||
|
};
|
||||||
|
var def = row.ToDefinition();
|
||||||
|
var back = AbCipDriverPage.AbCipDeviceRow.FromDefinition(def);
|
||||||
|
|
||||||
|
back.HostAddress.ShouldBe("ab://10.0.0.1/1,0");
|
||||||
|
back.PlcFamily.ShouldBe(AbCipPlcFamily.CompactLogix);
|
||||||
|
back.DeviceName.ShouldBe("PLC-A");
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void DeviceRow_preserves_unedited_fields()
|
||||||
|
{
|
||||||
|
var original = new AbCipDeviceOptions(
|
||||||
|
"ab://10.0.0.1/1,0", AbCipPlcFamily.ControlLogix, "PLC-A",
|
||||||
|
AllowPacking: true, ConnectionSize: 4002);
|
||||||
|
var row = AbCipDriverPage.AbCipDeviceRow.FromDefinition(original);
|
||||||
|
row.HostAddress = "ab://10.0.0.2/1,0";
|
||||||
|
|
||||||
|
var back = row.ToDefinition();
|
||||||
|
back.HostAddress.ShouldBe("ab://10.0.0.2/1,0");
|
||||||
|
back.AllowPacking.ShouldBe(true);
|
||||||
|
back.ConnectionSize.ShouldBe(4002);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void TagRow_round_trips_through_definition()
|
||||||
|
{
|
||||||
|
var row = new AbCipDriverPage.AbCipTagRow
|
||||||
|
{
|
||||||
|
Name = "Speed", DeviceHostAddress = "ab://10.0.0.1/1,0", TagPath = "Motor1.Speed",
|
||||||
|
DataType = AbCipDataType.Real, Writable = true,
|
||||||
|
};
|
||||||
|
var def = row.ToDefinition();
|
||||||
|
var back = AbCipDriverPage.AbCipTagRow.FromDefinition(def);
|
||||||
|
|
||||||
|
back.Name.ShouldBe("Speed");
|
||||||
|
back.DeviceHostAddress.ShouldBe("ab://10.0.0.1/1,0");
|
||||||
|
back.TagPath.ShouldBe("Motor1.Speed");
|
||||||
|
back.DataType.ShouldBe(AbCipDataType.Real);
|
||||||
|
back.Writable.ShouldBeTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void TagRow_preserves_unedited_fields()
|
||||||
|
{
|
||||||
|
var original = new AbCipTagDefinition(
|
||||||
|
"Speed", "ab://10.0.0.1/1,0", "Motor1.Speed", AbCipDataType.Structure,
|
||||||
|
Writable: true, WriteIdempotent: true,
|
||||||
|
Members: [new AbCipStructureMember("Sub", AbCipDataType.DInt)],
|
||||||
|
SafetyTag: true);
|
||||||
|
var row = AbCipDriverPage.AbCipTagRow.FromDefinition(original);
|
||||||
|
row.Name = "Renamed";
|
||||||
|
|
||||||
|
var back = row.ToDefinition();
|
||||||
|
back.Name.ShouldBe("Renamed");
|
||||||
|
back.WriteIdempotent.ShouldBeTrue();
|
||||||
|
back.SafetyTag.ShouldBeTrue();
|
||||||
|
back.Members.ShouldNotBeNull();
|
||||||
|
back.Members!.Count.ShouldBe(1);
|
||||||
|
back.Members[0].Name.ShouldBe("Sub");
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void ValidateDeviceRow_rejects_duplicate_host()
|
||||||
|
{
|
||||||
|
var rows = new List<AbCipDriverPage.AbCipDeviceRow> { new() { HostAddress = "ab://10.0.0.1/1,0" } };
|
||||||
|
AbCipDriverPage.AbCipDeviceRow.ValidateRow(new() { HostAddress = "ab://10.0.0.1/1,0" }, rows, null)
|
||||||
|
.ShouldNotBeNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void ValidateTagRow_rejects_duplicate_name()
|
||||||
|
{
|
||||||
|
var rows = new List<AbCipDriverPage.AbCipTagRow> { new() { Name = "Speed" } };
|
||||||
|
AbCipDriverPage.AbCipTagRow.ValidateRow(new() { Name = "Speed" }, rows, null)
|
||||||
|
.ShouldNotBeNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void Device_and_tag_lists_survive_options_serialize_round_trip()
|
||||||
|
{
|
||||||
|
var devices = new List<AbCipDeviceOptions>
|
||||||
|
{
|
||||||
|
new("ab://10.0.0.1/1,0", AbCipPlcFamily.ControlLogix, "PLC-1"),
|
||||||
|
new("ab://10.0.0.2/1,0", AbCipPlcFamily.CompactLogix, "PLC-2"),
|
||||||
|
};
|
||||||
|
var tags = new List<AbCipTagDefinition>
|
||||||
|
{
|
||||||
|
new("Speed", "ab://10.0.0.1/1,0", "Motor1.Speed", AbCipDataType.Real),
|
||||||
|
new("Run", "ab://10.0.0.2/1,0", "Motor2.Run", AbCipDataType.Bool),
|
||||||
|
};
|
||||||
|
var opts = new AbCipDriverPage.FormModel().ToOptions(devices, tags);
|
||||||
|
var json = JsonSerializer.Serialize(opts, TestJsonOpts);
|
||||||
|
var back = JsonSerializer.Deserialize<AbCipDriverOptions>(json, TestJsonOpts)!;
|
||||||
|
back.Devices.Count.ShouldBe(2);
|
||||||
|
back.Devices[0].HostAddress.ShouldBe("ab://10.0.0.1/1,0");
|
||||||
|
back.Tags.Count.ShouldBe(2);
|
||||||
|
back.Tags[0].Name.ShouldBe("Speed");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user