PortModal.razor 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. @using System.ComponentModel.DataAnnotations
  2. @using RackPeek.Domain.Resources.SubResources
  3. @if (IsOpen)
  4. {
  5. <div class="fixed inset-0 z-50 flex items-center justify-center"
  6. data-testid="@BaseTestId">
  7. <!-- Backdrop -->
  8. <div class="absolute inset-0 bg-black/70"
  9. data-testid="@($"{BaseTestId}-backdrop")"
  10. @onclick="Cancel">
  11. </div>
  12. <!-- Modal -->
  13. <div class="relative bg-zinc-900 border border-zinc-800 rounded w-full max-w-md p-4"
  14. data-testid="@($"{BaseTestId}-container")">
  15. <!-- Header -->
  16. <div class="flex justify-between items-center mb-4"
  17. data-testid="@($"{BaseTestId}-header")">
  18. <div class="text-zinc-100 text-sm font-medium"
  19. data-testid="@($"{BaseTestId}-title")">
  20. @(IsEdit ? "Modify Port" : "Add Port")
  21. </div>
  22. <button class="text-zinc-400 hover:text-zinc-200"
  23. data-testid="@($"{BaseTestId}-close")"
  24. @onclick="Cancel">
  25. </button>
  26. </div>
  27. <!-- Form -->
  28. <EditForm Model="_model"
  29. OnValidSubmit="HandleValidSubmit"
  30. data-testid="@($"{BaseTestId}-form")">
  31. <DataAnnotationsValidator/>
  32. <ValidationSummary class="text-xs text-red-400 mb-3"
  33. data-testid="@($"{BaseTestId}-validation-summary")"/>
  34. <div class="space-y-3 text-sm">
  35. <!-- Type -->
  36. <div data-testid="@($"{BaseTestId}-type-field")">
  37. <label class="block text-zinc-400 mb-1">
  38. Type
  39. </label>
  40. <InputSelect class="w-full bg-zinc-800 border border-zinc-700 rounded px-2 py-1 text-zinc-100"
  41. data-testid="@($"{BaseTestId}-type-input")"
  42. @bind-Value="_model.Type">
  43. <option value="">Select type</option>
  44. @foreach (var type in Nic.ValidNicTypes)
  45. {
  46. <option value="@type">@type</option>
  47. }
  48. </InputSelect>
  49. </div>
  50. <!-- Speed -->
  51. <div data-testid="@($"{BaseTestId}-speed-field")">
  52. <label class="block text-zinc-400 mb-1">
  53. Speed
  54. </label>
  55. <InputNumber class="w-full bg-zinc-800 border border-zinc-700 rounded px-2 py-1 text-zinc-100"
  56. data-testid="@($"{BaseTestId}-speed-input")"
  57. @bind-Value="_model.Speed"/>
  58. </div>
  59. <!-- Count -->
  60. <div data-testid="@($"{BaseTestId}-count-field")">
  61. <label class="block text-zinc-400 mb-1">
  62. Count
  63. </label>
  64. <InputNumber class="w-full bg-zinc-800 border border-zinc-700 rounded px-2 py-1 text-zinc-100"
  65. data-testid="@($"{BaseTestId}-count-input")"
  66. @bind-Value="_model.Count"/>
  67. </div>
  68. </div>
  69. <!-- Actions -->
  70. <div class="flex justify-between items-center mt-5"
  71. data-testid="@($"{BaseTestId}-actions")">
  72. @if (IsEdit)
  73. {
  74. <button type="button"
  75. class="text-red-400 hover:text-red-300 text-sm"
  76. data-testid="@($"{BaseTestId}-delete")"
  77. @onclick="HandleDelete">
  78. Delete Port
  79. </button>
  80. }
  81. else
  82. {
  83. <span></span>
  84. }
  85. <div class="flex gap-2">
  86. <button type="button"
  87. class="px-3 py-1 rounded border border-zinc-700 text-zinc-300 hover:bg-zinc-800"
  88. data-testid="@($"{BaseTestId}-cancel")"
  89. @onclick="Cancel">
  90. Cancel
  91. </button>
  92. <button type="submit"
  93. class="px-3 py-1 rounded bg-emerald-600 text-black hover:bg-emerald-500"
  94. data-testid="@($"{BaseTestId}-submit")">
  95. @(IsEdit ? "Update" : "Add")
  96. </button>
  97. </div>
  98. </div>
  99. </EditForm>
  100. </div>
  101. </div>
  102. }
  103. @code {
  104. [Parameter] public bool IsOpen { get; set; }
  105. [Parameter] public EventCallback<bool> IsOpenChanged { get; set; }
  106. [Parameter] public Port? Value { get; set; }
  107. [Parameter] public EventCallback<Port> OnSubmit { get; set; }
  108. [Parameter] public EventCallback<Port> OnDelete { get; set; }
  109. [Parameter] public string? TestIdPrefix { get; set; }
  110. private string BaseTestId =>
  111. string.IsNullOrWhiteSpace(TestIdPrefix)
  112. ? "port-modal"
  113. : $"{TestIdPrefix}-port-modal";
  114. private PortFormModel _model = new();
  115. private bool IsEdit => Value is not null;
  116. protected override void OnParametersSet()
  117. {
  118. if (IsOpen)
  119. {
  120. _model = Value is null
  121. ? new PortFormModel()
  122. : new PortFormModel
  123. {
  124. Type = Value.Type,
  125. Speed = Value.Speed,
  126. Count = Value.Count
  127. };
  128. }
  129. }
  130. private async Task HandleValidSubmit()
  131. {
  132. var port = new Port
  133. {
  134. Type = _model.Type,
  135. Speed = _model.Speed,
  136. Count = _model.Count
  137. };
  138. await OnSubmit.InvokeAsync(port);
  139. await Close();
  140. }
  141. private async Task HandleDelete()
  142. {
  143. if (Value is not null)
  144. {
  145. await OnDelete.InvokeAsync(Value);
  146. await Close();
  147. }
  148. }
  149. private async Task Cancel()
  150. {
  151. await Close();
  152. }
  153. private async Task Close()
  154. {
  155. _model = new PortFormModel();
  156. await IsOpenChanged.InvokeAsync(false);
  157. }
  158. private class PortFormModel
  159. {
  160. [Required] public string? Type { get; set; }
  161. [Range(0, 400)] public double? Speed { get; set; }
  162. [Range(1, 256)] public int? Count { get; set; }
  163. }
  164. }