UnifiedSetting.razor 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749
  1. @page "/setting/unifiedsetting"
  2. @using EasyTemplate.Service
  3. @using EasyTemplate.Tool.Entity.App
  4. @inject NozzleService nozzleService
  5. @inject IJSRuntime JSRuntime
  6. <div class="container-fluid">
  7. <div class="row">
  8. <!-- 油机列 -->
  9. <div class="col-md-3 border-end position-relative" style="height: calc(100vh - 100px);">
  10. <div class="d-flex justify-content-between align-items-center mb-3">
  11. <h5>油机列表</h5>
  12. <button class="btn btn-sm btn-primary" @onclick="ShowCreateEngineModal">
  13. <i class="fas fa-plus"></i>
  14. </button>
  15. </div>
  16. <!-- 显示所有油机选项 -->
  17. @* <div class="list-group mb-3">
  18. <button class="list-group-item list-group-item-action @(selectedEngineId == 0 ? "active" : "")"
  19. @onclick="() => SelectEngine(0)">
  20. 显示所有油机
  21. </button>
  22. </div> *@
  23. <!-- 油机列表 -->
  24. <div class="list-group flex-grow-1" style="max-height: 400px; overflow-y: auto; margin-bottom: 50px;">
  25. @if (paginatedEngines != null)
  26. {
  27. @foreach (var engine in paginatedEngines)
  28. {
  29. <div @onclick="() => SelectEngine(engine.EngineId)" class="list-group-item d-flex justify-content-between align-items-center py-1 @(selectedEngineId == engine.EngineId ? "active" : "")">
  30. <div style="cursor: pointer;">
  31. <div><strong>油机 @engine.EngineId</strong></div>
  32. </div>
  33. <div>
  34. <button class="btn btn-sm btn-info me-1 p-1" @onclick="() => ShowEditEngineModal(engine)">
  35. <i class="fas fa-edit"></i>
  36. </button>
  37. <button class="btn btn-sm btn-danger p-1" @onclick="() => DeleteEngine(engine.EngineId)">
  38. <i class="fas fa-trash"></i>
  39. </button>
  40. </div>
  41. </div>
  42. }
  43. }
  44. </div>
  45. <!-- 油机分页控件 -->
  46. @if (engines != null && engines.Count > pageSize)
  47. {
  48. <nav aria-label="油机分页" class="position-absolute" style="bottom: 50px; left: 0; right: 0;">
  49. <ul class="pagination pagination-sm justify-content-center mt-2 mb-0">
  50. <li class="page-item @(currentPage_Engine == 1 ? "disabled" : "")">
  51. <a class="page-link" @onclick="() => ChangePage_Engine(currentPage_Engine - 1)" tabindex="-1">上一页</a>
  52. </li>
  53. @for (int i = 1; i <= totalPages_Engine; i++)
  54. {
  55. <li class="page-item @(currentPage_Engine == i ? "active" : "")">
  56. <a class="page-link" @onclick="() => ChangePage_Engine(i)">@i</a>
  57. </li>
  58. }
  59. <li class="page-item @(currentPage_Engine == totalPages_Engine ? "disabled" : "")">
  60. <a class="page-link" @onclick="() => ChangePage_Engine(currentPage_Engine + 1)">下一页</a>
  61. </li>
  62. </ul>
  63. </nav>
  64. }
  65. </div>
  66. <!-- 主板列 -->
  67. <div class="col-md-3 border-end position-relative" style="height: calc(100vh - 100px);">
  68. <div class="d-flex justify-content-between align-items-center mb-3">
  69. <h5>主板列表</h5>
  70. <button class="btn btn-sm btn-primary" @onclick="ShowCreateBoardModal" disabled="@(selectedEngineId == 0)">
  71. <i class="fas fa-plus"></i>
  72. </button>
  73. </div>
  74. @* <!-- 显示所有主板选项 -->
  75. <div class="list-group mb-3">
  76. <button class="list-group-item list-group-item-action @(selectedBoardId == 0 ? "active" : "")"
  77. @onclick="() => SelectBoard(0)" disabled="@(selectedEngineId == 0)">
  78. 显示所有主板
  79. </button>
  80. </div> *@
  81. <!-- 主板列表 -->
  82. <div class="list-group flex-grow-1" style="max-height: 400px; overflow-y: auto; margin-bottom: 50px;">
  83. @if (paginatedBoards != null)
  84. {
  85. @foreach (var board in paginatedBoards)
  86. {
  87. <div @onclick="() => SelectBoard(board.BoardId)" class="list-group-item d-flex justify-content-between align-items-center py-1 @(selectedBoardId == board.BoardId ? "active" : "")">
  88. <div style="cursor: pointer;">
  89. <div><strong>主板 @board.BoardId</strong></div>
  90. </div>
  91. <div>
  92. <button class="btn btn-sm btn-info me-1 p-1" @onclick="() => ShowEditBoardModal(board)">
  93. <i class="fas fa-edit"></i>
  94. </button>
  95. <button class="btn btn-sm btn-danger p-1" @onclick="() => DeleteBoard(board.BoardId)">
  96. <i class="fas fa-trash"></i>
  97. </button>
  98. </div>
  99. </div>
  100. }
  101. }
  102. </div>
  103. <!-- 主板分页控件 -->
  104. @if (filteredBoards != null && filteredBoards.Count > pageSize)
  105. {
  106. <nav aria-label="主板分页" class="position-absolute" style="bottom: 50px; left: 0; right: 0;">
  107. <ul class="pagination pagination-sm justify-content-center mt-2 mb-0">
  108. <li class="page-item @(currentPage_Board == 1 ? "disabled" : "")">
  109. <a class="page-link" @onclick="() => ChangePage_Board(currentPage_Board - 1)" tabindex="-1">上一页</a>
  110. </li>
  111. @for (int i = 1; i <= totalPages_Board; i++)
  112. {
  113. <li class="page-item @(currentPage_Board == i ? "active" : "")">
  114. <a class="page-link" @onclick="() => ChangePage_Board(i)">@i</a>
  115. </li>
  116. }
  117. <li class="page-item @(currentPage_Board == totalPages_Board ? "disabled" : "")">
  118. <a class="page-link" @onclick="() => ChangePage_Board(currentPage_Board + 1)">下一页</a>
  119. </li>
  120. </ul>
  121. </nav>
  122. }
  123. </div>
  124. <!-- 油枪列 -->
  125. <div class="col-md-6 position-relative" style="height: calc(100vh - 100px);">
  126. <div class="d-flex justify-content-between align-items-center mb-3">
  127. <h5>油枪列表</h5>
  128. <button class="btn btn-sm btn-primary" @onclick="ShowCreateNozzleModal" disabled="@(selectedBoardId == 0)">
  129. <i class="fas fa-plus"></i>
  130. </button>
  131. </div>
  132. <!-- 油枪列表 -->
  133. <div class="list-group flex-grow-1" style="margin-bottom: 50px;">
  134. @if (paginatedNozzles != null)
  135. {
  136. @foreach (var nozzle in paginatedNozzles)
  137. {
  138. <div class="list-group-item d-flex justify-content-between align-items-center py-1">
  139. <div>
  140. <div><strong>油枪 @nozzle.NozzleId</strong></div>
  141. <small>节点: @nozzle.node | 产品: @nozzle.product | 状态: @(nozzle.disable ? "禁用" : "启用") | MN码: @nozzle.mncode</small>
  142. </div>
  143. <div>
  144. <button class="btn btn-sm btn-info me-1 p-1" @onclick="() => ShowEditNozzleModal(nozzle)">
  145. <i class="fas fa-edit"></i>
  146. </button>
  147. <button class="btn btn-sm btn-danger p-1" @onclick="() => DeleteNozzle(nozzle.NozzleId)">
  148. <i class="fas fa-trash"></i>
  149. </button>
  150. </div>
  151. </div>
  152. }
  153. }
  154. </div>
  155. <!-- 油枪分页控件 -->
  156. @if (filteredNozzles != null && filteredNozzles.Count > pageSize)
  157. {
  158. <nav aria-label="油枪分页" class="position-absolute" style="bottom: 50px; left: 0; right: 0;">
  159. <ul class="pagination pagination-sm justify-content-center mt-2 mb-0">
  160. <li class="page-item @(currentPage_Nozzle == 1 ? "disabled" : "")">
  161. <a class="page-link" @onclick="() => ChangePage_Nozzle(currentPage_Nozzle - 1)" tabindex="-1">上一页</a>
  162. </li>
  163. @for (int i = 1; i <= totalPages_Nozzle; i++)
  164. {
  165. <li class="page-item @(currentPage_Nozzle == i ? "active" : "")">
  166. <a class="page-link" @onclick="() => ChangePage_Nozzle(i)">@i</a>
  167. </li>
  168. }
  169. <li class="page-item @(currentPage_Nozzle == totalPages_Nozzle ? "disabled" : "")">
  170. <a class="page-link" @onclick="() => ChangePage_Nozzle(currentPage_Nozzle + 1)">下一页</a>
  171. </li>
  172. </ul>
  173. </nav>
  174. }
  175. </div>
  176. </div>
  177. </div>
  178. <!-- 油机模态框 -->
  179. <div class="modal @(showEngineModal ? "show" : "")" tabindex="-1" style="display: @(showEngineModal ? "block" : "none"); background-color: rgba(0,0,0,0.5);">
  180. <div class="modal-dialog">
  181. <div class="modal-content">
  182. <div class="modal-header">
  183. <h5 class="modal-title">@engineModalTitle</h5>
  184. <button type="button" class="btn-close" @onclick="CloseEngineModal"></button>
  185. </div>
  186. <div class="modal-body">
  187. <div class="mb-3">
  188. <label class="form-label">油机ID</label>
  189. <input class="form-control" @bind="currentEngine.EngineId" disabled="@isEngineEditMode" />
  190. </div>
  191. <div class="mb-3">
  192. <label class="form-label">工厂ID</label>
  193. <input class="form-control" @bind="currentEngine.factory_id" />
  194. </div>
  195. <div class="mb-3">
  196. <label class="form-label">工厂ID 2</label>
  197. <input class="form-control" @bind="currentEngine.factory_id_2" />
  198. </div>
  199. </div>
  200. <div class="modal-footer">
  201. <button type="button" class="btn btn-secondary" @onclick="CloseEngineModal">取消</button>
  202. <button type="button" class="btn btn-primary" @onclick="SaveEngine">保存</button>
  203. </div>
  204. </div>
  205. </div>
  206. </div>
  207. <!-- 主板模态框 -->
  208. <div class="modal @(showBoardModal ? "show" : "")" tabindex="-1" style="display: @(showBoardModal ? "block" : "none"); background-color: rgba(0,0,0,0.5);">
  209. <div class="modal-dialog">
  210. <div class="modal-content">
  211. <div class="modal-header">
  212. <h5 class="modal-title">@boardModalTitle</h5>
  213. <button type="button" class="btn-close" @onclick="CloseBoardModal"></button>
  214. </div>
  215. <div class="modal-body">
  216. <div class="mb-3">
  217. <label class="form-label">所属油机ID</label>
  218. <select class="form-select" @bind="currentBoard.EngineId" disabled="@isBoardEditMode">
  219. <option value="0">请选择油机</option>
  220. @if (engines != null)
  221. {
  222. @foreach (var engine in engines)
  223. {
  224. <option value="@engine.EngineId">@engine.EngineId</option>
  225. }
  226. }
  227. </select>
  228. </div>
  229. <div class="mb-3">
  230. <label class="form-label">主板ID</label>
  231. <input class="form-control" @bind="currentBoard.BoardId" disabled="@isBoardEditMode" />
  232. </div>
  233. <div class="mb-3">
  234. <label class="form-label">IP地址</label>
  235. <input class="form-control" @bind="currentBoard.ip" />
  236. </div>
  237. </div>
  238. <div class="modal-footer">
  239. <button type="button" class="btn btn-secondary" @onclick="CloseBoardModal">取消</button>
  240. <button type="button" class="btn btn-primary" @onclick="SaveBoard">保存</button>
  241. </div>
  242. </div>
  243. </div>
  244. </div>
  245. <!-- 油枪模态框 -->
  246. <div class="modal @(showNozzleModal ? "show" : "")" tabindex="-1" style="display: @(showNozzleModal ? "block" : "none"); background-color: rgba(0,0,0,0.5);">
  247. <div class="modal-dialog">
  248. <div class="modal-content">
  249. <div class="modal-header">
  250. <h5 class="modal-title">@nozzleModalTitle</h5>
  251. <button type="button" class="btn-close" @onclick="CloseNozzleModal"></button>
  252. </div>
  253. <div class="modal-body">
  254. <div class="mb-3">
  255. <label class="form-label">油枪ID</label>
  256. <input type="number" class="form-control" @bind="currentNozzle.NozzleId" disabled="@isNozzleEditMode" />
  257. </div>
  258. <div class="mb-3">
  259. <label class="form-label">所属主板ID</label>
  260. <select class="form-select" @bind="currentNozzle.BoardId" disabled="@isNozzleEditMode">
  261. <option value="0">请选择主板</option>
  262. @if (boards != null)
  263. {
  264. @foreach (var board in boards)
  265. {
  266. <option value="@board.BoardId">@board.BoardId (油机:@board.EngineId)</option>
  267. }
  268. }
  269. </select>
  270. </div>
  271. <div class="mb-3">
  272. <label class="form-label">节点</label>
  273. <input type="number" class="form-control" @bind="currentNozzle.node" />
  274. </div>
  275. <div class="mb-3">
  276. <label class="form-label">产品</label>
  277. <input type="number" class="form-control" @bind="currentNozzle.product" />
  278. </div>
  279. <div class="mb-3 form-check">
  280. <input type="checkbox" class="form-check-input" @bind="currentNozzle.disable" />
  281. <label class="form-check-label">禁用</label>
  282. </div>
  283. <div class="mb-3">
  284. <label class="form-label">MN码</label>
  285. <input class="form-control" @bind="currentNozzle.mncode" />
  286. </div>
  287. </div>
  288. <div class="modal-footer">
  289. <button type="button" class="btn btn-secondary" @onclick="CloseNozzleModal">取消</button>
  290. <button type="button" class="btn btn-primary" @onclick="SaveNozzle">保存</button>
  291. </div>
  292. </div>
  293. </div>
  294. </div>
  295. @code {
  296. // 数据列表
  297. private List<TEngine> engines = new();
  298. private List<TBoard> allBoards = new();
  299. private List<TBoard> filteredBoards = new();
  300. private List<TNozzle> allNozzles = new();
  301. private List<TNozzle> filteredNozzles = new();
  302. private List<TBoard> boards = new(); // 用于下拉选择
  303. // 分页相关
  304. private const int pageSize = 8; // 每页显示条数
  305. private int currentPage_Engine = 1;
  306. private int totalPages_Engine => engines == null ? 0 : (int)Math.Ceiling((double)engines.Count / pageSize);
  307. private List<TEngine> paginatedEngines => engines?.Skip((currentPage_Engine - 1) * pageSize).Take(pageSize).ToList() ?? new List<TEngine>();
  308. private int currentPage_Board = 1;
  309. private int totalPages_Board => filteredBoards == null ? 0 : (int)Math.Ceiling((double)filteredBoards.Count / pageSize);
  310. private List<TBoard> paginatedBoards => filteredBoards?.Skip((currentPage_Board - 1) * pageSize).Take(pageSize).ToList() ?? new List<TBoard>();
  311. private int currentPage_Nozzle = 1;
  312. private int totalPages_Nozzle => filteredNozzles == null ? 0 : (int)Math.Ceiling((double)filteredNozzles.Count / pageSize);
  313. private List<TNozzle> paginatedNozzles => filteredNozzles?.Skip((currentPage_Nozzle - 1) * pageSize).Take(pageSize).ToList() ?? new List<TNozzle>();
  314. // 当前选中项
  315. private int selectedEngineId = 0;
  316. private int selectedBoardId = 0;
  317. // 油机相关
  318. private TEngine currentEngine = new();
  319. private string engineModalTitle = "";
  320. private bool isEngineEditMode = false;
  321. private bool showEngineModal = false;
  322. // 主板相关
  323. private TBoard currentBoard = new();
  324. private string boardModalTitle = "";
  325. private bool isBoardEditMode = false;
  326. private bool showBoardModal = false;
  327. // 油枪相关
  328. private TNozzle currentNozzle = new();
  329. private string nozzleModalTitle = "";
  330. private bool isNozzleEditMode = false;
  331. private bool showNozzleModal = false;
  332. protected override async Task OnInitializedAsync()
  333. {
  334. await LoadAllData();
  335. }
  336. private async Task LoadAllData()
  337. {
  338. try
  339. {
  340. // 加载所有数据
  341. engines = await nozzleService.GetEnginesWithDetailsAsync();
  342. allBoards = await nozzleService.GetBoardsAsync();
  343. allNozzles = await nozzleService.GetNozzlesAsync();
  344. boards = new List<TBoard>(allBoards); // 用于下拉选择
  345. // 默认显示所有数据
  346. //filteredBoards = new List<TBoard>(allBoards);
  347. //filteredNozzles = new List<TNozzle>(allNozzles);
  348. // 重置分页
  349. currentPage_Engine = 1;
  350. currentPage_Board = 1;
  351. currentPage_Nozzle = 1;
  352. StateHasChanged();
  353. }
  354. catch (Exception ex)
  355. {
  356. await JSRuntime.InvokeVoidAsync("alert", $"加载数据失败: {ex.Message}");
  357. }
  358. }
  359. // 分页切换方法
  360. private void ChangePage_Engine(int page)
  361. {
  362. if (page >= 1 && page <= totalPages_Engine)
  363. {
  364. currentPage_Engine = page;
  365. StateHasChanged();
  366. }
  367. }
  368. private void ChangePage_Board(int page)
  369. {
  370. if (page >= 1 && page <= totalPages_Board)
  371. {
  372. currentPage_Board = page;
  373. StateHasChanged();
  374. }
  375. }
  376. private void ChangePage_Nozzle(int page)
  377. {
  378. if (page >= 1 && page <= totalPages_Nozzle)
  379. {
  380. currentPage_Nozzle = page;
  381. StateHasChanged();
  382. }
  383. }
  384. // 油机选择
  385. private void SelectEngine(int engineId)
  386. {
  387. selectedEngineId = engineId;
  388. selectedBoardId = 0; // 重置主板选择
  389. if (engineId == 0)
  390. {
  391. // 显示所有主板和油枪
  392. //filteredBoards = new List<TBoard>(allBoards);
  393. //filteredNozzles = new List<TNozzle>(allNozzles);
  394. }
  395. else
  396. {
  397. // 只显示选定油机的主板
  398. filteredBoards = allBoards.Where(b => b.EngineId == engineId).ToList();
  399. // 显示这些主板下的所有油枪
  400. var boardIds = filteredBoards.Select(b => b.BoardId).ToList();
  401. filteredNozzles = allNozzles.Where(n => boardIds.Contains(n.BoardId)).ToList();
  402. }
  403. // 重置分页
  404. currentPage_Board = 1;
  405. currentPage_Nozzle = 1;
  406. StateHasChanged();
  407. }
  408. // 主板选择
  409. private void SelectBoard(int boardId)
  410. {
  411. selectedBoardId = boardId;
  412. if (boardId == 0)
  413. {
  414. // 如果选择了"显示所有主板",则显示当前筛选条件下的所有油枪
  415. if (selectedEngineId == 0)
  416. {
  417. filteredNozzles = new List<TNozzle>(allNozzles);
  418. }
  419. else
  420. {
  421. var boardIds = filteredBoards.Select(b => b.BoardId).ToList();
  422. filteredNozzles = allNozzles.Where(n => boardIds.Contains(n.BoardId)).ToList();
  423. }
  424. }
  425. else
  426. {
  427. // 只显示选定主板的油枪
  428. filteredNozzles = allNozzles.Where(n => n.BoardId == boardId).ToList();
  429. }
  430. // 重置油枪分页
  431. currentPage_Nozzle = 1;
  432. StateHasChanged();
  433. }
  434. // ===== 油机操作 =====
  435. private void ShowCreateEngineModal()
  436. {
  437. currentEngine = new TEngine();
  438. engineModalTitle = "添加油机";
  439. isEngineEditMode = false;
  440. showEngineModal = true;
  441. StateHasChanged();
  442. }
  443. private void ShowEditEngineModal(TEngine engine)
  444. {
  445. currentEngine = new TEngine
  446. {
  447. EngineId = engine.EngineId,
  448. factory_id = engine.factory_id,
  449. factory_id_2 = engine.factory_id_2
  450. };
  451. engineModalTitle = "编辑油机";
  452. isEngineEditMode = true;
  453. showEngineModal = true;
  454. StateHasChanged();
  455. }
  456. private void CloseEngineModal()
  457. {
  458. showEngineModal = false;
  459. StateHasChanged();
  460. }
  461. private async Task SaveEngine()
  462. {
  463. try
  464. {
  465. if (isEngineEditMode)
  466. {
  467. await nozzleService.UpdateEngineAsync(currentEngine);
  468. }
  469. else
  470. {
  471. await nozzleService.CreateEngineAsync(currentEngine);
  472. }
  473. CloseEngineModal();
  474. await LoadAllData();
  475. // 重新选择当前油机以刷新视图
  476. if (selectedEngineId > 0)
  477. {
  478. SelectEngine(selectedEngineId);
  479. }
  480. }
  481. catch (Exception ex)
  482. {
  483. await JSRuntime.InvokeVoidAsync("alert", $"操作失败: {ex.Message}");
  484. }
  485. }
  486. private async Task DeleteEngine(int id)
  487. {
  488. var confirmed = await JSRuntime.InvokeAsync<bool>("confirm", "确定要删除这个油机吗?这将同时删除其下属的所有主板和油枪!");
  489. if (confirmed)
  490. {
  491. try
  492. {
  493. await nozzleService.DeleteEngineAsync(id);
  494. await LoadAllData();
  495. // 如果删除的是当前选中的油机,重置选择
  496. if (selectedEngineId == id)
  497. {
  498. SelectEngine(0);
  499. }
  500. }
  501. catch (Exception ex)
  502. {
  503. await JSRuntime.InvokeVoidAsync("alert", $"删除失败: {ex.Message}");
  504. }
  505. }
  506. }
  507. // ===== 主板操作 =====
  508. private void ShowCreateBoardModal()
  509. {
  510. if (selectedEngineId == 0)
  511. {
  512. JSRuntime.InvokeVoidAsync("alert", "请先选择一个油机");
  513. return;
  514. }
  515. currentBoard = new TBoard { EngineId = selectedEngineId };
  516. boardModalTitle = "添加主板";
  517. isBoardEditMode = false;
  518. showBoardModal = true;
  519. StateHasChanged();
  520. }
  521. private void ShowEditBoardModal(TBoard board)
  522. {
  523. currentBoard = new TBoard
  524. {
  525. BoardId = board.BoardId,
  526. EngineId = board.EngineId,
  527. ip = board.ip
  528. };
  529. boardModalTitle = "编辑主板";
  530. isBoardEditMode = true;
  531. showBoardModal = true;
  532. StateHasChanged();
  533. }
  534. private void CloseBoardModal()
  535. {
  536. showBoardModal = false;
  537. StateHasChanged();
  538. }
  539. private async Task SaveBoard()
  540. {
  541. if (currentBoard.EngineId <= 0)
  542. {
  543. await JSRuntime.InvokeVoidAsync("alert", "请选择有效的油机ID");
  544. return;
  545. }
  546. try
  547. {
  548. if (isBoardEditMode)
  549. {
  550. await nozzleService.UpdateBoardAsync(currentBoard);
  551. }
  552. else
  553. {
  554. await nozzleService.CreateBoardAsync(currentBoard);
  555. }
  556. CloseBoardModal();
  557. await LoadAllData();
  558. // 重新选择当前筛选条件
  559. if (selectedEngineId > 0)
  560. {
  561. SelectEngine(selectedEngineId);
  562. }
  563. }
  564. catch (Exception ex)
  565. {
  566. await JSRuntime.InvokeVoidAsync("alert", $"操作失败: {ex.Message}");
  567. }
  568. }
  569. private async Task DeleteBoard(int id)
  570. {
  571. var confirmed = await JSRuntime.InvokeAsync<bool>("confirm", "确定要删除这个主板吗?这将同时删除其下属的所有油枪!");
  572. if (confirmed)
  573. {
  574. try
  575. {
  576. await nozzleService.DeleteBoardAsync(id);
  577. await LoadAllData();
  578. // 如果删除的是当前选中的主板,重置选择
  579. if (selectedBoardId == id)
  580. {
  581. SelectBoard(0);
  582. }
  583. }
  584. catch (Exception ex)
  585. {
  586. await JSRuntime.InvokeVoidAsync("alert", $"删除失败: {ex.Message}");
  587. }
  588. }
  589. }
  590. // ===== 油枪操作 =====
  591. private void ShowCreateNozzleModal()
  592. {
  593. if (selectedBoardId == 0)
  594. {
  595. JSRuntime.InvokeVoidAsync("alert", "请先选择一个主板");
  596. return;
  597. }
  598. currentNozzle = new TNozzle { BoardId = selectedBoardId };
  599. nozzleModalTitle = "添加油枪";
  600. isNozzleEditMode = false;
  601. showNozzleModal = true;
  602. StateHasChanged();
  603. }
  604. private void ShowEditNozzleModal(TNozzle nozzle)
  605. {
  606. currentNozzle = new TNozzle
  607. {
  608. NozzleId = nozzle.NozzleId,
  609. BoardId = nozzle.BoardId,
  610. node = nozzle.node,
  611. product = nozzle.product,
  612. disable = nozzle.disable,
  613. mncode = nozzle.mncode
  614. };
  615. nozzleModalTitle = "编辑油枪";
  616. isNozzleEditMode = true;
  617. showNozzleModal = true;
  618. StateHasChanged();
  619. }
  620. private void CloseNozzleModal()
  621. {
  622. showNozzleModal = false;
  623. StateHasChanged();
  624. }
  625. private async Task SaveNozzle()
  626. {
  627. if (currentNozzle.BoardId <= 0)
  628. {
  629. await JSRuntime.InvokeVoidAsync("alert", "请选择有效的主板ID");
  630. return;
  631. }
  632. try
  633. {
  634. if (isNozzleEditMode)
  635. {
  636. await nozzleService.UpdateNozzleAsync(currentNozzle);
  637. }
  638. else
  639. {
  640. await nozzleService.CreateNozzleAsync(currentNozzle);
  641. }
  642. CloseNozzleModal();
  643. await LoadAllData();
  644. // 重新应用当前筛选
  645. if (selectedBoardId > 0)
  646. {
  647. SelectBoard(selectedBoardId);
  648. }
  649. else if (selectedEngineId > 0)
  650. {
  651. SelectEngine(selectedEngineId);
  652. }
  653. }
  654. catch (Exception ex)
  655. {
  656. await JSRuntime.InvokeVoidAsync("alert", $"操作失败: {ex.Message}");
  657. }
  658. }
  659. private async Task DeleteNozzle(int id)
  660. {
  661. var confirmed = await JSRuntime.InvokeAsync<bool>("confirm", "确定要删除这个油枪吗?");
  662. if (confirmed)
  663. {
  664. try
  665. {
  666. await nozzleService.DeleteNozzleAsync(id);
  667. await LoadAllData();
  668. // 重新应用当前筛选
  669. if (selectedBoardId > 0)
  670. {
  671. SelectBoard(selectedBoardId);
  672. }
  673. else if (selectedEngineId > 0)
  674. {
  675. SelectEngine(selectedEngineId);
  676. }
  677. }
  678. catch (Exception ex)
  679. {
  680. await JSRuntime.InvokeVoidAsync("alert", $"删除失败: {ex.Message}");
  681. }
  682. }
  683. }
  684. }