FirewallCardComponent.razor 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. @using RackPeek.Domain.Resources.Firewalls
  2. @using Shared.Rcl.Hardware
  3. @inject UpdateFirewallUseCase UpdateUseCase
  4. @inject IGetResourceByNameUseCase<Firewall> GetByNameUseCase
  5. @inject IDeleteResourceUseCase<Firewall> DeleteUseCase
  6. @inject ICloneResourceUseCase<Firewall> CloneUseCase
  7. @inject IRenameResourceUseCase<Firewall> RenameUseCase
  8. @inject NavigationManager Nav
  9. <div class="border border-zinc-800 rounded p-4 bg-zinc-900"
  10. data-testid=@($"firewall-item-{Firewall.Name.Replace(" ", "-")}")>
  11. <div class="flex justify-between items-center mb-3">
  12. <div class="text-zinc-100 hover:text-emerald-300">
  13. <NavLink href="@($"resources/hardware/{Uri.EscapeDataString(Firewall.Name)}")"
  14. class="block"
  15. data-testid=@($"firewall-item-{Firewall.Name.Replace(" ", "-")}-link")>
  16. @Firewall.Name
  17. </NavLink>
  18. </div>
  19. <div class="flex gap-3 text-xs">
  20. @if (!_isEditing)
  21. {
  22. <button data-testid="edit-firewall-button"
  23. class="text-zinc-400 hover:text-zinc-200"
  24. @onclick="BeginEdit">
  25. Edit
  26. </button>
  27. <button data-testid="rename-firewall-button"
  28. class="text-blue-400 hover:text-blue-300 transition"
  29. @onclick="OpenRename">
  30. Rename
  31. </button>
  32. <button data-testid="clone-firewall-button"
  33. class="text-emerald-400 hover:text-emerald-300 transition"
  34. @onclick="OpenClone">
  35. Clone
  36. </button>
  37. <button data-testid="delete-firewall-button"
  38. class="text-red-400 hover:text-red-300 transition"
  39. @onclick="ConfirmDelete">
  40. Delete
  41. </button>
  42. }
  43. else
  44. {
  45. <button data-testid="save-firewall-button"
  46. class="text-emerald-400 hover:text-emerald-300"
  47. @onclick="Save">
  48. Save
  49. </button>
  50. <button data-testid="cancel-firewall-button"
  51. class="text-zinc-500 hover:text-zinc-300"
  52. @onclick="Cancel">
  53. Cancel
  54. </button>
  55. }
  56. </div>
  57. </div>
  58. <div class="grid grid-cols-1 md:grid-cols-2 gap-3 text-sm">
  59. <!-- Model -->
  60. <div data-testid="firewall-model-section">
  61. <div class="text-zinc-400 mb-1">Model</div>
  62. @if (_isEditing)
  63. {
  64. <input data-testid="firewall-model-input"
  65. class="w-full px-3 py-2 rounded-md bg-zinc-800 text-zinc-100 border border-zinc-600"
  66. @bind="_edit.Model"/>
  67. }
  68. else if (!string.IsNullOrWhiteSpace(Firewall.Model))
  69. {
  70. <div class="text-zinc-300"
  71. data-testid="firewall-model-value">
  72. @Firewall.Model
  73. </div>
  74. }
  75. </div>
  76. <!-- Features -->
  77. <div data-testid="firewall-features-section">
  78. <div class="text-zinc-400 mb-1">Features</div>
  79. @if (_isEditing)
  80. {
  81. <div class="flex gap-4">
  82. <label class="flex items-center gap-2 text-zinc-300">
  83. <input type="checkbox"
  84. data-testid="firewall-managed-checkbox"
  85. @bind="_edit.Managed"/>
  86. Managed
  87. </label>
  88. <label class="flex items-center gap-2 text-zinc-300">
  89. <input type="checkbox"
  90. data-testid="firewall-poe-checkbox"
  91. @bind="_edit.Poe"/>
  92. PoE
  93. </label>
  94. </div>
  95. }
  96. else
  97. {
  98. <div class="flex gap-2 flex-wrap">
  99. @if (Firewall.Managed == true)
  100. {
  101. <span class="text-xs px-2 py-0.5 rounded bg-zinc-800 text-zinc-300"
  102. data-testid="firewall-managed-badge">
  103. Managed
  104. </span>
  105. }
  106. @if (Firewall.Poe == true)
  107. {
  108. <span class="text-xs px-2 py-0.5 rounded bg-zinc-800 text-zinc-300"
  109. data-testid="firewall-poe-badge">
  110. PoE
  111. </span>
  112. }
  113. </div>
  114. }
  115. </div>
  116. <!-- Ports -->
  117. <PortGroupEditor T="Firewall"
  118. Resource="Firewall"
  119. OnResourceChanged="r => Firewall = r"
  120. TestIdPrefix="firewall-ports"/>
  121. <ResourceTagEditor Resource="Firewall"
  122. TestIdPrefix="firewall"/>
  123. <ResourceLabelEditor Resource="Firewall"
  124. TestIdPrefix="firewall"/>
  125. <div class="md:col-span-2"
  126. data-testid="firewall-notes-section">
  127. <div class="text-zinc-400 mb-1">Notes</div>
  128. @if (_isEditing)
  129. {
  130. <MarkdownEditor @bind-Value="_edit.Notes"
  131. ShowActionButtons="false"
  132. TestIdPrefix="firewall-notes-editor"/>
  133. }
  134. else
  135. {
  136. <MarkdownViewer Value="@Firewall.Notes"
  137. ShowEditButton="false"
  138. TestIdPrefix="firewall-notes-viewer"/>
  139. }
  140. </div>
  141. </div>
  142. </div>
  143. <ConfirmModal IsOpen="_confirmDeleteOpen"
  144. IsOpenChanged="v => _confirmDeleteOpen = v"
  145. Title="Delete firewall"
  146. ConfirmText="Delete"
  147. ConfirmClass="bg-red-600 hover:bg-red-500"
  148. OnConfirm="DeleteServer"
  149. TestIdPrefix="Firewall">
  150. Are you sure you want to delete <strong>@Firewall.Name</strong>?
  151. </ConfirmModal>
  152. <StringValueModal IsOpen="_renameOpen"
  153. IsOpenChanged="v => _renameOpen = v"
  154. Title="Rename firewall"
  155. Description="Enter a new name for this firewall"
  156. Label="New firewall name"
  157. Value="@Firewall.Name"
  158. OnSubmit="HandleRenameSubmit"
  159. TestIdPrefix="firewall-rename"/>
  160. <StringValueModal IsOpen="_cloneOpen"
  161. IsOpenChanged="v => _cloneOpen = v"
  162. Title="Clone resource"
  163. Description="Enter a name for the cloned resource"
  164. Label="New resource name"
  165. Value="@($"{Firewall.Name}-copy")"
  166. OnSubmit="HandleCloneSubmit"
  167. TestIdPrefix="firewall-clone"/>
  168. @code {
  169. [Parameter] [EditorRequired] public Firewall Firewall { get; set; } = default!;
  170. bool _isEditing;
  171. FirewallEditModel _edit = new();
  172. void BeginEdit()
  173. {
  174. _edit = FirewallEditModel.From(Firewall);
  175. _isEditing = true;
  176. }
  177. async Task Save()
  178. {
  179. _isEditing = false;
  180. await UpdateUseCase.ExecuteAsync(
  181. Firewall.Name,
  182. _edit.Model,
  183. _edit.Managed,
  184. _edit.Poe,
  185. _edit.Notes);
  186. Firewall = await GetByNameUseCase.ExecuteAsync(Firewall.Name);
  187. }
  188. void Cancel()
  189. {
  190. _isEditing = false;
  191. }
  192. public class FirewallEditModel
  193. {
  194. public string? Model { get; set; }
  195. public bool? Managed { get; set; }
  196. public bool? Poe { get; set; }
  197. public string? Notes { get; set; }
  198. public static FirewallEditModel From(Firewall firewall)
  199. {
  200. return new FirewallEditModel
  201. {
  202. Model = firewall.Model,
  203. Managed = firewall.Managed,
  204. Poe = firewall.Poe,
  205. Notes = firewall.Notes
  206. };
  207. }
  208. }
  209. }
  210. @code {
  211. private bool _confirmDeleteOpen;
  212. [Parameter] public EventCallback<string> OnDeleted { get; set; }
  213. void ConfirmDelete()
  214. {
  215. _confirmDeleteOpen = true;
  216. }
  217. async Task DeleteServer()
  218. {
  219. _confirmDeleteOpen = false;
  220. await DeleteUseCase.ExecuteAsync(Firewall.Name);
  221. if (OnDeleted.HasDelegate)
  222. await OnDeleted.InvokeAsync(Firewall.Name);
  223. }
  224. }
  225. @code
  226. {
  227. bool _renameOpen;
  228. void OpenRename()
  229. {
  230. _renameOpen = true;
  231. }
  232. async Task HandleRenameSubmit(string newName)
  233. {
  234. await RenameUseCase.ExecuteAsync(Firewall.Name, newName);
  235. Nav.NavigateTo($"resources/hardware/{Uri.EscapeDataString(newName)}");
  236. }
  237. }
  238. @code
  239. {
  240. bool _cloneOpen;
  241. void OpenClone()
  242. {
  243. _cloneOpen = true;
  244. }
  245. async Task HandleCloneSubmit(string newName)
  246. {
  247. await CloneUseCase.ExecuteAsync(Firewall.Name, newName);
  248. Nav.NavigateTo($"resources/hardware/{Uri.EscapeDataString(newName)}");
  249. }
  250. }