@profoundlogic/coderflow-server 0.5.3 → 0.5.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (179) hide show
  1. package/dist/coder-server.js +1 -1
  2. package/dist/config.js +1 -1
  3. package/dist/lib/agent-keepalive.js +1 -1
  4. package/dist/lib/agent-models.js +1 -1
  5. package/dist/lib/api-keys.js +1 -1
  6. package/dist/lib/apiKeys.js +1 -1
  7. package/dist/lib/app-server-ports.js +1 -1
  8. package/dist/lib/auto-judge.js +1 -1
  9. package/dist/lib/automation-service.js +1 -1
  10. package/dist/lib/basic-auth.js +1 -1
  11. package/dist/lib/bindings.js +1 -1
  12. package/dist/lib/build-history.js +1 -1
  13. package/dist/lib/build-output-service.js +1 -1
  14. package/dist/lib/build-scheduler.js +1 -1
  15. package/dist/lib/build-service.js +1 -1
  16. package/dist/lib/ca-certificates.js +1 -1
  17. package/dist/lib/claude-oauth-refresh.js +1 -1
  18. package/dist/lib/cli/build.js +1 -1
  19. package/dist/lib/cli/cleanup-users.js +1 -1
  20. package/dist/lib/cli/config-command.js +1 -1
  21. package/dist/lib/cli/config.js +1 -1
  22. package/dist/lib/cli/create-user.js +1 -1
  23. package/dist/lib/cli/grant-admin.js +1 -1
  24. package/dist/lib/cli/init.js +1 -1
  25. package/dist/lib/cli/jira.js +1 -1
  26. package/dist/lib/cli/license.js +1 -1
  27. package/dist/lib/cli/list-roles.js +1 -1
  28. package/dist/lib/cli/list-users.js +1 -1
  29. package/dist/lib/cli/server-manager.js +1 -1
  30. package/dist/lib/cli/set-password.js +1 -1
  31. package/dist/lib/config-migration.js +1 -1
  32. package/dist/lib/container-credential-sync.js +1 -1
  33. package/dist/lib/container-tokens.js +1 -1
  34. package/dist/lib/data-dir.js +1 -1
  35. package/dist/lib/deployment-history.js +1 -1
  36. package/dist/lib/deployment-service.js +1 -1
  37. package/dist/lib/docker-utils.js +1 -1
  38. package/dist/lib/email.js +1 -1
  39. package/dist/lib/emailTemplates.js +1 -1
  40. package/dist/lib/entitlement.js +1 -1
  41. package/dist/lib/external-connections.js +1 -1
  42. package/dist/lib/fetch-utils.js +1 -1
  43. package/dist/lib/git-commit-details-route.js +1 -1
  44. package/dist/lib/git-history-diff-guardrails.js +1 -1
  45. package/dist/lib/git-provider-service.js +1 -1
  46. package/dist/lib/git-provider-setup/github-setup-handler.js +1 -1
  47. package/dist/lib/git-provider-setup/index.js +1 -1
  48. package/dist/lib/git-provider-setup/setup-factory.js +1 -1
  49. package/dist/lib/git-provider-setup/setup-interface.js +1 -1
  50. package/dist/lib/git-providers/azure-devops-provider.js +1 -1
  51. package/dist/lib/git-providers/github-app-provider.js +1 -1
  52. package/dist/lib/git-providers/index.js +1 -1
  53. package/dist/lib/git-providers/provider-factory.js +1 -1
  54. package/dist/lib/git-providers/provider-interface.js +1 -1
  55. package/dist/lib/github-urls.js +1 -1
  56. package/dist/lib/group-objective-linking.js +1 -1
  57. package/dist/lib/ibmi-sync.js +1 -1
  58. package/dist/lib/jira-client.js +1 -1
  59. package/dist/lib/judge-blinding.js +1 -1
  60. package/dist/lib/logger.js +1 -1
  61. package/dist/lib/migration-to-scoped-rbac.js +1 -1
  62. package/dist/lib/model-fetcher.js +1 -1
  63. package/dist/lib/notifications.js +1 -1
  64. package/dist/lib/objective-context.js +1 -1
  65. package/dist/lib/oidc-auth.js +1 -1
  66. package/dist/lib/oidc-device-flow.js +1 -1
  67. package/dist/lib/passwordTokens.js +1 -1
  68. package/dist/lib/permission-resolver.js +1 -1
  69. package/dist/lib/pin-cascade.js +1 -1
  70. package/dist/lib/provider-accounts.js +1 -1
  71. package/dist/lib/provider-oauth.js +1 -1
  72. package/dist/lib/provider-profile.js +1 -1
  73. package/dist/lib/provider-token-refresh.js +1 -1
  74. package/dist/lib/rbac-user-state.js +1 -1
  75. package/dist/lib/request-url.js +1 -1
  76. package/dist/lib/rewind.js +1 -1
  77. package/dist/lib/role-definitions.js +1 -1
  78. package/dist/lib/roles.js +1 -1
  79. package/dist/lib/secrets.js +1 -1
  80. package/dist/lib/setup-repo-git-auth.js +1 -1
  81. package/dist/lib/state-capture.js +1 -1
  82. package/dist/lib/static-files.js +1 -1
  83. package/dist/lib/task-name-format.js +1 -1
  84. package/dist/lib/task-name-generator.js +1 -1
  85. package/dist/lib/task-source-metadata.js +1 -1
  86. package/dist/lib/teams.js +1 -1
  87. package/dist/lib/user-git-oauth.js +1 -1
  88. package/dist/lib/user-git-tokens.js +1 -1
  89. package/dist/lib/users.js +1 -1
  90. package/dist/middleware/requireAuth.js +1 -1
  91. package/dist/middleware/requireInit.js +1 -1
  92. package/dist/middleware/requirePermission.js +1 -1
  93. package/dist/package.json +1 -1
  94. package/dist/playwright.config.js +1 -1
  95. package/dist/routes/apiKeys.js +1 -1
  96. package/dist/routes/auth-oidc.js +1 -1
  97. package/dist/routes/auth.js +1 -1
  98. package/dist/routes/automations.js +1 -1
  99. package/dist/routes/bindings.js +1 -1
  100. package/dist/routes/build.js +1 -1
  101. package/dist/routes/containers.js +1 -1
  102. package/dist/routes/deploy-task.js +1 -1
  103. package/dist/routes/environment-management.js +1 -1
  104. package/dist/routes/environments.js +1 -1
  105. package/dist/routes/external-skills.js +1 -1
  106. package/dist/routes/git-credentials.js +1 -1
  107. package/dist/routes/git-oauth.js +1 -1
  108. package/dist/routes/git-provider-setup.js +1 -1
  109. package/dist/routes/health.js +1 -1
  110. package/dist/routes/jira.js +1 -1
  111. package/dist/routes/objective-management.js +1 -1
  112. package/dist/routes/password.js +1 -1
  113. package/dist/routes/prompt.js +1 -1
  114. package/dist/routes/provider-auth.js +1 -1
  115. package/dist/routes/qa.js +1 -1
  116. package/dist/routes/roles.js +1 -1
  117. package/dist/routes/settings.js +1 -1
  118. package/dist/routes/skill-management.js +1 -1
  119. package/dist/routes/skills.js +1 -1
  120. package/dist/routes/tasks.js +1 -1
  121. package/dist/routes/teams.js +1 -1
  122. package/dist/routes/templates.js +1 -1
  123. package/dist/routes/test-task.js +1 -1
  124. package/dist/routes/test.js +1 -1
  125. package/dist/routes/users.js +1 -1
  126. package/dist/routes/visualizations.js +1 -1
  127. package/dist/scripts/create-user.js +1 -1
  128. package/dist/scripts/migrate-config-to-data-dir.js +1 -1
  129. package/dist/start.js +1 -1
  130. package/dist/web-ui/public/activity-detail-modal.js +1 -1
  131. package/dist/web-ui/public/activity-feed.js +1 -1
  132. package/dist/web-ui/public/activity-formatters.js +1 -1
  133. package/dist/web-ui/public/agent-event-parser.js +1 -1
  134. package/dist/web-ui/public/app.js +1 -1
  135. package/dist/web-ui/public/approve-dialog.js +1 -1
  136. package/dist/web-ui/public/automation-links.js +1 -1
  137. package/dist/web-ui/public/automation-schedule.js +1 -1
  138. package/dist/web-ui/public/comments-widget.js +1 -1
  139. package/dist/web-ui/public/diff-utils.js +1 -1
  140. package/dist/web-ui/public/environments.html +6 -6
  141. package/dist/web-ui/public/environments.js +1 -1
  142. package/dist/web-ui/public/feedback-widget.css +409 -0
  143. package/dist/web-ui/public/feedback-widget.js +1 -1
  144. package/dist/web-ui/public/git-history-lazy-utils.js +1 -1
  145. package/dist/web-ui/public/git-history.html +4 -4
  146. package/dist/web-ui/public/git-history.js +1 -1
  147. package/dist/web-ui/public/git-status.js +1 -1
  148. package/dist/web-ui/public/index.html +4 -4
  149. package/dist/web-ui/public/index.js +1 -1
  150. package/dist/web-ui/public/login.html +2 -2
  151. package/dist/web-ui/public/login.js +1 -1
  152. package/dist/web-ui/public/markdown-editor.js +1 -1
  153. package/dist/web-ui/public/markdown-file-editor.js +1 -1
  154. package/dist/web-ui/public/modal-maximize.js +1 -1
  155. package/dist/web-ui/public/notifications.js +1 -1
  156. package/dist/web-ui/public/pr-dialog.js +1 -1
  157. package/dist/web-ui/public/roles.html +1 -1
  158. package/dist/web-ui/public/roles.js +1 -1
  159. package/dist/web-ui/public/server-health.js +1 -1
  160. package/dist/web-ui/public/settings.html +4 -4
  161. package/dist/web-ui/public/settings.js +1 -1
  162. package/dist/web-ui/public/setup-password.html +2 -2
  163. package/dist/web-ui/public/setup-password.js +1 -1
  164. package/dist/web-ui/public/skills.html +6 -6
  165. package/dist/web-ui/public/skills.js +1 -1
  166. package/dist/web-ui/public/sse-client.js +1 -1
  167. package/dist/web-ui/public/sse-shared-worker.js +1 -1
  168. package/dist/web-ui/public/styles.css +2975 -2022
  169. package/dist/web-ui/public/task.html +101 -5
  170. package/dist/web-ui/public/task.js +1 -1
  171. package/dist/web-ui/public/teams.html +1 -1
  172. package/dist/web-ui/public/teams.js +1 -1
  173. package/dist/web-ui/public/terminal.html +2 -2
  174. package/dist/web-ui/public/terminal.js +1 -1
  175. package/dist/web-ui/public/theme.js +1 -1
  176. package/dist/web-ui/public/users.html +2 -2
  177. package/dist/web-ui/public/users.js +1 -1
  178. package/dist/web-ui/public/variant-grouping.js +1 -1
  179. package/package.json +1 -1
@@ -22,7 +22,7 @@
22
22
  }
23
23
  })();
24
24
  </script>
25
- <link rel="stylesheet" href="styles.css?v=83">
25
+ <link rel="stylesheet" href="styles.css?v=84">
26
26
  <script type="module" src="teams.js"></script>
27
27
  <!-- Dev QA shortcut: Ctrl+Shift+Q to launch current page in QA mode -->
28
28
  </head>
@@ -1 +1 @@
1
- import{API,Utils,initializeAdminMenu}from'./app.js';import{initTheme,mountThemeToggle}from'./theme.js';const state={'teams':[],'users':[],'roles':[],'environments':[],'teamBindings':[],'currentUser':null,'editingTeamId':null,'bindingsTeamId':null,'deletingTeamId':null,'modalSnapshot':null};document['addEventListener']('DOMContentLoaded',async()=>{initTheme(),mountThemeToggle();try{state['currentUser']=await API['getCurrentUser']();}catch{window['location']['href']='login.html';return;}await initializeAdminMenu(state['currentUser']);const _0x1eabfd=state['currentUser'];if(!_0x1eabfd['hasAllPermissions']&&!_0x1eabfd['canManageTeams']&&!_0x1eabfd['hasEnvironmentBindings']&&!_0x1eabfd['hasEnvironmentAdmin']){Utils['showToast']('Access\x20denied:\x20You\x20do\x20not\x20have\x20permission\x20to\x20view\x20teams','error'),setTimeout(()=>{window['location']['href']='index.html';},0x7d0);return;}(_0x1eabfd['hasAllPermissions']||_0x1eabfd['canManageTeams'])&&(document['getElementById']('create-team-btn')['hidden']=![]),bindEventListeners(),await Promise['all']([loadTeams(),loadUsers(),loadRoles(),loadEnvironments()]),renderTeams();});async function loadTeams(){const _0x3aab44=document['getElementById']('teams-loading'),_0x539ccd=document['getElementById']('teams-error'),_0x370914=document['getElementById']('teams-content');try{_0x3aab44['hidden']=![],_0x539ccd['hidden']=!![],_0x370914['hidden']=!![];const {teams:_0x5dbf98}=await API['getTeams']();state['teams']=_0x5dbf98||[],renderTeams(),_0x3aab44['hidden']=!![],_0x370914['hidden']=![];}catch(_0x210263){_0x3aab44['hidden']=!![],_0x539ccd['hidden']=![],document['getElementById']('teams-error-message')['textContent']=_0x210263['message'];}}async function loadUsers(){try{const {users:_0x2d9075}=await API['getUsers']();state['users']=_0x2d9075||[];}catch{state['users']=[];}}async function loadRoles(){try{const {roles:_0x34b253}=await API['getRoleDefinitions']();state['roles']=_0x34b253||[];}catch{state['roles']=[];}}async function loadEnvironments(){try{const _0x49805d=await API['getEnvironments']();state['environments']=_0x49805d['environments']||[];}catch{state['environments']=[];}}function renderTeams(){const _0x350417=document['getElementById']('teams-table-body'),_0x253840=document['getElementById']('team-count');if(!_0x350417)return;_0x253840&&(_0x253840['textContent']=state['teams']['length']+'\x20team'+(state['teams']['length']===0x1?'':'s'));if(state['teams']['length']===0x0){_0x350417['innerHTML']='\x0a\x20\x20\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<td\x20colspan=\x223\x22\x20style=\x22text-align:\x20center;\x20color:\x20var(--color-text-secondary);\x20padding:\x202rem;\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20No\x20teams\x20yet.\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(state['currentUser']?.['hasAllPermissions']||state['currentUser']?.['canManageTeams']?'Click\x20<strong>Create\x20Team</strong>\x20to\x20get\x20started.':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20</tr>';return;}const _0x51503e=state['currentUser']?.['hasAllPermissions'];_0x350417['innerHTML']=state['teams']['map'](_0x1e3878=>{const _0x562ce3=_0x51503e||_0x1e3878['canManageTeam'],_0x259d97=_0x51503e||_0x1e3878['canManageMembers'],_0x3635ea=_0x562ce3||_0x259d97,_0x534db3=(_0x1e3878['members']||[])['length'];return'\x0a\x20\x20\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20style=\x22font-weight:\x20600;\x22>'+Utils['escapeHtml'](_0x1e3878['name'])+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x1e3878['description']?'<div\x20style=\x22font-size:\x200.8em;\x20color:\x20var(--color-text-secondary);\x20margin-top:\x200.15rem;\x22>'+Utils['escapeHtml'](_0x1e3878['description'])+'</div>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<td>'+_0x534db3+'</td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22user-actions\x22\x20style=\x22display:\x20flex;\x20gap:\x200.25rem;\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x3635ea?'<button\x20class=\x22btn-icon\x22\x20title=\x22Edit\x20team\x22\x20onclick=\x22window.openTeamModal(\x27'+_0x1e3878['id']+'\x27)\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2216\x22\x20height=\x2216\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M11\x204H4a2\x202\x200\x200\x200-2\x202v14a2\x202\x200\x200\x200\x202\x202h14a2\x202\x200\x200\x200\x202-2v-7\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M18.5\x202.5a2.121\x202.121\x200\x200\x201\x203\x203L12\x2015l-4\x201\x201-4\x209.5-9.5z\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x562ce3?'<button\x20class=\x22btn-icon\x22\x20title=\x22Access\x20bindings\x22\x20onclick=\x22window.openBindingsModal(\x27'+_0x1e3878['id']+'\x27)\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2216\x22\x20height=\x2216\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M12\x2022s8-4\x208-10V5l-8-3-8\x203v7c0\x206\x208\x2010\x208\x2010z\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x562ce3?'<button\x20class=\x22btn-icon\x20btn-danger\x22\x20title=\x22Delete\x20team\x22\x20onclick=\x22window.deleteTeam(\x27'+_0x1e3878['id']+'\x27)\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2216\x22\x20height=\x2216\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<polyline\x20points=\x223\x206\x205\x206\x2021\x206\x22></polyline>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M19\x206v14a2\x202\x200\x200\x201-2\x202H7a2\x202\x200\x200\x201-2-2V6m3\x200V4a2\x202\x200\x200\x201\x202-2h4a2\x202\x200\x200\x201\x202\x202v2\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20</tr>';})['join']('');}function bindEventListeners(){document['getElementById']('create-team-btn')?.['addEventListener']('click',showCreateTeamModal),document['getElementById']('retry-teams-btn')?.['addEventListener']('click',loadTeams),document['getElementById']('close-team-modal')?.['addEventListener']('click',requestHideTeamModal),document['getElementById']('cancel-team')?.['addEventListener']('click',requestHideTeamModal),document['getElementById']('save-team')?.['addEventListener']('click',saveTeam),document['querySelector']('#team-modal\x20.modal-overlay')?.['addEventListener']('click',requestHideTeamModal),document['getElementById']('add-member-btn')?.['addEventListener']('click',addMember),document['getElementById']('close-bindings-modal')?.['addEventListener']('click',hideBindingsModal),document['querySelector']('#bindings-modal\x20.modal-overlay')?.['addEventListener']('click',hideBindingsModal),document['getElementById']('add-binding-btn')?.['addEventListener']('click',()=>{document['getElementById']('add-binding-form')['hidden']=![],document['getElementById']('add-binding-btn')['hidden']=!![],populateBindingForm();}),document['getElementById']('cancel-binding-btn')?.['addEventListener']('click',()=>{document['getElementById']('add-binding-form')['hidden']=!![],document['getElementById']('add-binding-btn')['hidden']=![];}),document['getElementById']('save-binding-btn')?.['addEventListener']('click',saveBinding),document['getElementById']('binding-resource-type')?.['addEventListener']('change',onBindingResourceTypeChange),document['getElementById']('close-delete-team-modal')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('cancel-delete-team')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('confirm-delete-team')?.['addEventListener']('click',confirmDeleteTeam),document['querySelector']('#delete-team-modal\x20.modal-overlay')?.['addEventListener']('click',hideDeleteModal);}function showCreateTeamModal(){state['editingTeamId']=null,state['modalSnapshot']={'name':'','description':''},document['getElementById']('team-modal-title')['textContent']='Create\x20Team',document['getElementById']('team-name')['value']='',document['getElementById']('team-name')['disabled']=![],document['getElementById']('team-description')['value']='',document['getElementById']('team-description')['disabled']=![],document['getElementById']('team-form-error')['hidden']=!![],document['getElementById']('members-section')['hidden']=!![],document['getElementById']('save-team')['hidden']=![],document['getElementById']('team-modal-footer')['hidden']=![],document['getElementById']('team-modal')['hidden']=![],setTimeout(()=>document['getElementById']('team-name')['focus'](),0x64);}window['openTeamModal']=function(_0xf71de1){const _0x1d8e78=state['teams']['find'](_0x1056e8=>_0x1056e8['id']===_0xf71de1);if(!_0x1d8e78)return;state['editingTeamId']=_0xf71de1;const _0x4f1769=state['currentUser']?.['hasAllPermissions'],_0x448718=_0x4f1769||_0x1d8e78['canManageTeam'],_0x55434d=_0x4f1769||_0x1d8e78['canManageMembers'],_0x133150=_0x448718,_0x389f88=_0x448718||_0x55434d,_0x344471=_0x1d8e78['name'],_0x25e19f=_0x1d8e78['description']||'';state['modalSnapshot']={'name':_0x344471,'description':_0x25e19f},document['getElementById']('team-modal-title')['textContent']=_0x344471,document['getElementById']('team-name')['value']=_0x344471,document['getElementById']('team-name')['disabled']=!_0x133150,document['getElementById']('team-description')['value']=_0x25e19f,document['getElementById']('team-description')['disabled']=!_0x133150,document['getElementById']('team-form-error')['hidden']=!![],document['getElementById']('members-section')['hidden']=![],document['getElementById']('save-team')['hidden']=!_0x133150,renderMembersList(_0x1d8e78),updateMemberSelect(_0x1d8e78),document['getElementById']('team-modal')['hidden']=![];};function isTeamModalDirty(){if(!state['modalSnapshot'])return![];const _0x38f46b=document['getElementById']('team-name')['value']['trim'](),_0x8ac4c6=document['getElementById']('team-description')['value']['trim']();return _0x38f46b!==state['modalSnapshot']['name']||_0x8ac4c6!==state['modalSnapshot']['description'];}function requestHideTeamModal(){if(isTeamModalDirty()){if(!confirm('You\x20have\x20unsaved\x20changes.\x20Discard\x20them?'))return;}hideTeamModal();}function hideTeamModal(){document['getElementById']('team-modal')['hidden']=!![],document['getElementById']('team-name')['disabled']=![],document['getElementById']('team-description')['disabled']=![],state['editingTeamId']=null,state['modalSnapshot']=null;}window['openBindingsModal']=function(_0xfa6d2){const _0x2a46e5=state['teams']['find'](_0x256aac=>_0x256aac['id']===_0xfa6d2);if(!_0x2a46e5)return;state['bindingsTeamId']=_0xfa6d2,document['getElementById']('bindings-modal-title')['textContent']='Access\x20Bindings\x20—\x20'+_0x2a46e5['name'],document['getElementById']('add-binding-form')['hidden']=!![],document['getElementById']('bindings-modal')['hidden']=![],loadTeamBindings(_0xfa6d2);};function hideBindingsModal(){document['getElementById']('bindings-modal')['hidden']=!![],state['bindingsTeamId']=null;}function renderMembersList(_0x54b930){const _0x5d37ce=state['currentUser']?.['hasAllPermissions'],_0x3e03b6=_0x5d37ce||_0x54b930['canManageMembers']||_0x54b930['canManageTeam'],_0x384f43=_0x54b930['members']||[],_0x4cc701=document['getElementById']('members-list');if(_0x384f43['length']===0x0){_0x4cc701['innerHTML']='<p\x20style=\x22font-size:\x200.875rem;\x20color:\x20var(--color-text-secondary);\x20margin:\x200.25rem\x200;\x22>No\x20members\x20yet</p>';return;}_0x4cc701['innerHTML']=_0x384f43['map'](_0x568fb7=>{const _0x4991a6=state['users']['find'](_0x58ecad=>_0x58ecad['id']===_0x568fb7),_0x1e503d=_0x4991a6?''+Utils['escapeHtml'](_0x4991a6['username'])+(_0x4991a6['name']?'\x20<span\x20style=\x22color:\x20var(--color-text-secondary);\x20font-size:\x200.85em;\x22>('+Utils['escapeHtml'](_0x4991a6['name'])+')</span>':''):Utils['escapeHtml'](_0x568fb7);return'\x0a\x20\x20\x20\x20\x20\x20<div\x20style=\x22display:\x20flex;\x20align-items:\x20center;\x20justify-content:\x20space-between;\x20padding:\x200.35rem\x200;\x20border-bottom:\x201px\x20solid\x20var(--color-border);\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<span\x20style=\x22font-size:\x200.875rem;\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x1e503d+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x3e03b6?'<button\x20type=\x22button\x22\x20class=\x22btn-ghost\x20btn-small\x22\x20onclick=\x22window.removeMemberClick(\x27'+_0x568fb7+'\x27)\x22\x20style=\x22color:\x20var(--color-danger);\x20padding:\x200.15rem\x200.4rem;\x20font-size:\x200.8rem;\x22>Remove</button>':'')+'\x0a\x20\x20\x20\x20\x20\x20</div>';})['join']('');const _0x409b69=document['getElementById']('add-member-row');if(_0x409b69)_0x409b69['style']['display']=_0x3e03b6?'flex':'none';}function updateMemberSelect(_0x343072){const _0x4948fe=document['getElementById']('add-member-select');if(!_0x4948fe)return;const _0x2fd6ed=new Set(_0x343072['members']||[]),_0x512886=state['users']['filter'](_0x29db0b=>!_0x2fd6ed['has'](_0x29db0b['id']));_0x4948fe['innerHTML']='<option\x20value=\x22\x22>Add\x20a\x20member…</option>'+_0x512886['map'](_0x20e0bf=>'<option\x20value=\x22'+_0x20e0bf['id']+'\x22>'+Utils['escapeHtml'](_0x20e0bf['username'])+(_0x20e0bf['name']?'\x20('+Utils['escapeHtml'](_0x20e0bf['name'])+')':'')+'</option>')['join']('');}async function addMember(){const _0xcf9b91=document['getElementById']('add-member-select'),_0x5a21f7=_0xcf9b91['value'];if(!_0x5a21f7||!state['editingTeamId'])return;try{const {team:_0x5193e6}=await API['addTeamMember'](state['editingTeamId'],_0x5a21f7);syncTeamInState(_0x5193e6),renderMembersList(_0x5193e6),updateMemberSelect(_0x5193e6),_0xcf9b91['value']='',Utils['showToast']('Member\x20added','success');}catch(_0x30ff69){Utils['showToast'](_0x30ff69['message'],'error');}}window['removeMemberClick']=async function(_0x18fed0){if(!state['editingTeamId'])return;try{const {team:_0x50ee68}=await API['removeTeamMember'](state['editingTeamId'],_0x18fed0);syncTeamInState(_0x50ee68),renderMembersList(_0x50ee68),updateMemberSelect(_0x50ee68),Utils['showToast']('Member\x20removed','success');}catch(_0x4d3dfe){Utils['showToast'](_0x4d3dfe['message'],'error');}};async function loadTeamBindings(_0x2119d9){const _0x37f430=document['getElementById']('bindings-loading'),_0x337ffd=document['getElementById']('bindings-empty'),_0x38a1d5=document['getElementById']('bindings-list-container'),_0x1ac011=document['getElementById']('add-binding-btn');_0x37f430['hidden']=![],_0x337ffd['hidden']=!![],_0x38a1d5['style']['display']='none',_0x1ac011['hidden']=!![],document['getElementById']('add-binding-form')['hidden']=!![];try{const {bindings:_0x28934f}=await API['getBindings']({'subject_type':'team','subject_id':_0x2119d9});state['teamBindings']=_0x28934f||[],_0x37f430['hidden']=!![],state['teamBindings']['length']===0x0?_0x337ffd['hidden']=![]:(_0x38a1d5['style']['display']='',renderBindingsTable()),(state['currentUser']?.['hasAllPermissions']||state['currentUser']?.['canManageTeams'])&&(_0x1ac011['hidden']=![]);}catch(_0x23ae5b){_0x37f430['hidden']=!![],Utils['showToast']('Failed\x20to\x20load\x20bindings:\x20'+_0x23ae5b['message'],'error');}}function renderBindingsTable(){const _0x52db12=document['getElementById']('bindings-table-body');if(!_0x52db12)return;const _0x4fdf9c=state['currentUser']?.['hasAllPermissions']||state['currentUser']?.['canManageTeams'];_0x52db12['innerHTML']=state['teamBindings']['map'](_0x4118ec=>{const _0x3180c8=_0x4118ec['resource_type']==='server'?'Server':_0x4118ec['resource_id']==='*'?'All\x20'+_0x4118ec['resource_type']+'s':Utils['escapeHtml'](_0x4118ec['resource_name']||_0x4118ec['resource_id']||'');return'\x0a\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20<td>'+_0x3180c8+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['escapeHtml'](_0x4118ec['role_name']||_0x4118ec['role_id'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x4fdf9c?'<button\x20class=\x22btn-ghost\x20btn-small\x22\x20onclick=\x22window.deleteBindingClick(\x27'+_0x4118ec['id']+'\x27)\x22\x20style=\x22color:\x20var(--color-danger);\x22>Remove</button>':'—')+'\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20</tr>';})['join']('');}function populateBindingForm(){const _0x25d137=document['getElementById']('binding-resource-type');_0x25d137['value']='environment',updateBindingFormForResourceType('environment');}function onBindingResourceTypeChange(){const _0xede738=document['getElementById']('binding-resource-type')['value'];updateBindingFormForResourceType(_0xede738);}function updateBindingFormForResourceType(_0x135710){const _0x50f54a=document['getElementById']('binding-resource-container'),_0x4e4dea=document['getElementById']('binding-resource'),_0xbaffda=document['getElementById']('binding-role');if(_0x135710==='server')_0x50f54a['style']['display']='none';else{_0x50f54a['style']['display']='';if(_0x135710==='environment'){const _0x3a564a=new Set(state['teamBindings']['filter'](_0x146c60=>_0x146c60['resource_type']==='environment')['map'](_0x3ca168=>_0x3ca168['resource_id'])),_0x5300a7=state['environments']['filter'](_0x4e80be=>!_0x3a564a['has'](_0x4e80be['name']));_0x4e4dea['innerHTML']=_0x5300a7['length']?_0x5300a7['map'](_0x1108a4=>'<option\x20value=\x22'+Utils['escapeHtml'](_0x1108a4['name'])+'\x22>'+Utils['escapeHtml'](_0x1108a4['name'])+'</option>')['join'](''):'<option\x20value=\x22\x22>All\x20environments\x20already\x20bound</option>';}else{if(_0x135710==='team'){const _0x20ea1c=new Set(state['teamBindings']['filter'](_0x2a2472=>_0x2a2472['resource_type']==='team')['map'](_0x1d9dab=>_0x1d9dab['resource_id'])),_0x5cfbad=state['teams']['filter'](_0x24a996=>!_0x20ea1c['has'](_0x24a996['id']));_0x4e4dea['innerHTML']=_0x5cfbad['length']?_0x5cfbad['map'](_0x5789e6=>'<option\x20value=\x22'+_0x5789e6['id']+'\x22>'+Utils['escapeHtml'](_0x5789e6['name'])+'</option>')['join'](''):'<option\x20value=\x22\x22>All\x20teams\x20already\x20bound</option>';}}}const _0x554667=state['roles']['filter'](_0x292df9=>_0x292df9['resource_type']===_0x135710);_0xbaffda['innerHTML']=_0x554667['length']?_0x554667['filter'](_0x2f9a64=>{const _0x2e0667=_0x135710==='server'?undefined:_0x4e4dea['value'];return!state['teamBindings']['some'](_0x1fad82=>_0x1fad82['resource_type']===_0x135710&&_0x1fad82['role_id']===_0x2f9a64['id']&&(_0x135710==='server'||_0x1fad82['resource_id']===_0x2e0667));})['map'](_0x49a526=>'<option\x20value=\x22'+_0x49a526['id']+'\x22>'+Utils['escapeHtml'](_0x49a526['name'])+'</option>')['join']('')||'<option\x20value=\x22\x22>No\x20available\x20roles</option>':'<option\x20value=\x22\x22>No\x20roles\x20for\x20this\x20resource\x20type</option>';}async function saveBinding(){const _0xef5222=document['getElementById']('binding-resource-type')['value'],_0x55db2b=document['getElementById']('binding-role')['value'];if(!_0x55db2b||!state['bindingsTeamId'])return;const _0x31d2cc=_0xef5222==='server'?undefined:document['getElementById']('binding-resource')['value'];if(_0xef5222!=='server'&&!_0x31d2cc)return;const _0x20a818=document['getElementById']('save-binding-btn');_0x20a818['disabled']=!![],_0x20a818['textContent']='Adding…';try{const _0x4099ef={'subject_type':'team','subject_id':state['bindingsTeamId'],'role_id':_0x55db2b,'resource_type':_0xef5222};if(_0x31d2cc)_0x4099ef['resource_id']=_0x31d2cc;await API['createBinding'](_0x4099ef),Utils['showToast']('Binding\x20added','success'),document['getElementById']('add-binding-form')['hidden']=!![],document['getElementById']('add-binding-btn')['hidden']=![],await loadTeamBindings(state['bindingsTeamId']);}catch(_0x50cb33){Utils['showToast'](_0x50cb33['message'],'error');}finally{_0x20a818['disabled']=![],_0x20a818['textContent']='Add\x20Binding';}}window['deleteBindingClick']=async function(_0x191f0c){try{await API['deleteBinding'](_0x191f0c),Utils['showToast']('Binding\x20removed','success'),await loadTeamBindings(state['bindingsTeamId']);}catch(_0x2f8f8e){Utils['showToast'](_0x2f8f8e['message'],'error');}};async function saveTeam(){const _0x41fd01=document['getElementById']('team-name')['value']['trim'](),_0x17ae54=document['getElementById']('team-description')['value']['trim'](),_0x552909=document['getElementById']('team-form-error'),_0x339f31=document['getElementById']('save-team');if(!_0x41fd01){_0x552909['textContent']='Team\x20name\x20is\x20required',_0x552909['hidden']=![];return;}_0x552909['hidden']=!![];const _0x4fde8a=_0x339f31['textContent'];_0x339f31['disabled']=!![],_0x339f31['textContent']=state['editingTeamId']?'Saving…':'Creating…';try{if(state['editingTeamId']){const {team:_0x7faf72}=await API['updateTeam'](state['editingTeamId'],{'name':_0x41fd01,'description':_0x17ae54});syncTeamInState(_0x7faf72),document['getElementById']('team-modal-title')['textContent']=_0x7faf72['name'],Utils['showToast']('Team\x20updated','success');}else await API['createTeam']({'name':_0x41fd01,'description':_0x17ae54}),Utils['showToast']('Team\x20created','success');_0x339f31['disabled']=![],_0x339f31['textContent']=_0x4fde8a,hideTeamModal(),await loadTeams();}catch(_0x5e1ddc){_0x552909['textContent']=_0x5e1ddc['message'],_0x552909['hidden']=![],_0x339f31['disabled']=![],_0x339f31['textContent']=_0x4fde8a;}}window['deleteTeam']=function(_0xdba7e2){const _0x1de0d9=state['teams']['find'](_0x30639e=>_0x30639e['id']===_0xdba7e2);if(!_0x1de0d9)return;state['deletingTeamId']=_0xdba7e2,document['getElementById']('delete-team-name')['textContent']=_0x1de0d9['name'],document['getElementById']('delete-team-modal')['hidden']=![];};function hideDeleteModal(){document['getElementById']('delete-team-modal')['hidden']=!![],state['deletingTeamId']=null;}async function confirmDeleteTeam(){if(!state['deletingTeamId'])return;const _0x235e30=document['getElementById']('confirm-delete-team');_0x235e30['disabled']=!![],_0x235e30['textContent']='Deleting…';try{await API['deleteTeam'](state['deletingTeamId']),Utils['showToast']('Team\x20deleted','success'),_0x235e30['disabled']=![],_0x235e30['textContent']='Delete\x20Team',hideDeleteModal(),await loadTeams();}catch(_0x3b7abd){Utils['showToast']('Failed\x20to\x20delete\x20team:\x20'+_0x3b7abd['message'],'error'),_0x235e30['disabled']=![],_0x235e30['textContent']='Delete\x20Team';}}function syncTeamInState(_0x3c40a5){const _0x2b372f=state['teams']['findIndex'](_0x1eec4d=>_0x1eec4d['id']===_0x3c40a5['id']);if(_0x2b372f>=0x0)state['teams'][_0x2b372f]=_0x3c40a5;renderTeams();}
1
+ import{API,Utils,initializeAdminMenu}from'./app.js';import{initTheme,mountThemeToggle}from'./theme.js';const state={'teams':[],'users':[],'roles':[],'environments':[],'teamBindings':[],'currentUser':null,'editingTeamId':null,'bindingsTeamId':null,'deletingTeamId':null,'modalSnapshot':null};document['addEventListener']('DOMContentLoaded',async()=>{initTheme(),mountThemeToggle();try{state['currentUser']=await API['getCurrentUser']();}catch{window['location']['href']='login.html';return;}await initializeAdminMenu(state['currentUser']);const _0x308d92=state['currentUser'];if(!_0x308d92['hasAllPermissions']&&!_0x308d92['canManageTeams']&&!_0x308d92['hasEnvironmentBindings']&&!_0x308d92['hasEnvironmentAdmin']){Utils['showToast']('Access\x20denied:\x20You\x20do\x20not\x20have\x20permission\x20to\x20view\x20teams','error'),setTimeout(()=>{window['location']['href']='index.html';},0x7d0);return;}(_0x308d92['hasAllPermissions']||_0x308d92['canManageTeams'])&&(document['getElementById']('create-team-btn')['hidden']=![]),bindEventListeners(),await Promise['all']([loadTeams(),loadUsers(),loadRoles(),loadEnvironments()]),renderTeams();});async function loadTeams(){const _0x5ba163=document['getElementById']('teams-loading'),_0x4e130f=document['getElementById']('teams-error'),_0x4ee585=document['getElementById']('teams-content');try{_0x5ba163['hidden']=![],_0x4e130f['hidden']=!![],_0x4ee585['hidden']=!![];const {teams:_0x288c05}=await API['getTeams']();state['teams']=_0x288c05||[],renderTeams(),_0x5ba163['hidden']=!![],_0x4ee585['hidden']=![];}catch(_0x52f8e6){_0x5ba163['hidden']=!![],_0x4e130f['hidden']=![],document['getElementById']('teams-error-message')['textContent']=_0x52f8e6['message'];}}async function loadUsers(){try{const {users:_0xa4b5db}=await API['getUsers']();state['users']=_0xa4b5db||[];}catch{state['users']=[];}}async function loadRoles(){try{const {roles:_0x1c6e06}=await API['getRoleDefinitions']();state['roles']=_0x1c6e06||[];}catch{state['roles']=[];}}async function loadEnvironments(){try{const _0x30c0cc=await API['getEnvironments']();state['environments']=_0x30c0cc['environments']||[];}catch{state['environments']=[];}}function renderTeams(){const _0x147633=document['getElementById']('teams-table-body'),_0x143b0f=document['getElementById']('team-count');if(!_0x147633)return;_0x143b0f&&(_0x143b0f['textContent']=state['teams']['length']+'\x20team'+(state['teams']['length']===0x1?'':'s'));if(state['teams']['length']===0x0){_0x147633['innerHTML']='\x0a\x20\x20\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<td\x20colspan=\x223\x22\x20style=\x22text-align:\x20center;\x20color:\x20var(--color-text-secondary);\x20padding:\x202rem;\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20No\x20teams\x20yet.\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(state['currentUser']?.['hasAllPermissions']||state['currentUser']?.['canManageTeams']?'Click\x20<strong>Create\x20Team</strong>\x20to\x20get\x20started.':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20</tr>';return;}const _0x4c5fed=state['currentUser']?.['hasAllPermissions'];_0x147633['innerHTML']=state['teams']['map'](_0x198a9f=>{const _0x20d8e2=_0x4c5fed||_0x198a9f['canManageTeam'],_0x3d240c=_0x4c5fed||_0x198a9f['canManageMembers'],_0x138dbf=_0x20d8e2||_0x3d240c,_0x3f97ec=(_0x198a9f['members']||[])['length'];return'\x0a\x20\x20\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20style=\x22font-weight:\x20600;\x22>'+Utils['escapeHtml'](_0x198a9f['name'])+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x198a9f['description']?'<div\x20style=\x22font-size:\x200.8em;\x20color:\x20var(--color-text-secondary);\x20margin-top:\x200.15rem;\x22>'+Utils['escapeHtml'](_0x198a9f['description'])+'</div>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<td>'+_0x3f97ec+'</td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22user-actions\x22\x20style=\x22display:\x20flex;\x20gap:\x200.25rem;\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x138dbf?'<button\x20class=\x22btn-icon\x22\x20title=\x22Edit\x20team\x22\x20onclick=\x22window.openTeamModal(\x27'+_0x198a9f['id']+'\x27)\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2216\x22\x20height=\x2216\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M11\x204H4a2\x202\x200\x200\x200-2\x202v14a2\x202\x200\x200\x200\x202\x202h14a2\x202\x200\x200\x200\x202-2v-7\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M18.5\x202.5a2.121\x202.121\x200\x200\x201\x203\x203L12\x2015l-4\x201\x201-4\x209.5-9.5z\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x20d8e2?'<button\x20class=\x22btn-icon\x22\x20title=\x22Access\x20bindings\x22\x20onclick=\x22window.openBindingsModal(\x27'+_0x198a9f['id']+'\x27)\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2216\x22\x20height=\x2216\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M12\x2022s8-4\x208-10V5l-8-3-8\x203v7c0\x206\x208\x2010\x208\x2010z\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x20d8e2?'<button\x20class=\x22btn-icon\x20btn-danger\x22\x20title=\x22Delete\x20team\x22\x20onclick=\x22window.deleteTeam(\x27'+_0x198a9f['id']+'\x27)\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2216\x22\x20height=\x2216\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<polyline\x20points=\x223\x206\x205\x206\x2021\x206\x22></polyline>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M19\x206v14a2\x202\x200\x200\x201-2\x202H7a2\x202\x200\x200\x201-2-2V6m3\x200V4a2\x202\x200\x200\x201\x202-2h4a2\x202\x200\x200\x201\x202\x202v2\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20</tr>';})['join']('');}function bindEventListeners(){document['getElementById']('create-team-btn')?.['addEventListener']('click',showCreateTeamModal),document['getElementById']('retry-teams-btn')?.['addEventListener']('click',loadTeams),document['getElementById']('close-team-modal')?.['addEventListener']('click',requestHideTeamModal),document['getElementById']('cancel-team')?.['addEventListener']('click',requestHideTeamModal),document['getElementById']('save-team')?.['addEventListener']('click',saveTeam),document['querySelector']('#team-modal\x20.modal-overlay')?.['addEventListener']('click',requestHideTeamModal),document['getElementById']('add-member-btn')?.['addEventListener']('click',addMember),document['getElementById']('close-bindings-modal')?.['addEventListener']('click',hideBindingsModal),document['querySelector']('#bindings-modal\x20.modal-overlay')?.['addEventListener']('click',hideBindingsModal),document['getElementById']('add-binding-btn')?.['addEventListener']('click',()=>{document['getElementById']('add-binding-form')['hidden']=![],document['getElementById']('add-binding-btn')['hidden']=!![],populateBindingForm();}),document['getElementById']('cancel-binding-btn')?.['addEventListener']('click',()=>{document['getElementById']('add-binding-form')['hidden']=!![],document['getElementById']('add-binding-btn')['hidden']=![];}),document['getElementById']('save-binding-btn')?.['addEventListener']('click',saveBinding),document['getElementById']('binding-resource-type')?.['addEventListener']('change',onBindingResourceTypeChange),document['getElementById']('close-delete-team-modal')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('cancel-delete-team')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('confirm-delete-team')?.['addEventListener']('click',confirmDeleteTeam),document['querySelector']('#delete-team-modal\x20.modal-overlay')?.['addEventListener']('click',hideDeleteModal);}function showCreateTeamModal(){state['editingTeamId']=null,state['modalSnapshot']={'name':'','description':''},document['getElementById']('team-modal-title')['textContent']='Create\x20Team',document['getElementById']('team-name')['value']='',document['getElementById']('team-name')['disabled']=![],document['getElementById']('team-description')['value']='',document['getElementById']('team-description')['disabled']=![],document['getElementById']('team-form-error')['hidden']=!![],document['getElementById']('members-section')['hidden']=!![],document['getElementById']('save-team')['hidden']=![],document['getElementById']('team-modal-footer')['hidden']=![],document['getElementById']('team-modal')['hidden']=![],setTimeout(()=>document['getElementById']('team-name')['focus'](),0x64);}window['openTeamModal']=function(_0x5f1a08){const _0x22923f=state['teams']['find'](_0x2964c2=>_0x2964c2['id']===_0x5f1a08);if(!_0x22923f)return;state['editingTeamId']=_0x5f1a08;const _0x4e5daa=state['currentUser']?.['hasAllPermissions'],_0x3e2b38=_0x4e5daa||_0x22923f['canManageTeam'],_0x18177c=_0x4e5daa||_0x22923f['canManageMembers'],_0x2c109f=_0x3e2b38,_0x253899=_0x3e2b38||_0x18177c,_0x514780=_0x22923f['name'],_0x308bd5=_0x22923f['description']||'';state['modalSnapshot']={'name':_0x514780,'description':_0x308bd5},document['getElementById']('team-modal-title')['textContent']=_0x514780,document['getElementById']('team-name')['value']=_0x514780,document['getElementById']('team-name')['disabled']=!_0x2c109f,document['getElementById']('team-description')['value']=_0x308bd5,document['getElementById']('team-description')['disabled']=!_0x2c109f,document['getElementById']('team-form-error')['hidden']=!![],document['getElementById']('members-section')['hidden']=![],document['getElementById']('save-team')['hidden']=!_0x2c109f,renderMembersList(_0x22923f),updateMemberSelect(_0x22923f),document['getElementById']('team-modal')['hidden']=![];};function isTeamModalDirty(){if(!state['modalSnapshot'])return![];const _0x8caf08=document['getElementById']('team-name')['value']['trim'](),_0x466504=document['getElementById']('team-description')['value']['trim']();return _0x8caf08!==state['modalSnapshot']['name']||_0x466504!==state['modalSnapshot']['description'];}function requestHideTeamModal(){if(isTeamModalDirty()){if(!confirm('You\x20have\x20unsaved\x20changes.\x20Discard\x20them?'))return;}hideTeamModal();}function hideTeamModal(){document['getElementById']('team-modal')['hidden']=!![],document['getElementById']('team-name')['disabled']=![],document['getElementById']('team-description')['disabled']=![],state['editingTeamId']=null,state['modalSnapshot']=null;}window['openBindingsModal']=function(_0x471621){const _0xcf8588=state['teams']['find'](_0xa8c8a8=>_0xa8c8a8['id']===_0x471621);if(!_0xcf8588)return;state['bindingsTeamId']=_0x471621,document['getElementById']('bindings-modal-title')['textContent']='Access\x20Bindings\x20—\x20'+_0xcf8588['name'],document['getElementById']('add-binding-form')['hidden']=!![],document['getElementById']('bindings-modal')['hidden']=![],loadTeamBindings(_0x471621);};function hideBindingsModal(){document['getElementById']('bindings-modal')['hidden']=!![],state['bindingsTeamId']=null;}function renderMembersList(_0x4925c2){const _0xb6d4ae=state['currentUser']?.['hasAllPermissions'],_0x3e460e=_0xb6d4ae||_0x4925c2['canManageMembers']||_0x4925c2['canManageTeam'],_0x1c4680=_0x4925c2['members']||[],_0x9181fd=document['getElementById']('members-list');if(_0x1c4680['length']===0x0){_0x9181fd['innerHTML']='<p\x20style=\x22font-size:\x200.875rem;\x20color:\x20var(--color-text-secondary);\x20margin:\x200.25rem\x200;\x22>No\x20members\x20yet</p>';return;}_0x9181fd['innerHTML']=_0x1c4680['map'](_0xecd7f3=>{const _0x3621e8=state['users']['find'](_0x307d4c=>_0x307d4c['id']===_0xecd7f3),_0x5c362d=_0x3621e8?''+Utils['escapeHtml'](_0x3621e8['username'])+(_0x3621e8['name']?'\x20<span\x20style=\x22color:\x20var(--color-text-secondary);\x20font-size:\x200.85em;\x22>('+Utils['escapeHtml'](_0x3621e8['name'])+')</span>':''):Utils['escapeHtml'](_0xecd7f3);return'\x0a\x20\x20\x20\x20\x20\x20<div\x20style=\x22display:\x20flex;\x20align-items:\x20center;\x20justify-content:\x20space-between;\x20padding:\x200.35rem\x200;\x20border-bottom:\x201px\x20solid\x20var(--color-border);\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<span\x20style=\x22font-size:\x200.875rem;\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x5c362d+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x3e460e?'<button\x20type=\x22button\x22\x20class=\x22btn-ghost\x20btn-small\x22\x20onclick=\x22window.removeMemberClick(\x27'+_0xecd7f3+'\x27)\x22\x20style=\x22color:\x20var(--color-danger);\x20padding:\x200.15rem\x200.4rem;\x20font-size:\x200.8rem;\x22>Remove</button>':'')+'\x0a\x20\x20\x20\x20\x20\x20</div>';})['join']('');const _0x7d27c=document['getElementById']('add-member-row');if(_0x7d27c)_0x7d27c['style']['display']=_0x3e460e?'flex':'none';}function updateMemberSelect(_0x41e89e){const _0x467ded=document['getElementById']('add-member-select');if(!_0x467ded)return;const _0x53d7cb=new Set(_0x41e89e['members']||[]),_0x1d4792=state['users']['filter'](_0x456f83=>!_0x53d7cb['has'](_0x456f83['id']));_0x467ded['innerHTML']='<option\x20value=\x22\x22>Add\x20a\x20member…</option>'+_0x1d4792['map'](_0x30aca9=>'<option\x20value=\x22'+_0x30aca9['id']+'\x22>'+Utils['escapeHtml'](_0x30aca9['username'])+(_0x30aca9['name']?'\x20('+Utils['escapeHtml'](_0x30aca9['name'])+')':'')+'</option>')['join']('');}async function addMember(){const _0x3283ee=document['getElementById']('add-member-select'),_0x510750=_0x3283ee['value'];if(!_0x510750||!state['editingTeamId'])return;try{const {team:_0x316c00}=await API['addTeamMember'](state['editingTeamId'],_0x510750);syncTeamInState(_0x316c00),renderMembersList(_0x316c00),updateMemberSelect(_0x316c00),_0x3283ee['value']='',Utils['showToast']('Member\x20added','success');}catch(_0x4ba25a){Utils['showToast'](_0x4ba25a['message'],'error');}}window['removeMemberClick']=async function(_0x28167b){if(!state['editingTeamId'])return;try{const {team:_0x280513}=await API['removeTeamMember'](state['editingTeamId'],_0x28167b);syncTeamInState(_0x280513),renderMembersList(_0x280513),updateMemberSelect(_0x280513),Utils['showToast']('Member\x20removed','success');}catch(_0x490a4b){Utils['showToast'](_0x490a4b['message'],'error');}};async function loadTeamBindings(_0xcb1b3b){const _0x577d13=document['getElementById']('bindings-loading'),_0x4de05f=document['getElementById']('bindings-empty'),_0x17ae6a=document['getElementById']('bindings-list-container'),_0xd29a9c=document['getElementById']('add-binding-btn');_0x577d13['hidden']=![],_0x4de05f['hidden']=!![],_0x17ae6a['style']['display']='none',_0xd29a9c['hidden']=!![],document['getElementById']('add-binding-form')['hidden']=!![];try{const {bindings:_0x1b7694}=await API['getBindings']({'subject_type':'team','subject_id':_0xcb1b3b});state['teamBindings']=_0x1b7694||[],_0x577d13['hidden']=!![],state['teamBindings']['length']===0x0?_0x4de05f['hidden']=![]:(_0x17ae6a['style']['display']='',renderBindingsTable()),(state['currentUser']?.['hasAllPermissions']||state['currentUser']?.['canManageTeams'])&&(_0xd29a9c['hidden']=![]);}catch(_0xe746a1){_0x577d13['hidden']=!![],Utils['showToast']('Failed\x20to\x20load\x20bindings:\x20'+_0xe746a1['message'],'error');}}function renderBindingsTable(){const _0x3a28ba=document['getElementById']('bindings-table-body');if(!_0x3a28ba)return;const _0x47d333=state['currentUser']?.['hasAllPermissions']||state['currentUser']?.['canManageTeams'];_0x3a28ba['innerHTML']=state['teamBindings']['map'](_0x368ca0=>{const _0x1a8e58=_0x368ca0['resource_type']==='server'?'Server':_0x368ca0['resource_id']==='*'?'All\x20'+_0x368ca0['resource_type']+'s':Utils['escapeHtml'](_0x368ca0['resource_name']||_0x368ca0['resource_id']||'');return'\x0a\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20<td>'+_0x1a8e58+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['escapeHtml'](_0x368ca0['role_name']||_0x368ca0['role_id'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x47d333?'<button\x20class=\x22btn-ghost\x20btn-small\x22\x20onclick=\x22window.deleteBindingClick(\x27'+_0x368ca0['id']+'\x27)\x22\x20style=\x22color:\x20var(--color-danger);\x22>Remove</button>':'—')+'\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20</tr>';})['join']('');}function populateBindingForm(){const _0xe31a4d=document['getElementById']('binding-resource-type');_0xe31a4d['value']='environment',updateBindingFormForResourceType('environment');}function onBindingResourceTypeChange(){const _0x5d3630=document['getElementById']('binding-resource-type')['value'];updateBindingFormForResourceType(_0x5d3630);}function updateBindingFormForResourceType(_0x19d091){const _0xe8c41=document['getElementById']('binding-resource-container'),_0xe052bc=document['getElementById']('binding-resource'),_0x1a157e=document['getElementById']('binding-role');if(_0x19d091==='server')_0xe8c41['style']['display']='none';else{_0xe8c41['style']['display']='';if(_0x19d091==='environment'){const _0x211cb7=new Set(state['teamBindings']['filter'](_0x4c2115=>_0x4c2115['resource_type']==='environment')['map'](_0x22dc4f=>_0x22dc4f['resource_id'])),_0x288359=state['environments']['filter'](_0x2d2ce7=>!_0x211cb7['has'](_0x2d2ce7['name']));_0xe052bc['innerHTML']=_0x288359['length']?_0x288359['map'](_0x446db6=>'<option\x20value=\x22'+Utils['escapeHtml'](_0x446db6['name'])+'\x22>'+Utils['escapeHtml'](_0x446db6['name'])+'</option>')['join'](''):'<option\x20value=\x22\x22>All\x20environments\x20already\x20bound</option>';}else{if(_0x19d091==='team'){const _0x50a555=new Set(state['teamBindings']['filter'](_0x1939e1=>_0x1939e1['resource_type']==='team')['map'](_0x2c0f15=>_0x2c0f15['resource_id'])),_0x2bb1e9=state['teams']['filter'](_0x19f557=>!_0x50a555['has'](_0x19f557['id']));_0xe052bc['innerHTML']=_0x2bb1e9['length']?_0x2bb1e9['map'](_0x3aa52b=>'<option\x20value=\x22'+_0x3aa52b['id']+'\x22>'+Utils['escapeHtml'](_0x3aa52b['name'])+'</option>')['join'](''):'<option\x20value=\x22\x22>All\x20teams\x20already\x20bound</option>';}}}const _0x396ce4=state['roles']['filter'](_0xef63a7=>_0xef63a7['resource_type']===_0x19d091);_0x1a157e['innerHTML']=_0x396ce4['length']?_0x396ce4['filter'](_0x4ebd48=>{const _0x3e88b1=_0x19d091==='server'?undefined:_0xe052bc['value'];return!state['teamBindings']['some'](_0x32d089=>_0x32d089['resource_type']===_0x19d091&&_0x32d089['role_id']===_0x4ebd48['id']&&(_0x19d091==='server'||_0x32d089['resource_id']===_0x3e88b1));})['map'](_0x6c01ae=>'<option\x20value=\x22'+_0x6c01ae['id']+'\x22>'+Utils['escapeHtml'](_0x6c01ae['name'])+'</option>')['join']('')||'<option\x20value=\x22\x22>No\x20available\x20roles</option>':'<option\x20value=\x22\x22>No\x20roles\x20for\x20this\x20resource\x20type</option>';}async function saveBinding(){const _0x2265ae=document['getElementById']('binding-resource-type')['value'],_0x32d705=document['getElementById']('binding-role')['value'];if(!_0x32d705||!state['bindingsTeamId'])return;const _0x3e5484=_0x2265ae==='server'?undefined:document['getElementById']('binding-resource')['value'];if(_0x2265ae!=='server'&&!_0x3e5484)return;const _0x3e19f6=document['getElementById']('save-binding-btn');_0x3e19f6['disabled']=!![],_0x3e19f6['textContent']='Adding…';try{const _0x2d70cd={'subject_type':'team','subject_id':state['bindingsTeamId'],'role_id':_0x32d705,'resource_type':_0x2265ae};if(_0x3e5484)_0x2d70cd['resource_id']=_0x3e5484;await API['createBinding'](_0x2d70cd),Utils['showToast']('Binding\x20added','success'),document['getElementById']('add-binding-form')['hidden']=!![],document['getElementById']('add-binding-btn')['hidden']=![],await loadTeamBindings(state['bindingsTeamId']);}catch(_0x1f40f){Utils['showToast'](_0x1f40f['message'],'error');}finally{_0x3e19f6['disabled']=![],_0x3e19f6['textContent']='Add\x20Binding';}}window['deleteBindingClick']=async function(_0x4c8141){try{await API['deleteBinding'](_0x4c8141),Utils['showToast']('Binding\x20removed','success'),await loadTeamBindings(state['bindingsTeamId']);}catch(_0x11ba63){Utils['showToast'](_0x11ba63['message'],'error');}};async function saveTeam(){const _0x437a79=document['getElementById']('team-name')['value']['trim'](),_0x5b9b2c=document['getElementById']('team-description')['value']['trim'](),_0x2283c1=document['getElementById']('team-form-error'),_0x461103=document['getElementById']('save-team');if(!_0x437a79){_0x2283c1['textContent']='Team\x20name\x20is\x20required',_0x2283c1['hidden']=![];return;}_0x2283c1['hidden']=!![];const _0x236a2e=_0x461103['textContent'];_0x461103['disabled']=!![],_0x461103['textContent']=state['editingTeamId']?'Saving…':'Creating…';try{if(state['editingTeamId']){const {team:_0x337338}=await API['updateTeam'](state['editingTeamId'],{'name':_0x437a79,'description':_0x5b9b2c});syncTeamInState(_0x337338),document['getElementById']('team-modal-title')['textContent']=_0x337338['name'],Utils['showToast']('Team\x20updated','success');}else await API['createTeam']({'name':_0x437a79,'description':_0x5b9b2c}),Utils['showToast']('Team\x20created','success');_0x461103['disabled']=![],_0x461103['textContent']=_0x236a2e,hideTeamModal(),await loadTeams();}catch(_0x31b7a7){_0x2283c1['textContent']=_0x31b7a7['message'],_0x2283c1['hidden']=![],_0x461103['disabled']=![],_0x461103['textContent']=_0x236a2e;}}window['deleteTeam']=function(_0x42cdcc){const _0x5e2b2=state['teams']['find'](_0x28ec15=>_0x28ec15['id']===_0x42cdcc);if(!_0x5e2b2)return;state['deletingTeamId']=_0x42cdcc,document['getElementById']('delete-team-name')['textContent']=_0x5e2b2['name'],document['getElementById']('delete-team-modal')['hidden']=![];};function hideDeleteModal(){document['getElementById']('delete-team-modal')['hidden']=!![],state['deletingTeamId']=null;}async function confirmDeleteTeam(){if(!state['deletingTeamId'])return;const _0x43e7f7=document['getElementById']('confirm-delete-team');_0x43e7f7['disabled']=!![],_0x43e7f7['textContent']='Deleting…';try{await API['deleteTeam'](state['deletingTeamId']),Utils['showToast']('Team\x20deleted','success'),_0x43e7f7['disabled']=![],_0x43e7f7['textContent']='Delete\x20Team',hideDeleteModal(),await loadTeams();}catch(_0x42a60c){Utils['showToast']('Failed\x20to\x20delete\x20team:\x20'+_0x42a60c['message'],'error'),_0x43e7f7['disabled']=![],_0x43e7f7['textContent']='Delete\x20Team';}}function syncTeamInState(_0x323c27){const _0x3d9289=state['teams']['findIndex'](_0x3f36bf=>_0x3f36bf['id']===_0x323c27['id']);if(_0x3d9289>=0x0)state['teams'][_0x3d9289]=_0x323c27;renderTeams();}
@@ -22,9 +22,9 @@
22
22
  }
23
23
  })();
24
24
  </script>
25
- <link rel="stylesheet" href="styles.css?v=83">
25
+ <link rel="stylesheet" href="styles.css?v=84">
26
26
  <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/xterm@5.3.0/css/xterm.css">
27
- <script type="module" src="terminal.js?v=59"></script>
27
+ <script type="module" src="terminal.js?v=60"></script>
28
28
  <!-- Dev QA shortcut: Ctrl+Shift+Q to launch current page in QA mode -->
29
29
  </head>
30
30
  <body class="terminal-body">
@@ -1 +1 @@
1
- import{API,Utils}from'./app.js';import{Terminal}from'https://cdn.jsdelivr.net/npm/xterm@5.3.0/+esm';import{FitAddon}from'https://cdn.jsdelivr.net/npm/xterm-addon-fit@0.7.0/+esm';import{getCurrentTheme,initTheme,mountThemeToggle,onThemeChange}from'./theme.js';const params=Utils['getQueryParams'](),containerId=params['container'],command=(params['cmd']||'')['trim'](),useServerExec=params['server_exec']==='true',titleEl=document['getElementById']('terminal-title'),subtitleEl=document['getElementById']('terminal-subtitle'),statusEl=document['getElementById']('terminal-status'),terminalEl=document['getElementById']('terminal-container'),closeBtn=document['getElementById']('close-terminal');initTheme(),mountThemeToggle({'targetSelectors':['.terminal-actions']});const getXtermTheme=()=>{return{'background':'#0f172a','foreground':'#e2e8f0','cursor':'#e2e8f0','cursorAccent':'#0f172a'};};closeBtn&&closeBtn['addEventListener']('click',()=>{window['close']();});if(!containerId){updateStatus('Missing\x20container\x20identifier','error');throw new Error('container\x20parameter\x20is\x20required');}document['title']='Terminal\x20·\x20'+containerId['slice'](0x0,0xc),enrichContainerDetails(containerId)['catch'](()=>{subtitleEl['textContent']='Container\x20'+containerId['substring'](0x0,0xc);});const term=new Terminal({'convertEol':!![],'cursorBlink':!![],'fontFamily':'JetBrains\x20Mono,\x20SFMono-Regular,\x20Consolas,\x20monospace','fontSize':0x10,'theme':getXtermTheme()}),fitAddon=new FitAddon();term['loadAddon'](fitAddon),term['open'](terminalEl),term['focus']();let isFitting=![];const doFit=()=>{if(isFitting)return;isFitting=!![],fitAddon['fit'](),sendResize(),term['scrollToBottom'](),setTimeout(()=>{isFitting=![];},0x32);};window['addEventListener']('resize',()=>{requestAnimationFrame(()=>{!isFitting&&doFit();});});const resizeObserver=new ResizeObserver(()=>{requestAnimationFrame(()=>{!isFitting&&doFit();});});resizeObserver['observe'](terminalEl);const socketProtocol=window['location']['protocol']==='https:'?'wss':'ws',socketUrl=new URL('/ws/containers/'+encodeURIComponent(containerId),window['location']['href']);command&&useServerExec&&socketUrl['searchParams']['set']('cmd',command);socketUrl['protocol']=socketProtocol+':';const socket=new WebSocket(socketUrl['toString']());term['onData'](_0x1e6e45=>{socket['readyState']===WebSocket['OPEN']&&socket['send'](JSON['stringify']({'type':'data','data':_0x1e6e45}));});let commandSent=![],hasReceivedOutput=![];socket['addEventListener']('message',_0xd8ce87=>{try{const _0x1422e0=JSON['parse'](_0xd8ce87['data']);if(_0x1422e0['type']==='data'&&typeof _0x1422e0['data']==='string')term['write'](_0x1422e0['data']),!hasReceivedOutput&&_0x1422e0['data']['length']>0x0&&(hasReceivedOutput=!![],command&&!useServerExec&&!commandSent&&(commandSent=!![],setTimeout(()=>{socket['readyState']===WebSocket['OPEN']&&socket['send'](JSON['stringify']({'type':'data','data':command+'\x0d'}));},0xc8)));else{if(_0x1422e0['type']==='status'&&_0x1422e0['status']){if(_0x1422e0['status']==='connected'){if(command&&useServerExec)updateStatus('Running:\x20'+command,'info');else command?updateStatus('Preparing:\x20'+command,'info'):updateStatus('Terminal\x20connected','success');doFit(),term['focus'](),command&&!useServerExec&&!commandSent&&setTimeout(()=>{!commandSent&&socket['readyState']===WebSocket['OPEN']&&(commandSent=!![],socket['send'](JSON['stringify']({'type':'data','data':command+'\x0d'})),updateStatus('Running:\x20'+command,'info'));},0x7d0);}else updateStatus('Session\x20'+_0x1422e0['status'],_0x1422e0['status']==='connected'?'success':'info');}else _0x1422e0['type']==='error'&&_0x1422e0['message']&&updateStatus(_0x1422e0['message'],'error');}}catch(_0x1ecf50){console['error']('Failed\x20to\x20parse\x20terminal\x20message',_0x1ecf50);}}),socket['addEventListener']('close',_0x461368=>{const _0x4c84eb=_0x461368['reason']||'Connection\x20closed';updateStatus(_0x4c84eb,_0x461368['wasClean']?'info':'error');}),socket['addEventListener']('error',_0x36a408=>{console['error']('Terminal\x20websocket\x20error',_0x36a408),updateStatus('Connection\x20error','error');});function sendResize(){if(socket['readyState']!==WebSocket['OPEN'])return;const _0x24c922=term['cols'],_0x319ff5=term['rows'];socket['send'](JSON['stringify']({'type':'resize','cols':_0x24c922,'rows':_0x319ff5}));}function updateStatus(_0x5bad41,_0x15ce96='info'){if(!statusEl)return;statusEl['textContent']=_0x5bad41,statusEl['dataset']['variant']=_0x15ce96;}async function enrichContainerDetails(_0x47f75f){try{const _0x56b1ac=await API['getContainer'](_0x47f75f);updateTerminalDetails(_0x56b1ac);}catch{try{const _0x59c22a=await API['getContainers'](),_0x2a0502=(_0x59c22a['containers']||[])['find'](_0x10564c=>_0x10564c['containerId']===_0x47f75f||_0x10564c['name']===_0x47f75f||_0x10564c['fullContainerId']?.['startsWith'](_0x47f75f));if(_0x2a0502){updateTerminalDetails(_0x2a0502);return;}subtitleEl['textContent']='Container\x20'+_0x47f75f['substring'](0x0,0xc);}catch{subtitleEl['textContent']='Container\x20'+_0x47f75f['substring'](0x0,0xc);}}}function updateTerminalDetails(_0x40942f){const _0xf03212=_0x40942f['containerId']||_0x40942f['name']||_0x40942f['fullContainerId']||containerId,_0x56fda6=_0x40942f['environment']?'Environment\x20'+_0x40942f['environment']:'Environment\x20unknown',_0x5dc795=_0x40942f['defaultAgent']?'\x20·\x20Agent\x20'+_0x40942f['defaultAgent']:'',_0x31fddb=_0x40942f['createdAt']?'\x20·\x20Started\x20'+Utils['formatRelativeTime'](_0x40942f['createdAt']):'';titleEl['textContent']='Terminal\x20·\x20'+_0xf03212['substring'](0x0,0xc);const _0x171723=command?'\x20·\x20Command\x20'+command:'';subtitleEl['textContent']=''+_0x56fda6+_0x5dc795+_0x31fddb+_0x171723;}
1
+ import{API,Utils}from'./app.js';import{Terminal}from'https://cdn.jsdelivr.net/npm/xterm@5.3.0/+esm';import{FitAddon}from'https://cdn.jsdelivr.net/npm/xterm-addon-fit@0.7.0/+esm';import{getCurrentTheme,initTheme,mountThemeToggle,onThemeChange}from'./theme.js';const params=Utils['getQueryParams'](),containerId=params['container'],command=(params['cmd']||'')['trim'](),useServerExec=params['server_exec']==='true',titleEl=document['getElementById']('terminal-title'),subtitleEl=document['getElementById']('terminal-subtitle'),statusEl=document['getElementById']('terminal-status'),terminalEl=document['getElementById']('terminal-container'),closeBtn=document['getElementById']('close-terminal');initTheme(),mountThemeToggle({'targetSelectors':['.terminal-actions']});const getXtermTheme=()=>{return{'background':'#0f172a','foreground':'#e2e8f0','cursor':'#e2e8f0','cursorAccent':'#0f172a'};};closeBtn&&closeBtn['addEventListener']('click',()=>{window['close']();});if(!containerId){updateStatus('Missing\x20container\x20identifier','error');throw new Error('container\x20parameter\x20is\x20required');}document['title']='Terminal\x20·\x20'+containerId['slice'](0x0,0xc),enrichContainerDetails(containerId)['catch'](()=>{subtitleEl['textContent']='Container\x20'+containerId['substring'](0x0,0xc);});const term=new Terminal({'convertEol':!![],'cursorBlink':!![],'fontFamily':'JetBrains\x20Mono,\x20SFMono-Regular,\x20Consolas,\x20monospace','fontSize':0x10,'theme':getXtermTheme()}),fitAddon=new FitAddon();term['loadAddon'](fitAddon),term['open'](terminalEl),term['focus']();let isFitting=![];const doFit=()=>{if(isFitting)return;isFitting=!![],fitAddon['fit'](),sendResize(),term['scrollToBottom'](),setTimeout(()=>{isFitting=![];},0x32);};window['addEventListener']('resize',()=>{requestAnimationFrame(()=>{!isFitting&&doFit();});});const resizeObserver=new ResizeObserver(()=>{requestAnimationFrame(()=>{!isFitting&&doFit();});});resizeObserver['observe'](terminalEl);const socketProtocol=window['location']['protocol']==='https:'?'wss':'ws',socketUrl=new URL('/ws/containers/'+encodeURIComponent(containerId),window['location']['href']);command&&useServerExec&&socketUrl['searchParams']['set']('cmd',command);socketUrl['protocol']=socketProtocol+':';const socket=new WebSocket(socketUrl['toString']());term['onData'](_0x5dcba9=>{socket['readyState']===WebSocket['OPEN']&&socket['send'](JSON['stringify']({'type':'data','data':_0x5dcba9}));});let commandSent=![],hasReceivedOutput=![];socket['addEventListener']('message',_0x1d45a2=>{try{const _0x15e0c6=JSON['parse'](_0x1d45a2['data']);if(_0x15e0c6['type']==='data'&&typeof _0x15e0c6['data']==='string')term['write'](_0x15e0c6['data']),!hasReceivedOutput&&_0x15e0c6['data']['length']>0x0&&(hasReceivedOutput=!![],command&&!useServerExec&&!commandSent&&(commandSent=!![],setTimeout(()=>{socket['readyState']===WebSocket['OPEN']&&socket['send'](JSON['stringify']({'type':'data','data':command+'\x0d'}));},0xc8)));else{if(_0x15e0c6['type']==='status'&&_0x15e0c6['status']){if(_0x15e0c6['status']==='connected'){if(command&&useServerExec)updateStatus('Running:\x20'+command,'info');else command?updateStatus('Preparing:\x20'+command,'info'):updateStatus('Terminal\x20connected','success');doFit(),term['focus'](),command&&!useServerExec&&!commandSent&&setTimeout(()=>{!commandSent&&socket['readyState']===WebSocket['OPEN']&&(commandSent=!![],socket['send'](JSON['stringify']({'type':'data','data':command+'\x0d'})),updateStatus('Running:\x20'+command,'info'));},0x7d0);}else updateStatus('Session\x20'+_0x15e0c6['status'],_0x15e0c6['status']==='connected'?'success':'info');}else _0x15e0c6['type']==='error'&&_0x15e0c6['message']&&updateStatus(_0x15e0c6['message'],'error');}}catch(_0x5c173e){console['error']('Failed\x20to\x20parse\x20terminal\x20message',_0x5c173e);}}),socket['addEventListener']('close',_0x2ae9c3=>{const _0x444823=_0x2ae9c3['reason']||'Connection\x20closed';updateStatus(_0x444823,_0x2ae9c3['wasClean']?'info':'error');}),socket['addEventListener']('error',_0x3d90a0=>{console['error']('Terminal\x20websocket\x20error',_0x3d90a0),updateStatus('Connection\x20error','error');});function sendResize(){if(socket['readyState']!==WebSocket['OPEN'])return;const _0x12b8e9=term['cols'],_0x390c1c=term['rows'];socket['send'](JSON['stringify']({'type':'resize','cols':_0x12b8e9,'rows':_0x390c1c}));}function updateStatus(_0x2d327d,_0xc4bdf6='info'){if(!statusEl)return;statusEl['textContent']=_0x2d327d,statusEl['dataset']['variant']=_0xc4bdf6;}async function enrichContainerDetails(_0x397ed2){try{const _0x1255fd=await API['getContainer'](_0x397ed2);updateTerminalDetails(_0x1255fd);}catch{try{const _0x4fdb27=await API['getContainers'](),_0x4dae1c=(_0x4fdb27['containers']||[])['find'](_0x1aaa65=>_0x1aaa65['containerId']===_0x397ed2||_0x1aaa65['name']===_0x397ed2||_0x1aaa65['fullContainerId']?.['startsWith'](_0x397ed2));if(_0x4dae1c){updateTerminalDetails(_0x4dae1c);return;}subtitleEl['textContent']='Container\x20'+_0x397ed2['substring'](0x0,0xc);}catch{subtitleEl['textContent']='Container\x20'+_0x397ed2['substring'](0x0,0xc);}}}function updateTerminalDetails(_0x15ad9d){const _0x3c3591=_0x15ad9d['containerId']||_0x15ad9d['name']||_0x15ad9d['fullContainerId']||containerId,_0x5bc4cf=_0x15ad9d['environment']?'Environment\x20'+_0x15ad9d['environment']:'Environment\x20unknown',_0x265c57=_0x15ad9d['defaultAgent']?'\x20·\x20Agent\x20'+_0x15ad9d['defaultAgent']:'',_0x1bbcc0=_0x15ad9d['createdAt']?'\x20·\x20Started\x20'+Utils['formatRelativeTime'](_0x15ad9d['createdAt']):'';titleEl['textContent']='Terminal\x20·\x20'+_0x3c3591['substring'](0x0,0xc);const _0x38ae0e=command?'\x20·\x20Command\x20'+command:'';subtitleEl['textContent']=''+_0x5bc4cf+_0x265c57+_0x1bbcc0+_0x38ae0e;}
@@ -1 +1 @@
1
- const THEME_STORAGE_KEY='profound-coder-theme',THEME_CHANGE_EVENT='profound-theme-change',VALID_THEMES=new Set(['light','dark']),DEFAULT_TARGET_SELECTORS=['#theme-toggle-container','.header-actions','.status-actions','.terminal-actions','.top-bar'];function readStoredTheme(){try{const _0x18e76e=localStorage['getItem'](THEME_STORAGE_KEY);if(_0x18e76e&&VALID_THEMES['has'](_0x18e76e))return _0x18e76e;}catch(_0x294998){console['warn']('[Theme]\x20Failed\x20to\x20read\x20stored\x20theme\x20preference:',_0x294998);}return null;}export function getPreferredTheme(){const _0x5520b9=readStoredTheme();if(_0x5520b9)return _0x5520b9;const _0x202851=window['matchMedia']&&window['matchMedia']('(prefers-color-scheme:\x20dark)')['matches'];return _0x202851?'dark':'light';}export function getCurrentTheme(){const _0x3fa883=document['documentElement']['dataset']['theme'];if(_0x3fa883&&VALID_THEMES['has'](_0x3fa883))return _0x3fa883;return getPreferredTheme();}export function applyTheme(_0x22fbba,{persist:persist=!![],emitEvent:emitEvent=!![]}={}){const _0x2f4196=VALID_THEMES['has'](_0x22fbba)?_0x22fbba:'light';document['documentElement']['dataset']['theme']=_0x2f4196,document['documentElement']['style']['colorScheme']=_0x2f4196==='dark'?'dark':'light';if(persist)try{localStorage['setItem'](THEME_STORAGE_KEY,_0x2f4196);}catch(_0x776490){console['warn']('[Theme]\x20Failed\x20to\x20persist\x20theme\x20preference:',_0x776490);}return emitEvent&&window['dispatchEvent'](new CustomEvent(THEME_CHANGE_EVENT,{'detail':{'theme':_0x2f4196}})),_0x2f4196;}export function initTheme(_0x46cc6e={}){const {emitEvent:emitEvent=![]}=_0x46cc6e;return applyTheme(getPreferredTheme(),{'persist':![],'emitEvent':emitEvent});}export function toggleTheme(){const _0x11d43a=getCurrentTheme()==='dark'?'light':'dark';return applyTheme(_0x11d43a);}export function onThemeChange(_0xfc8ed9,{runImmediately:runImmediately=![]}={}){const _0x33ac53=_0x47b0c6=>{_0xfc8ed9(_0x47b0c6['detail']['theme']);};return window['addEventListener'](THEME_CHANGE_EVENT,_0x33ac53),runImmediately&&_0xfc8ed9(getCurrentTheme()),()=>window['removeEventListener'](THEME_CHANGE_EVENT,_0x33ac53);}export function mountThemeToggle(_0x5deffd={}){const {targetSelectors:targetSelectors=DEFAULT_TARGET_SELECTORS}=_0x5deffd,_0x2291fe=targetSelectors['map'](_0xd9aea0=>document['querySelector'](_0xd9aea0))['find'](Boolean),_0x2870aa=document['createElement']('button');_0x2870aa['type']='button',_0x2870aa['className']='btn-ghost\x20theme-toggle-btn',_0x2870aa['id']='theme-toggle-btn',_0x2870aa['setAttribute']('aria-pressed','false'),_0x2870aa['style']['padding']='8px',_0x2870aa['style']['lineHeight']='0',_0x2870aa['style']['width']='36px',_0x2870aa['style']['height']='36px',_0x2870aa['style']['display']='flex',_0x2870aa['style']['alignItems']='center',_0x2870aa['style']['justifyContent']='center';const _0x49bc79=document['createElement']('span');_0x49bc79['className']='theme-icon',_0x49bc79['style']['display']='flex',_0x2870aa['append'](_0x49bc79);const _0x4f87f6='<svg\x20xmlns=\x22http://www.w3.org/2000/svg\x22\x20width=\x2220\x22\x20height=\x2220\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22><path\x20d=\x22M21\x2012.79A9\x209\x200\x201\x201\x2011.21\x203\x207\x207\x200\x200\x200\x2021\x2012.79z\x22></path></svg>',_0x370f47='<svg\x20xmlns=\x22http://www.w3.org/2000/svg\x22\x20width=\x2220\x22\x20height=\x2220\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22><circle\x20cx=\x2212\x22\x20cy=\x2212\x22\x20r=\x225\x22></circle><line\x20x1=\x2212\x22\x20y1=\x221\x22\x20x2=\x2212\x22\x20y2=\x223\x22></line><line\x20x1=\x2212\x22\x20y1=\x2221\x22\x20x2=\x2212\x22\x20y2=\x2223\x22></line><line\x20x1=\x224.22\x22\x20y1=\x224.22\x22\x20x2=\x225.64\x22\x20y2=\x225.64\x22></line><line\x20x1=\x2218.36\x22\x20y1=\x2218.36\x22\x20x2=\x2219.78\x22\x20y2=\x2219.78\x22></line><line\x20x1=\x221\x22\x20y1=\x2212\x22\x20x2=\x223\x22\x20y2=\x2212\x22></line><line\x20x1=\x2221\x22\x20y1=\x2212\x22\x20x2=\x2223\x22\x20y2=\x2212\x22></line><line\x20x1=\x224.22\x22\x20y1=\x2219.78\x22\x20x2=\x225.64\x22\x20y2=\x2218.36\x22></line><line\x20x1=\x2218.36\x22\x20y1=\x225.64\x22\x20x2=\x2219.78\x22\x20y2=\x224.22\x22></line></svg>',_0x5977ba=_0x41daa5=>{const _0x90dbf8=_0x41daa5==='dark';_0x49bc79['innerHTML']=_0x90dbf8?_0x4f87f6:_0x370f47,_0x2870aa['title']=_0x90dbf8?'Switch\x20to\x20light\x20mode':'Switch\x20to\x20dark\x20mode',_0x2870aa['dataset']['theme']=_0x41daa5,_0x2870aa['setAttribute']('aria-pressed',String(_0x90dbf8));};return _0x2870aa['addEventListener']('click',()=>{const _0x4190ab=getCurrentTheme()==='dark'?'light':'dark';applyTheme(_0x4190ab);}),_0x2291fe?_0x2291fe['appendChild'](_0x2870aa):(_0x2870aa['classList']['add']('theme-toggle-floating'),document['body']['appendChild'](_0x2870aa)),_0x5977ba(initTheme()),onThemeChange(_0x5977ba),_0x2870aa;}export{THEME_STORAGE_KEY,THEME_CHANGE_EVENT};
1
+ const THEME_STORAGE_KEY='profound-coder-theme',THEME_CHANGE_EVENT='profound-theme-change',VALID_THEMES=new Set(['light','dark']),DEFAULT_TARGET_SELECTORS=['#theme-toggle-container','.header-actions','.status-actions','.terminal-actions','.top-bar'];function readStoredTheme(){try{const _0x260210=localStorage['getItem'](THEME_STORAGE_KEY);if(_0x260210&&VALID_THEMES['has'](_0x260210))return _0x260210;}catch(_0xa14f5c){console['warn']('[Theme]\x20Failed\x20to\x20read\x20stored\x20theme\x20preference:',_0xa14f5c);}return null;}export function getPreferredTheme(){const _0x336f46=readStoredTheme();if(_0x336f46)return _0x336f46;const _0x182e81=window['matchMedia']&&window['matchMedia']('(prefers-color-scheme:\x20dark)')['matches'];return _0x182e81?'dark':'light';}export function getCurrentTheme(){const _0x35c4e5=document['documentElement']['dataset']['theme'];if(_0x35c4e5&&VALID_THEMES['has'](_0x35c4e5))return _0x35c4e5;return getPreferredTheme();}export function applyTheme(_0x2723e,{persist:persist=!![],emitEvent:emitEvent=!![]}={}){const _0x1a98c6=VALID_THEMES['has'](_0x2723e)?_0x2723e:'light';document['documentElement']['dataset']['theme']=_0x1a98c6,document['documentElement']['style']['colorScheme']=_0x1a98c6==='dark'?'dark':'light';if(persist)try{localStorage['setItem'](THEME_STORAGE_KEY,_0x1a98c6);}catch(_0x178c74){console['warn']('[Theme]\x20Failed\x20to\x20persist\x20theme\x20preference:',_0x178c74);}return emitEvent&&window['dispatchEvent'](new CustomEvent(THEME_CHANGE_EVENT,{'detail':{'theme':_0x1a98c6}})),_0x1a98c6;}export function initTheme(_0x19b3f0={}){const {emitEvent:emitEvent=![]}=_0x19b3f0;return applyTheme(getPreferredTheme(),{'persist':![],'emitEvent':emitEvent});}export function toggleTheme(){const _0x1914e7=getCurrentTheme()==='dark'?'light':'dark';return applyTheme(_0x1914e7);}export function onThemeChange(_0x86f1ff,{runImmediately:runImmediately=![]}={}){const _0xb88ed6=_0x5bb568=>{_0x86f1ff(_0x5bb568['detail']['theme']);};return window['addEventListener'](THEME_CHANGE_EVENT,_0xb88ed6),runImmediately&&_0x86f1ff(getCurrentTheme()),()=>window['removeEventListener'](THEME_CHANGE_EVENT,_0xb88ed6);}export function mountThemeToggle(_0x664dc2={}){const {targetSelectors:targetSelectors=DEFAULT_TARGET_SELECTORS}=_0x664dc2,_0x1d4a08=targetSelectors['map'](_0x272191=>document['querySelector'](_0x272191))['find'](Boolean),_0x3f3aaf=document['createElement']('button');_0x3f3aaf['type']='button',_0x3f3aaf['className']='btn-ghost\x20theme-toggle-btn',_0x3f3aaf['id']='theme-toggle-btn',_0x3f3aaf['setAttribute']('aria-pressed','false'),_0x3f3aaf['style']['padding']='8px',_0x3f3aaf['style']['lineHeight']='0',_0x3f3aaf['style']['width']='36px',_0x3f3aaf['style']['height']='36px',_0x3f3aaf['style']['display']='flex',_0x3f3aaf['style']['alignItems']='center',_0x3f3aaf['style']['justifyContent']='center';const _0x33d8ba=document['createElement']('span');_0x33d8ba['className']='theme-icon',_0x33d8ba['style']['display']='flex',_0x3f3aaf['append'](_0x33d8ba);const _0x17c295='<svg\x20xmlns=\x22http://www.w3.org/2000/svg\x22\x20width=\x2220\x22\x20height=\x2220\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22><path\x20d=\x22M21\x2012.79A9\x209\x200\x201\x201\x2011.21\x203\x207\x207\x200\x200\x200\x2021\x2012.79z\x22></path></svg>',_0x222d55='<svg\x20xmlns=\x22http://www.w3.org/2000/svg\x22\x20width=\x2220\x22\x20height=\x2220\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22><circle\x20cx=\x2212\x22\x20cy=\x2212\x22\x20r=\x225\x22></circle><line\x20x1=\x2212\x22\x20y1=\x221\x22\x20x2=\x2212\x22\x20y2=\x223\x22></line><line\x20x1=\x2212\x22\x20y1=\x2221\x22\x20x2=\x2212\x22\x20y2=\x2223\x22></line><line\x20x1=\x224.22\x22\x20y1=\x224.22\x22\x20x2=\x225.64\x22\x20y2=\x225.64\x22></line><line\x20x1=\x2218.36\x22\x20y1=\x2218.36\x22\x20x2=\x2219.78\x22\x20y2=\x2219.78\x22></line><line\x20x1=\x221\x22\x20y1=\x2212\x22\x20x2=\x223\x22\x20y2=\x2212\x22></line><line\x20x1=\x2221\x22\x20y1=\x2212\x22\x20x2=\x2223\x22\x20y2=\x2212\x22></line><line\x20x1=\x224.22\x22\x20y1=\x2219.78\x22\x20x2=\x225.64\x22\x20y2=\x2218.36\x22></line><line\x20x1=\x2218.36\x22\x20y1=\x225.64\x22\x20x2=\x2219.78\x22\x20y2=\x224.22\x22></line></svg>',_0x5c8e98=_0xb3e664=>{const _0x25f162=_0xb3e664==='dark';_0x33d8ba['innerHTML']=_0x25f162?_0x17c295:_0x222d55,_0x3f3aaf['title']=_0x25f162?'Switch\x20to\x20light\x20mode':'Switch\x20to\x20dark\x20mode',_0x3f3aaf['dataset']['theme']=_0xb3e664,_0x3f3aaf['setAttribute']('aria-pressed',String(_0x25f162));};return _0x3f3aaf['addEventListener']('click',()=>{const _0x41563c=getCurrentTheme()==='dark'?'light':'dark';applyTheme(_0x41563c);}),_0x1d4a08?_0x1d4a08['appendChild'](_0x3f3aaf):(_0x3f3aaf['classList']['add']('theme-toggle-floating'),document['body']['appendChild'](_0x3f3aaf)),_0x5c8e98(initTheme()),onThemeChange(_0x5c8e98),_0x3f3aaf;}export{THEME_STORAGE_KEY,THEME_CHANGE_EVENT};
@@ -22,8 +22,8 @@
22
22
  }
23
23
  })();
24
24
  </script>
25
- <link rel="stylesheet" href="styles.css?v=83">
26
- <script type="module" src="users.js?v=60"></script>
25
+ <link rel="stylesheet" href="styles.css?v=84">
26
+ <script type="module" src="users.js?v=61"></script>
27
27
  <!-- Dev QA shortcut: Ctrl+Shift+Q to launch current page in QA mode -->
28
28
  </head>
29
29
  <body>
@@ -1 +1 @@
1
- import{API,Utils,ConfirmDialog,initializeAdminMenu}from'./app.js';import{initTheme,mountThemeToggle}from'./theme.js';const state={'users':[],'editingUserId':null,'bindingsUserId':null,'deletingUserId':null,'setupLinkData':null,'emailConfigured':![],'currentUser':null,'roles':[],'environments':[],'teams':[],'userBindings':[],'bindingsDirty':![]};document['addEventListener']('DOMContentLoaded',async()=>{initTheme(),mountThemeToggle(),await checkAdminAccess(),await initializeAdminMenu(state['currentUser']),await Promise['all']([checkEmailConfig(),loadRoles(),loadEnvironments(),loadTeams()]),bindEventListeners(),await loadUsers();});async function checkAdminAccess(){try{const _0x2d01af=await API['getCurrentUser']();state['currentUser']=_0x2d01af;if(!_0x2d01af['hasAllPermissions']&&!_0x2d01af['canCreateUsers']){Utils['showToast']('Access\x20denied:\x20Server\x20admin\x20privileges\x20required','error'),setTimeout(()=>{window['location']['href']='index.html';},0x7d0);return;}}catch(_0x43a9be){window['location']['href']='login.html';}}async function checkEmailConfig(){try{const _0x5f227a=await API['getEmailConfig']();state['emailConfigured']=_0x5f227a['configured'];}catch(_0x35d43f){console['error']('Failed\x20to\x20check\x20email\x20config',_0x35d43f),state['emailConfigured']=![];}}async function loadRoles(){try{const {roles:_0x476f35}=await API['getRoleDefinitions']();state['roles']=_0x476f35||[];}catch{state['roles']=[];}}async function loadEnvironments(){try{const _0x12225c=await API['getEnvironments']();state['environments']=_0x12225c['environments']||[];}catch{state['environments']=[];}}async function loadTeams(){try{const {teams:_0x132fb8}=await API['getTeams']();state['teams']=_0x132fb8||[];}catch{state['teams']=[];}}function bindEventListeners(){document['getElementById']('create-user-btn')?.['addEventListener']('click',showCreateUserModal),document['getElementById']('close-user-modal')?.['addEventListener']('click',hideUserModal),document['getElementById']('cancel-user')?.['addEventListener']('click',hideUserModal),document['getElementById']('save-user')?.['addEventListener']('click',saveUser),document['querySelector']('#user-modal\x20.modal-overlay')?.['addEventListener']('click',hideUserModal),document['getElementById']('close-delete-modal')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('cancel-delete')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('confirm-delete')?.['addEventListener']('click',confirmDeleteUser),document['querySelector']('#delete-modal\x20.modal-overlay')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('close-setup-link-modal')?.['addEventListener']('click',hideSetupLinkModal),document['getElementById']('close-setup-link')?.['addEventListener']('click',hideSetupLinkModal),document['getElementById']('copy-setup-link')?.['addEventListener']('click',copySetupLink),document['getElementById']('email-setup-link')?.['addEventListener']('click',emailSetupLink),document['querySelector']('#setup-link-modal\x20.modal-overlay')?.['addEventListener']('click',hideSetupLinkModal),document['getElementById']('skip-password')?.['addEventListener']('change',handleSkipPasswordChange),document['getElementById']('retry-users-btn')?.['addEventListener']('click',loadUsers),document['getElementById']('close-bindings-modal')?.['addEventListener']('click',hideAccessBindingsModal),document['getElementById']('close-bindings-modal-footer')?.['addEventListener']('click',hideAccessBindingsModal),document['querySelector']('#bindings-modal\x20.modal-overlay')?.['addEventListener']('click',hideAccessBindingsModal),document['getElementById']('add-user-binding-btn')?.['addEventListener']('click',()=>{document['getElementById']('add-user-binding-form')['hidden']=![],document['getElementById']('add-user-binding-btn')['hidden']=!![],populateUserBindingForm();}),document['getElementById']('cancel-user-binding-btn')?.['addEventListener']('click',()=>{document['getElementById']('add-user-binding-form')['hidden']=!![],document['getElementById']('add-user-binding-btn')['hidden']=![];}),document['getElementById']('save-user-binding-btn')?.['addEventListener']('click',saveUserBinding),document['getElementById']('user-binding-resource-type')?.['addEventListener']('change',onUserBindingResourceTypeChange);}async function loadUsers(){const _0x4b5cea=document['getElementById']('users-loading'),_0x224ba3=document['getElementById']('users-error'),_0x3aaba8=document['getElementById']('users-content');try{_0x4b5cea['hidden']=![],_0x224ba3['hidden']=!![],_0x3aaba8['hidden']=!![];const _0x45e8c9=await API['getUsers']();state['users']=_0x45e8c9['users']||[],renderUsers(),_0x4b5cea['hidden']=!![],_0x3aaba8['hidden']=![];}catch(_0x4ae961){console['error']('Failed\x20to\x20load\x20users',_0x4ae961),_0x4b5cea['hidden']=!![],_0x224ba3['hidden']=![],document['getElementById']('users-error-message')['textContent']=_0x4ae961['message'];}}function renderUsers(){const _0x94bd4e=document['getElementById']('users-table-body'),_0x152404=document['getElementById']('user-count');if(!_0x94bd4e)return;_0x152404&&(_0x152404['textContent']=state['users']['length']+'\x20user'+(state['users']['length']===0x1?'':'s'));const _0x24819a=[...state['users']]['sort']((_0x2ea902,_0x4244f4)=>{const _0x5e874e=new Date(_0x2ea902['created_at'])['getTime'](),_0x71aa45=new Date(_0x4244f4['created_at'])['getTime']();return _0x71aa45-_0x5e874e;});_0x94bd4e['innerHTML']=_0x24819a['map'](_0x5f4a19=>'\x0a\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22user-username\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+Utils['escapeHtml'](_0x5f4a19['username'])+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['escapeHtml'](_0x5f4a19['name'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['escapeHtml'](_0x5f4a19['email'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22access-badges\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x5f4a19['isServerAdmin']?'<span\x20class=\x22role-badge\x20role-admin\x22>Server\x20Admin</span>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x5f4a19['hasUserBinding']?'<span\x20class=\x22role-badge\x20role-user-binding\x22>User\x20Binding</span>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x5f4a19['hasTeamBinding']?'<span\x20class=\x22role-badge\x20role-team-binding\x22>Team\x20Binding</span>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['formatRelativeTime'](_0x5f4a19['created_at'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22user-actions\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-icon\x20btn-small\x22\x20onclick=\x22window.editUser(\x27'+_0x5f4a19['id']+'\x27)\x22\x20title=\x22Edit\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2214\x22\x20height=\x2214\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M11\x204H4a2\x202\x200\x200\x200-2\x202v14a2\x202\x200\x200\x200\x202\x202h14a2\x202\x200\x200\x200\x202-2v-7\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M18.5\x202.5a2.121\x202.121\x200\x200\x201\x203\x203L12\x2015l-4\x201\x201-4\x209.5-9.5z\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-icon\x20btn-small\x22\x20onclick=\x22window.showAccessBindings(\x27'+_0x5f4a19['id']+'\x27)\x22\x20title=\x22Access\x20Bindings\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2214\x22\x20height=\x2214\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M12\x2022s8-4\x208-10V5l-8-3-8\x203v7c0\x206\x208\x2010\x208\x2010z\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-icon\x20btn-small\x20btn-danger\x22\x20onclick=\x22window.deleteUser(\x27'+_0x5f4a19['id']+'\x27)\x22\x20title=\x22Delete\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2214\x22\x20height=\x2214\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<polyline\x20points=\x223\x206\x205\x206\x2021\x206\x22></polyline>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M19\x206v14a2\x202\x200\x200\x201-2\x202H7a2\x202\x200\x200\x201-2-2V6m3\x200V4a2\x202\x200\x200\x201\x202-2h4a2\x202\x200\x200\x201\x202\x202v2\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-secondary\x20btn-small\x22\x20onclick=\x22window.resetPassword(\x27'+_0x5f4a19['id']+'\x27)\x22>Reset\x20Password</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20</tr>\x0a\x20\x20')['join']('');}window['showAccessBindings']=function(_0x95445d){const _0x40f8ff=state['users']['find'](_0x9b56a3=>_0x9b56a3['id']===_0x95445d);if(!_0x40f8ff)return;state['bindingsUserId']=_0x95445d,state['bindingsDirty']=![];const _0x35a782=document['getElementById']('bindings-modal'),_0x14dcdd=document['getElementById']('bindings-modal-title');_0x14dcdd['textContent']='Access\x20Bindings\x20—\x20'+_0x40f8ff['name'],document['getElementById']('add-user-binding-form')['hidden']=!![],document['getElementById']('add-user-binding-btn')['hidden']=![],_0x35a782['hidden']=![],loadUserBindings(_0x95445d);};function hasUnsavedBindingChanges(){const _0x1b8fea=document['getElementById']('add-user-binding-form');return _0x1b8fea&&!_0x1b8fea['hidden'];}async function hideAccessBindingsModal(){if(hasUnsavedBindingChanges()){if(!confirm('You\x20have\x20an\x20unsaved\x20binding.\x20Discard\x20changes?'))return;document['getElementById']('add-user-binding-form')['hidden']=!![],document['getElementById']('add-user-binding-btn')['hidden']=![];}const _0x3ac181=document['getElementById']('bindings-modal');_0x3ac181['hidden']=!![];const _0x234169=state['bindingsDirty'];state['bindingsUserId']=null,state['userBindings']=[],state['bindingsDirty']=![],_0x234169&&await loadUsers();}async function loadUserBindings(_0x2fb3a7){const _0x59e63a=document['getElementById']('user-bindings-loading'),_0x50b6d8=document['getElementById']('user-bindings-empty'),_0x126c5b=document['getElementById']('user-bindings-list-container'),_0x4eed68=document['getElementById']('add-user-binding-btn'),_0x28c0b9=document['getElementById']('user-team-bindings-note'),_0x2bfbd4=document['getElementById']('user-team-bindings-text');_0x59e63a['hidden']=![],_0x50b6d8['hidden']=!![],_0x126c5b['style']['display']='none',_0x4eed68['hidden']=!![],_0x28c0b9['hidden']=!![],document['getElementById']('add-user-binding-form')['hidden']=!![];try{const [_0xaecf37,_0x5b12f3]=await Promise['all']([API['getBindings']({'subject_type':'user','subject_id':_0x2fb3a7}),API['getTeams']()]);state['userBindings']=_0xaecf37['bindings']||[];const _0x3cc36e=(_0x5b12f3['teams']||[])['filter'](_0x25ddc7=>(_0x25ddc7['members']||[])['includes'](_0x2fb3a7)),_0x19d1db=[];if(_0x3cc36e['length']>0x0){const _0x5ef884=await Promise['all'](_0x3cc36e['map'](_0x57f3c1=>API['getBindings']({'subject_type':'team','subject_id':_0x57f3c1['id']})));for(let _0x277593=0x0;_0x277593<_0x3cc36e['length'];_0x277593++){const _0x21da2e=_0x5ef884[_0x277593]['bindings']||[];_0x21da2e['length']>0x0&&_0x19d1db['push'](_0x3cc36e[_0x277593]);}}_0x59e63a['hidden']=!![];state['userBindings']['length']===0x0?_0x50b6d8['hidden']=![]:(_0x126c5b['style']['display']='',renderUserBindingsTable());if(_0x19d1db['length']>0x0){const _0x2d611a=_0x19d1db['map'](_0x2d43f4=>_0x2d43f4['name']),_0x158ce4=_0x2d611a['length']<=0x3?_0x2d611a['join'](',\x20'):_0x2d611a['slice'](0x0,0x3)['join'](',\x20')+(',\x20and\x20'+(_0x2d611a['length']-0x3)+'\x20more');_0x2bfbd4['textContent']='This\x20user\x20also\x20has\x20access\x20via\x20team\x20memberships\x20('+_0x158ce4+').';}else _0x2bfbd4['textContent']='This\x20user\x20has\x20no\x20access\x20via\x20team\x20memberships.';_0x28c0b9['hidden']=![],_0x4eed68['hidden']=![];}catch(_0x163b75){_0x59e63a['hidden']=!![],Utils['showToast']('Failed\x20to\x20load\x20bindings:\x20'+_0x163b75['message'],'error');}}function renderUserBindingsTable(){const _0x1a0fc5=document['getElementById']('user-bindings-table-body');if(!_0x1a0fc5)return;_0x1a0fc5['innerHTML']=state['userBindings']['map'](_0x27c24b=>{const _0x15ea51=_0x27c24b['resource_type']==='server'?'Server':_0x27c24b['resource_id']==='*'?'All\x20'+_0x27c24b['resource_type']+'s':Utils['escapeHtml'](_0x27c24b['resource_id']||'');return'\x0a\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20<td>'+_0x15ea51+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['escapeHtml'](_0x27c24b['role_name']||_0x27c24b['role_id'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-ghost\x20btn-small\x22\x20onclick=\x22window.deleteUserBindingClick(\x27'+_0x27c24b['id']+'\x27)\x22\x20style=\x22color:\x20var(--color-danger);\x22>Remove</button>\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20</tr>';})['join']('');}function populateUserBindingForm(){const _0x1dc4ba=document['getElementById']('user-binding-resource-type');_0x1dc4ba['value']='environment',updateUserBindingFormForResourceType('environment');}function onUserBindingResourceTypeChange(){const _0x33333a=document['getElementById']('user-binding-resource-type')['value'];updateUserBindingFormForResourceType(_0x33333a);}function updateUserBindingFormForResourceType(_0x23ce98){const _0x4d22e3=document['getElementById']('user-binding-resource-container'),_0x33a5a5=document['getElementById']('user-binding-resource'),_0x1ba4ad=document['getElementById']('user-binding-role');if(_0x23ce98==='server')_0x4d22e3['style']['display']='none';else{_0x4d22e3['style']['display']='';if(_0x23ce98==='environment'){const _0x4a8492=new Set(state['userBindings']['filter'](_0x2d373f=>_0x2d373f['resource_type']==='environment')['map'](_0xe7db36=>_0xe7db36['resource_id'])),_0xa89039=state['environments']['filter'](_0x109a9d=>!_0x4a8492['has'](_0x109a9d['name']));_0x33a5a5['innerHTML']=_0xa89039['length']?_0xa89039['map'](_0x4503bf=>'<option\x20value=\x22'+Utils['escapeHtml'](_0x4503bf['name'])+'\x22>'+Utils['escapeHtml'](_0x4503bf['name'])+'</option>')['join'](''):'<option\x20value=\x22\x22>All\x20environments\x20already\x20bound</option>';}else{if(_0x23ce98==='team'){const _0x29841f=new Set(state['userBindings']['filter'](_0x31a07f=>_0x31a07f['resource_type']==='team')['map'](_0x751882=>_0x751882['resource_id'])),_0x4f1b67=state['teams']['filter'](_0x2968d8=>!_0x29841f['has'](_0x2968d8['id']));_0x33a5a5['innerHTML']=_0x4f1b67['length']?_0x4f1b67['map'](_0x3e75e8=>'<option\x20value=\x22'+_0x3e75e8['id']+'\x22>'+Utils['escapeHtml'](_0x3e75e8['name'])+'</option>')['join'](''):'<option\x20value=\x22\x22>All\x20teams\x20already\x20bound</option>';}}}const _0x14caed=state['roles']['filter'](_0x3e2dd6=>_0x3e2dd6['resource_type']===_0x23ce98);_0x1ba4ad['innerHTML']=_0x14caed['length']?_0x14caed['filter'](_0x485534=>{const _0x54d45a=_0x23ce98==='server'?undefined:_0x33a5a5['value'];return!state['userBindings']['some'](_0x4204b5=>_0x4204b5['resource_type']===_0x23ce98&&_0x4204b5['role_id']===_0x485534['id']&&(_0x23ce98==='server'||_0x4204b5['resource_id']===_0x54d45a));})['map'](_0x5d1aed=>'<option\x20value=\x22'+_0x5d1aed['id']+'\x22>'+Utils['escapeHtml'](_0x5d1aed['name'])+'</option>')['join']('')||'<option\x20value=\x22\x22>No\x20available\x20roles</option>':'<option\x20value=\x22\x22>No\x20roles\x20for\x20this\x20resource\x20type</option>';}async function saveUserBinding(){const _0x511957=document['getElementById']('user-binding-resource-type')['value'],_0x30ba74=document['getElementById']('user-binding-role')['value'];if(!_0x30ba74||!state['bindingsUserId'])return;const _0x321649=_0x511957==='server'?undefined:document['getElementById']('user-binding-resource')['value'];if(_0x511957!=='server'&&!_0x321649)return;const _0x351286=document['getElementById']('save-user-binding-btn');_0x351286['disabled']=!![],_0x351286['textContent']='Adding…';try{const _0x296ee4={'subject_type':'user','subject_id':state['bindingsUserId'],'role_id':_0x30ba74,'resource_type':_0x511957};if(_0x321649)_0x296ee4['resource_id']=_0x321649;await API['createBinding'](_0x296ee4),Utils['showToast']('Binding\x20added','success'),state['bindingsDirty']=!![],document['getElementById']('add-user-binding-form')['hidden']=!![],document['getElementById']('add-user-binding-btn')['hidden']=![],await loadUserBindings(state['bindingsUserId']);}catch(_0x3f8ef8){Utils['showToast'](_0x3f8ef8['message'],'error');}finally{_0x351286['disabled']=![],_0x351286['textContent']='Add\x20Binding';}}window['deleteUserBindingClick']=async function(_0x172a65){try{await API['deleteBinding'](_0x172a65),Utils['showToast']('Binding\x20removed','success'),state['bindingsDirty']=!![],await loadUserBindings(state['bindingsUserId']);}catch(_0x239ca7){Utils['showToast'](_0x239ca7['message'],'error');}};function showCreateUserModal(){state['editingUserId']=null;const _0x245feb=document['getElementById']('user-modal'),_0x10e634=document['getElementById']('modal-title'),_0x3d72c5=document['getElementById']('user-password'),_0x92e00e=document['getElementById']('password-required'),_0x81ca7d=document['getElementById']('password-hint'),_0x513dd5=document['getElementById']('skip-password-container'),_0x105a2a=document['getElementById']('skip-password');_0x10e634['textContent']='Create\x20User',document['getElementById']('user-form')['reset'](),_0x3d72c5['required']=!![];if(_0x92e00e)_0x92e00e['textContent']='*';_0x513dd5&&(_0x513dd5['style']['display']='block',_0x105a2a['checked']=![]),_0x81ca7d&&(_0x81ca7d['textContent']='Minimum\x208\x20characters'),_0x245feb['hidden']=![],setTimeout(()=>{document['getElementById']('user-username')['focus']();},0x64);}window['editUser']=function(_0x306f4b){const _0x1bddac=state['users']['find'](_0x4189ba=>_0x4189ba['id']===_0x306f4b);if(!_0x1bddac)return;state['editingUserId']=_0x306f4b;const _0xb469dd=document['getElementById']('user-modal'),_0x58e70d=document['getElementById']('modal-title'),_0x44776c=document['getElementById']('user-password'),_0x106b34=document['getElementById']('password-required'),_0x5022e1=document['getElementById']('password-hint'),_0x617685=document['getElementById']('skip-password-container');_0x58e70d['textContent']='Edit\x20User',document['getElementById']('user-username')['value']=_0x1bddac['username'],document['getElementById']('user-name')['value']=_0x1bddac['name'],document['getElementById']('user-email')['value']=_0x1bddac['email'],_0x44776c['value']='',_0x44776c['required']=![];if(_0x106b34)_0x106b34['textContent']='';_0x617685&&(_0x617685['style']['display']='none'),_0x5022e1&&(_0x5022e1['textContent']='Leave\x20blank\x20to\x20keep\x20current\x20password'),_0xb469dd['hidden']=![],setTimeout(()=>{document['getElementById']('user-username')['focus']();},0x64);};function hideUserModal(){const _0x11419c=document['getElementById']('user-modal'),_0x393f71=document['getElementById']('form-error'),_0x38a5d8=document['getElementById']('save-user');_0x11419c['hidden']=!![],_0x393f71['hidden']=!![],state['editingUserId']=null,_0x38a5d8&&(_0x38a5d8['disabled']=![],_0x38a5d8['textContent']='Save\x20User');}async function saveUser(){const _0x1d21d5=document['getElementById']('user-form'),_0x353795=document['getElementById']('skip-password')?.['checked']||![];if(!_0x353795&&!_0x1d21d5['checkValidity']()){_0x1d21d5['reportValidity']();return;}const _0x24e032=document['getElementById']('form-error'),_0x257802=document['getElementById']('save-user'),_0x3e278a=document['getElementById']('user-username')['value']['trim'](),_0x2a05d7=document['getElementById']('user-name')['value']['trim'](),_0x16adb2=document['getElementById']('user-email')['value']['trim'](),_0x1d1afd=document['getElementById']('user-password')['value'];if(!_0x3e278a||!_0x2a05d7||!_0x16adb2){_0x24e032['textContent']='Please\x20fill\x20in\x20all\x20required\x20fields',_0x24e032['hidden']=![],_0x24e032['scrollIntoView']({'behavior':'smooth','block':'nearest'});return;}_0x24e032['hidden']=!![];const _0x416861=_0x257802['textContent'];_0x257802['disabled']=!![],_0x257802['textContent']=state['editingUserId']?'Updating...':'Creating...';try{if(state['editingUserId']){const _0x4cd9fa={'username':_0x3e278a,'name':_0x2a05d7,'email':_0x16adb2};_0x1d1afd&&(_0x4cd9fa['password']=_0x1d1afd),await API['updateUser'](state['editingUserId'],_0x4cd9fa),Utils['showToast']('User\x20updated\x20successfully','success'),await loadUsers(),hideUserModal();}else{const _0x332f63={'username':_0x3e278a,'name':_0x2a05d7,'email':_0x16adb2};if(!_0x353795){if(!_0x1d1afd)throw new Error('Password\x20is\x20required\x20for\x20new\x20users');_0x332f63['password']=_0x1d1afd;}const _0x247a3d=await API['createUser'](_0x332f63),_0x4777b7=_0x247a3d['user'];if(_0x353795||!_0x1d1afd){const _0x52b63e=await API['createPasswordToken'](_0x4777b7['id'],'setup',![]);state['setupLinkData']={'username':_0x4777b7['username'],'email':_0x4777b7['email'],'url':_0x52b63e['token']['setupUrl'],'userId':_0x4777b7['id'],'type':'setup','emailSent':![]},await loadUsers(),hideUserModal(),showSetupLinkModal();}else Utils['showToast']('User\x20created\x20successfully','success'),await loadUsers(),hideUserModal();}}catch(_0x16da8d){console['error']('Failed\x20to\x20save\x20user',_0x16da8d),_0x24e032['textContent']=_0x16da8d['message'],_0x24e032['hidden']=![],_0x24e032['scrollIntoView']({'behavior':'smooth','block':'nearest'}),_0x257802['disabled']=![],_0x257802['textContent']=_0x416861;}}window['deleteUser']=function(_0x372fa9){const _0x1c2bd4=state['users']['find'](_0x1a6b99=>_0x1a6b99['id']===_0x372fa9);if(!_0x1c2bd4)return;state['deletingUserId']=_0x372fa9;const _0x287c6e=document['getElementById']('delete-modal'),_0x2814d4=document['getElementById']('delete-user-name');_0x2814d4['textContent']=_0x1c2bd4['name']+'\x20('+_0x1c2bd4['username']+')',_0x287c6e['hidden']=![];};function hideDeleteModal(){const _0x5bbd58=document['getElementById']('delete-modal');_0x5bbd58['hidden']=!![],state['deletingUserId']=null;const _0x2af55a=document['getElementById']('confirm-delete');_0x2af55a&&(_0x2af55a['disabled']=![],_0x2af55a['textContent']='Delete\x20User');}async function confirmDeleteUser(){if(!state['deletingUserId'])return;const _0x47afea=document['getElementById']('confirm-delete'),_0x362e9b=_0x47afea['textContent'];_0x47afea['disabled']=!![],_0x47afea['textContent']='Deleting...';try{await API['deleteUser'](state['deletingUserId']),Utils['showToast']('User\x20deleted\x20successfully','success'),await loadUsers(),hideDeleteModal();}catch(_0x2f6896){console['error']('Failed\x20to\x20delete\x20user',_0x2f6896),Utils['showToast']('Failed\x20to\x20delete\x20user:\x20'+_0x2f6896['message'],'error'),_0x47afea['disabled']=![],_0x47afea['textContent']=_0x362e9b;}}function handleSkipPasswordChange(_0x453cba){const _0x2e530f=document['getElementById']('user-password'),_0x55f694=_0x453cba['target']['checked'];_0x55f694?(_0x2e530f['required']=![],_0x2e530f['disabled']=!![],_0x2e530f['value']=''):(_0x2e530f['required']=!![],_0x2e530f['disabled']=![]);}function showSetupLinkModal(){if(!state['setupLinkData'])return;const _0x48b73c=document['getElementById']('setup-link-modal'),_0x1a33e4=document['getElementById']('setup-link-username'),_0x3e9452=document['getElementById']('setup-link-url'),_0x3b0a3c=document['getElementById']('email-setup-link'),_0x2b7bb1=document['getElementById']('email-button-text');_0x1a33e4['textContent']=state['setupLinkData']['username'],_0x3e9452['textContent']=state['setupLinkData']['url'],_0x3b0a3c&&_0x2b7bb1&&(_0x3b0a3c['disabled']=![],_0x2b7bb1['textContent']='Email\x20Link',state['emailConfigured']&&!state['setupLinkData']['emailSent']?_0x3b0a3c['style']['display']='inline-block':_0x3b0a3c['style']['display']='none'),_0x48b73c['hidden']=![];}function hideSetupLinkModal(){const _0x2d4284=document['getElementById']('setup-link-modal');_0x2d4284['hidden']=!![],state['setupLinkData']=null;}async function copySetupLink(){if(!state['setupLinkData'])return;try{await navigator['clipboard']['writeText'](state['setupLinkData']['url']),Utils['showToast']('Link\x20copied\x20to\x20clipboard','success');}catch(_0xf2888f){console['error']('Failed\x20to\x20copy\x20link',_0xf2888f),Utils['showToast']('Failed\x20to\x20copy\x20link','error');}}async function emailSetupLink(){if(!state['setupLinkData'])return;const _0x4303e9=document['getElementById']('email-setup-link'),_0x26bf27=document['getElementById']('email-button-text');if(!_0x4303e9||!_0x26bf27)return;const _0x3bfd7a=_0x26bf27['textContent'];_0x4303e9['disabled']=!![],_0x26bf27['textContent']='Sending...';try{const _0x489d5e=await API['createPasswordToken'](state['setupLinkData']['userId'],state['setupLinkData']['type'],!![]);if(_0x489d5e['emailSent'])Utils['showToast']('Email\x20sent\x20to\x20'+state['setupLinkData']['email'],'success'),state['setupLinkData']['emailSent']=!![],_0x4303e9['style']['display']='none';else{const _0x6a647b=_0x489d5e['emailError']||'Failed\x20to\x20send\x20email';Utils['showToast'](_0x6a647b,'error'),_0x4303e9['disabled']=![],_0x26bf27['textContent']=_0x3bfd7a;}}catch(_0x41cbad){console['error']('Failed\x20to\x20email\x20setup\x20link',_0x41cbad),Utils['showToast']('Failed\x20to\x20send\x20email:\x20'+_0x41cbad['message'],'error'),_0x4303e9['disabled']=![],_0x26bf27['textContent']=_0x3bfd7a;}}window['resetPassword']=async function(_0xbbbcd7){const _0x3980a3=state['users']['find'](_0x50b7df=>_0x50b7df['id']===_0xbbbcd7);if(!_0x3980a3)return;const _0xfce42=await ConfirmDialog['show']({'title':'Reset\x20Password','message':'Generate\x20a\x20password\x20reset\x20link\x20for\x20'+_0x3980a3['name']+'\x20('+_0x3980a3['username']+')?\x20This\x20will\x20create\x20a\x20link\x20that\x20they\x20can\x20use\x20to\x20set\x20a\x20new\x20password.','confirmText':'Generate\x20Link'});if(!_0xfce42)return;try{const _0x33a47a=await API['createPasswordToken'](_0xbbbcd7,'reset',![]);state['setupLinkData']={'username':_0x3980a3['username'],'email':_0x3980a3['email'],'url':_0x33a47a['token']['setupUrl'],'userId':_0x3980a3['id'],'type':'reset','emailSent':![]},showSetupLinkModal();}catch(_0x519e18){console['error']('Failed\x20to\x20create\x20password\x20reset\x20link',_0x519e18),Utils['showToast']('Failed\x20to\x20create\x20password\x20reset\x20link:\x20'+_0x519e18['message'],'error');}};
1
+ import{API,Utils,ConfirmDialog,initializeAdminMenu}from'./app.js';import{initTheme,mountThemeToggle}from'./theme.js';const state={'users':[],'editingUserId':null,'bindingsUserId':null,'deletingUserId':null,'setupLinkData':null,'emailConfigured':![],'currentUser':null,'roles':[],'environments':[],'teams':[],'userBindings':[],'bindingsDirty':![]};document['addEventListener']('DOMContentLoaded',async()=>{initTheme(),mountThemeToggle(),await checkAdminAccess(),await initializeAdminMenu(state['currentUser']),await Promise['all']([checkEmailConfig(),loadRoles(),loadEnvironments(),loadTeams()]),bindEventListeners(),await loadUsers();});async function checkAdminAccess(){try{const _0x5b87fb=await API['getCurrentUser']();state['currentUser']=_0x5b87fb;if(!_0x5b87fb['hasAllPermissions']&&!_0x5b87fb['canCreateUsers']){Utils['showToast']('Access\x20denied:\x20Server\x20admin\x20privileges\x20required','error'),setTimeout(()=>{window['location']['href']='index.html';},0x7d0);return;}}catch(_0x147a01){window['location']['href']='login.html';}}async function checkEmailConfig(){try{const _0x15015c=await API['getEmailConfig']();state['emailConfigured']=_0x15015c['configured'];}catch(_0x3e16ac){console['error']('Failed\x20to\x20check\x20email\x20config',_0x3e16ac),state['emailConfigured']=![];}}async function loadRoles(){try{const {roles:_0x44d91c}=await API['getRoleDefinitions']();state['roles']=_0x44d91c||[];}catch{state['roles']=[];}}async function loadEnvironments(){try{const _0x4864e0=await API['getEnvironments']();state['environments']=_0x4864e0['environments']||[];}catch{state['environments']=[];}}async function loadTeams(){try{const {teams:_0xe8a975}=await API['getTeams']();state['teams']=_0xe8a975||[];}catch{state['teams']=[];}}function bindEventListeners(){document['getElementById']('create-user-btn')?.['addEventListener']('click',showCreateUserModal),document['getElementById']('close-user-modal')?.['addEventListener']('click',hideUserModal),document['getElementById']('cancel-user')?.['addEventListener']('click',hideUserModal),document['getElementById']('save-user')?.['addEventListener']('click',saveUser),document['querySelector']('#user-modal\x20.modal-overlay')?.['addEventListener']('click',hideUserModal),document['getElementById']('close-delete-modal')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('cancel-delete')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('confirm-delete')?.['addEventListener']('click',confirmDeleteUser),document['querySelector']('#delete-modal\x20.modal-overlay')?.['addEventListener']('click',hideDeleteModal),document['getElementById']('close-setup-link-modal')?.['addEventListener']('click',hideSetupLinkModal),document['getElementById']('close-setup-link')?.['addEventListener']('click',hideSetupLinkModal),document['getElementById']('copy-setup-link')?.['addEventListener']('click',copySetupLink),document['getElementById']('email-setup-link')?.['addEventListener']('click',emailSetupLink),document['querySelector']('#setup-link-modal\x20.modal-overlay')?.['addEventListener']('click',hideSetupLinkModal),document['getElementById']('skip-password')?.['addEventListener']('change',handleSkipPasswordChange),document['getElementById']('retry-users-btn')?.['addEventListener']('click',loadUsers),document['getElementById']('close-bindings-modal')?.['addEventListener']('click',hideAccessBindingsModal),document['getElementById']('close-bindings-modal-footer')?.['addEventListener']('click',hideAccessBindingsModal),document['querySelector']('#bindings-modal\x20.modal-overlay')?.['addEventListener']('click',hideAccessBindingsModal),document['getElementById']('add-user-binding-btn')?.['addEventListener']('click',()=>{document['getElementById']('add-user-binding-form')['hidden']=![],document['getElementById']('add-user-binding-btn')['hidden']=!![],populateUserBindingForm();}),document['getElementById']('cancel-user-binding-btn')?.['addEventListener']('click',()=>{document['getElementById']('add-user-binding-form')['hidden']=!![],document['getElementById']('add-user-binding-btn')['hidden']=![];}),document['getElementById']('save-user-binding-btn')?.['addEventListener']('click',saveUserBinding),document['getElementById']('user-binding-resource-type')?.['addEventListener']('change',onUserBindingResourceTypeChange);}async function loadUsers(){const _0x4311a8=document['getElementById']('users-loading'),_0x3ea49e=document['getElementById']('users-error'),_0x5d81c6=document['getElementById']('users-content');try{_0x4311a8['hidden']=![],_0x3ea49e['hidden']=!![],_0x5d81c6['hidden']=!![];const _0x4238ab=await API['getUsers']();state['users']=_0x4238ab['users']||[],renderUsers(),_0x4311a8['hidden']=!![],_0x5d81c6['hidden']=![];}catch(_0x431316){console['error']('Failed\x20to\x20load\x20users',_0x431316),_0x4311a8['hidden']=!![],_0x3ea49e['hidden']=![],document['getElementById']('users-error-message')['textContent']=_0x431316['message'];}}function renderUsers(){const _0x17801b=document['getElementById']('users-table-body'),_0x1dc717=document['getElementById']('user-count');if(!_0x17801b)return;_0x1dc717&&(_0x1dc717['textContent']=state['users']['length']+'\x20user'+(state['users']['length']===0x1?'':'s'));const _0xace311=[...state['users']]['sort']((_0x5c51f1,_0x5e9306)=>{const _0x357d67=new Date(_0x5c51f1['created_at'])['getTime'](),_0x1981a7=new Date(_0x5e9306['created_at'])['getTime']();return _0x1981a7-_0x357d67;});_0x17801b['innerHTML']=_0xace311['map'](_0x45de2f=>'\x0a\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22user-username\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+Utils['escapeHtml'](_0x45de2f['username'])+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['escapeHtml'](_0x45de2f['name'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['escapeHtml'](_0x45de2f['email'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22access-badges\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x45de2f['isServerAdmin']?'<span\x20class=\x22role-badge\x20role-admin\x22>Server\x20Admin</span>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x45de2f['hasUserBinding']?'<span\x20class=\x22role-badge\x20role-user-binding\x22>User\x20Binding</span>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x45de2f['hasTeamBinding']?'<span\x20class=\x22role-badge\x20role-team-binding\x22>Team\x20Binding</span>':'')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['formatRelativeTime'](_0x45de2f['created_at'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22user-actions\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-icon\x20btn-small\x22\x20onclick=\x22window.editUser(\x27'+_0x45de2f['id']+'\x27)\x22\x20title=\x22Edit\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2214\x22\x20height=\x2214\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M11\x204H4a2\x202\x200\x200\x200-2\x202v14a2\x202\x200\x200\x200\x202\x202h14a2\x202\x200\x200\x200\x202-2v-7\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M18.5\x202.5a2.121\x202.121\x200\x200\x201\x203\x203L12\x2015l-4\x201\x201-4\x209.5-9.5z\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-icon\x20btn-small\x22\x20onclick=\x22window.showAccessBindings(\x27'+_0x45de2f['id']+'\x27)\x22\x20title=\x22Access\x20Bindings\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2214\x22\x20height=\x2214\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M12\x2022s8-4\x208-10V5l-8-3-8\x203v7c0\x206\x208\x2010\x208\x2010z\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-icon\x20btn-small\x20btn-danger\x22\x20onclick=\x22window.deleteUser(\x27'+_0x45de2f['id']+'\x27)\x22\x20title=\x22Delete\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20width=\x2214\x22\x20height=\x2214\x22\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<polyline\x20points=\x223\x206\x205\x206\x2021\x206\x22></polyline>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M19\x206v14a2\x202\x200\x200\x201-2\x202H7a2\x202\x200\x200\x201-2-2V6m3\x200V4a2\x202\x200\x200\x201\x202-2h4a2\x202\x200\x200\x201\x202\x202v2\x22></path>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-secondary\x20btn-small\x22\x20onclick=\x22window.resetPassword(\x27'+_0x45de2f['id']+'\x27)\x22>Reset\x20Password</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20</tr>\x0a\x20\x20')['join']('');}window['showAccessBindings']=function(_0x4b9234){const _0x4a0623=state['users']['find'](_0x47acc8=>_0x47acc8['id']===_0x4b9234);if(!_0x4a0623)return;state['bindingsUserId']=_0x4b9234,state['bindingsDirty']=![];const _0x5d8a61=document['getElementById']('bindings-modal'),_0xaa3191=document['getElementById']('bindings-modal-title');_0xaa3191['textContent']='Access\x20Bindings\x20—\x20'+_0x4a0623['name'],document['getElementById']('add-user-binding-form')['hidden']=!![],document['getElementById']('add-user-binding-btn')['hidden']=![],_0x5d8a61['hidden']=![],loadUserBindings(_0x4b9234);};function hasUnsavedBindingChanges(){const _0x7c0fb7=document['getElementById']('add-user-binding-form');return _0x7c0fb7&&!_0x7c0fb7['hidden'];}async function hideAccessBindingsModal(){if(hasUnsavedBindingChanges()){if(!confirm('You\x20have\x20an\x20unsaved\x20binding.\x20Discard\x20changes?'))return;document['getElementById']('add-user-binding-form')['hidden']=!![],document['getElementById']('add-user-binding-btn')['hidden']=![];}const _0x4c4571=document['getElementById']('bindings-modal');_0x4c4571['hidden']=!![];const _0x26644b=state['bindingsDirty'];state['bindingsUserId']=null,state['userBindings']=[],state['bindingsDirty']=![],_0x26644b&&await loadUsers();}async function loadUserBindings(_0x2d7cae){const _0x40e17f=document['getElementById']('user-bindings-loading'),_0x30c444=document['getElementById']('user-bindings-empty'),_0x5125b6=document['getElementById']('user-bindings-list-container'),_0x169566=document['getElementById']('add-user-binding-btn'),_0x570af0=document['getElementById']('user-team-bindings-note'),_0x315b8b=document['getElementById']('user-team-bindings-text');_0x40e17f['hidden']=![],_0x30c444['hidden']=!![],_0x5125b6['style']['display']='none',_0x169566['hidden']=!![],_0x570af0['hidden']=!![],document['getElementById']('add-user-binding-form')['hidden']=!![];try{const [_0x11a8ab,_0x2ea740]=await Promise['all']([API['getBindings']({'subject_type':'user','subject_id':_0x2d7cae}),API['getTeams']()]);state['userBindings']=_0x11a8ab['bindings']||[];const _0x2b894a=(_0x2ea740['teams']||[])['filter'](_0x540e4a=>(_0x540e4a['members']||[])['includes'](_0x2d7cae)),_0x3154a4=[];if(_0x2b894a['length']>0x0){const _0x14c9f8=await Promise['all'](_0x2b894a['map'](_0x5686ec=>API['getBindings']({'subject_type':'team','subject_id':_0x5686ec['id']})));for(let _0x3f2f43=0x0;_0x3f2f43<_0x2b894a['length'];_0x3f2f43++){const _0x526282=_0x14c9f8[_0x3f2f43]['bindings']||[];_0x526282['length']>0x0&&_0x3154a4['push'](_0x2b894a[_0x3f2f43]);}}_0x40e17f['hidden']=!![];state['userBindings']['length']===0x0?_0x30c444['hidden']=![]:(_0x5125b6['style']['display']='',renderUserBindingsTable());if(_0x3154a4['length']>0x0){const _0xba8a69=_0x3154a4['map'](_0x25544d=>_0x25544d['name']),_0x8617=_0xba8a69['length']<=0x3?_0xba8a69['join'](',\x20'):_0xba8a69['slice'](0x0,0x3)['join'](',\x20')+(',\x20and\x20'+(_0xba8a69['length']-0x3)+'\x20more');_0x315b8b['textContent']='This\x20user\x20also\x20has\x20access\x20via\x20team\x20memberships\x20('+_0x8617+').';}else _0x315b8b['textContent']='This\x20user\x20has\x20no\x20access\x20via\x20team\x20memberships.';_0x570af0['hidden']=![],_0x169566['hidden']=![];}catch(_0x216475){_0x40e17f['hidden']=!![],Utils['showToast']('Failed\x20to\x20load\x20bindings:\x20'+_0x216475['message'],'error');}}function renderUserBindingsTable(){const _0xc0193a=document['getElementById']('user-bindings-table-body');if(!_0xc0193a)return;_0xc0193a['innerHTML']=state['userBindings']['map'](_0x1de130=>{const _0x294f80=_0x1de130['resource_type']==='server'?'Server':_0x1de130['resource_id']==='*'?'All\x20'+_0x1de130['resource_type']+'s':Utils['escapeHtml'](_0x1de130['resource_id']||'');return'\x0a\x20\x20\x20\x20<tr>\x0a\x20\x20\x20\x20\x20\x20<td>'+_0x294f80+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>'+Utils['escapeHtml'](_0x1de130['role_name']||_0x1de130['role_id'])+'</td>\x0a\x20\x20\x20\x20\x20\x20<td>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22btn-ghost\x20btn-small\x22\x20onclick=\x22window.deleteUserBindingClick(\x27'+_0x1de130['id']+'\x27)\x22\x20style=\x22color:\x20var(--color-danger);\x22>Remove</button>\x0a\x20\x20\x20\x20\x20\x20</td>\x0a\x20\x20\x20\x20</tr>';})['join']('');}function populateUserBindingForm(){const _0x350d4f=document['getElementById']('user-binding-resource-type');_0x350d4f['value']='environment',updateUserBindingFormForResourceType('environment');}function onUserBindingResourceTypeChange(){const _0x57d1f1=document['getElementById']('user-binding-resource-type')['value'];updateUserBindingFormForResourceType(_0x57d1f1);}function updateUserBindingFormForResourceType(_0x24bc6c){const _0x4a95a5=document['getElementById']('user-binding-resource-container'),_0x3d3613=document['getElementById']('user-binding-resource'),_0x400ca4=document['getElementById']('user-binding-role');if(_0x24bc6c==='server')_0x4a95a5['style']['display']='none';else{_0x4a95a5['style']['display']='';if(_0x24bc6c==='environment'){const _0x504e80=new Set(state['userBindings']['filter'](_0x4629df=>_0x4629df['resource_type']==='environment')['map'](_0x79b7f0=>_0x79b7f0['resource_id'])),_0x45c810=state['environments']['filter'](_0x10d274=>!_0x504e80['has'](_0x10d274['name']));_0x3d3613['innerHTML']=_0x45c810['length']?_0x45c810['map'](_0x2f2c2f=>'<option\x20value=\x22'+Utils['escapeHtml'](_0x2f2c2f['name'])+'\x22>'+Utils['escapeHtml'](_0x2f2c2f['name'])+'</option>')['join'](''):'<option\x20value=\x22\x22>All\x20environments\x20already\x20bound</option>';}else{if(_0x24bc6c==='team'){const _0x5057f8=new Set(state['userBindings']['filter'](_0x14dcf9=>_0x14dcf9['resource_type']==='team')['map'](_0x201d99=>_0x201d99['resource_id'])),_0x403145=state['teams']['filter'](_0x13c473=>!_0x5057f8['has'](_0x13c473['id']));_0x3d3613['innerHTML']=_0x403145['length']?_0x403145['map'](_0x23bc17=>'<option\x20value=\x22'+_0x23bc17['id']+'\x22>'+Utils['escapeHtml'](_0x23bc17['name'])+'</option>')['join'](''):'<option\x20value=\x22\x22>All\x20teams\x20already\x20bound</option>';}}}const _0x265984=state['roles']['filter'](_0x381b70=>_0x381b70['resource_type']===_0x24bc6c);_0x400ca4['innerHTML']=_0x265984['length']?_0x265984['filter'](_0x5af1cf=>{const _0x409af2=_0x24bc6c==='server'?undefined:_0x3d3613['value'];return!state['userBindings']['some'](_0x1f5eee=>_0x1f5eee['resource_type']===_0x24bc6c&&_0x1f5eee['role_id']===_0x5af1cf['id']&&(_0x24bc6c==='server'||_0x1f5eee['resource_id']===_0x409af2));})['map'](_0x83d067=>'<option\x20value=\x22'+_0x83d067['id']+'\x22>'+Utils['escapeHtml'](_0x83d067['name'])+'</option>')['join']('')||'<option\x20value=\x22\x22>No\x20available\x20roles</option>':'<option\x20value=\x22\x22>No\x20roles\x20for\x20this\x20resource\x20type</option>';}async function saveUserBinding(){const _0x4da06f=document['getElementById']('user-binding-resource-type')['value'],_0x119443=document['getElementById']('user-binding-role')['value'];if(!_0x119443||!state['bindingsUserId'])return;const _0x4d64e1=_0x4da06f==='server'?undefined:document['getElementById']('user-binding-resource')['value'];if(_0x4da06f!=='server'&&!_0x4d64e1)return;const _0x14825d=document['getElementById']('save-user-binding-btn');_0x14825d['disabled']=!![],_0x14825d['textContent']='Adding…';try{const _0x19a7af={'subject_type':'user','subject_id':state['bindingsUserId'],'role_id':_0x119443,'resource_type':_0x4da06f};if(_0x4d64e1)_0x19a7af['resource_id']=_0x4d64e1;await API['createBinding'](_0x19a7af),Utils['showToast']('Binding\x20added','success'),state['bindingsDirty']=!![],document['getElementById']('add-user-binding-form')['hidden']=!![],document['getElementById']('add-user-binding-btn')['hidden']=![],await loadUserBindings(state['bindingsUserId']);}catch(_0x42702d){Utils['showToast'](_0x42702d['message'],'error');}finally{_0x14825d['disabled']=![],_0x14825d['textContent']='Add\x20Binding';}}window['deleteUserBindingClick']=async function(_0x33b86e){try{await API['deleteBinding'](_0x33b86e),Utils['showToast']('Binding\x20removed','success'),state['bindingsDirty']=!![],await loadUserBindings(state['bindingsUserId']);}catch(_0x3a55fa){Utils['showToast'](_0x3a55fa['message'],'error');}};function showCreateUserModal(){state['editingUserId']=null;const _0x36514d=document['getElementById']('user-modal'),_0x43d04b=document['getElementById']('modal-title'),_0x12580e=document['getElementById']('user-password'),_0x2e6594=document['getElementById']('password-required'),_0x177b09=document['getElementById']('password-hint'),_0x364e33=document['getElementById']('skip-password-container'),_0x3c60af=document['getElementById']('skip-password');_0x43d04b['textContent']='Create\x20User',document['getElementById']('user-form')['reset'](),_0x12580e['required']=!![];if(_0x2e6594)_0x2e6594['textContent']='*';_0x364e33&&(_0x364e33['style']['display']='block',_0x3c60af['checked']=![]),_0x177b09&&(_0x177b09['textContent']='Minimum\x208\x20characters'),_0x36514d['hidden']=![],setTimeout(()=>{document['getElementById']('user-username')['focus']();},0x64);}window['editUser']=function(_0x1976bf){const _0x28ec4e=state['users']['find'](_0x1f7d75=>_0x1f7d75['id']===_0x1976bf);if(!_0x28ec4e)return;state['editingUserId']=_0x1976bf;const _0x1063b4=document['getElementById']('user-modal'),_0x53a0d9=document['getElementById']('modal-title'),_0x25cd34=document['getElementById']('user-password'),_0x1048e5=document['getElementById']('password-required'),_0x40e659=document['getElementById']('password-hint'),_0x5c2d02=document['getElementById']('skip-password-container');_0x53a0d9['textContent']='Edit\x20User',document['getElementById']('user-username')['value']=_0x28ec4e['username'],document['getElementById']('user-name')['value']=_0x28ec4e['name'],document['getElementById']('user-email')['value']=_0x28ec4e['email'],_0x25cd34['value']='',_0x25cd34['required']=![];if(_0x1048e5)_0x1048e5['textContent']='';_0x5c2d02&&(_0x5c2d02['style']['display']='none'),_0x40e659&&(_0x40e659['textContent']='Leave\x20blank\x20to\x20keep\x20current\x20password'),_0x1063b4['hidden']=![],setTimeout(()=>{document['getElementById']('user-username')['focus']();},0x64);};function hideUserModal(){const _0x47fd3c=document['getElementById']('user-modal'),_0x555efa=document['getElementById']('form-error'),_0x50a07b=document['getElementById']('save-user');_0x47fd3c['hidden']=!![],_0x555efa['hidden']=!![],state['editingUserId']=null,_0x50a07b&&(_0x50a07b['disabled']=![],_0x50a07b['textContent']='Save\x20User');}async function saveUser(){const _0x4bfb36=document['getElementById']('user-form'),_0x135a04=document['getElementById']('skip-password')?.['checked']||![];if(!_0x135a04&&!_0x4bfb36['checkValidity']()){_0x4bfb36['reportValidity']();return;}const _0x28d640=document['getElementById']('form-error'),_0x34ca78=document['getElementById']('save-user'),_0xdbf807=document['getElementById']('user-username')['value']['trim'](),_0x512cf1=document['getElementById']('user-name')['value']['trim'](),_0x4da729=document['getElementById']('user-email')['value']['trim'](),_0x4d94eb=document['getElementById']('user-password')['value'];if(!_0xdbf807||!_0x512cf1||!_0x4da729){_0x28d640['textContent']='Please\x20fill\x20in\x20all\x20required\x20fields',_0x28d640['hidden']=![],_0x28d640['scrollIntoView']({'behavior':'smooth','block':'nearest'});return;}_0x28d640['hidden']=!![];const _0xcf982e=_0x34ca78['textContent'];_0x34ca78['disabled']=!![],_0x34ca78['textContent']=state['editingUserId']?'Updating...':'Creating...';try{if(state['editingUserId']){const _0x4f6e70={'username':_0xdbf807,'name':_0x512cf1,'email':_0x4da729};_0x4d94eb&&(_0x4f6e70['password']=_0x4d94eb),await API['updateUser'](state['editingUserId'],_0x4f6e70),Utils['showToast']('User\x20updated\x20successfully','success'),await loadUsers(),hideUserModal();}else{const _0x466de6={'username':_0xdbf807,'name':_0x512cf1,'email':_0x4da729};if(!_0x135a04){if(!_0x4d94eb)throw new Error('Password\x20is\x20required\x20for\x20new\x20users');_0x466de6['password']=_0x4d94eb;}const _0x4ea3d9=await API['createUser'](_0x466de6),_0x428a3a=_0x4ea3d9['user'];if(_0x135a04||!_0x4d94eb){const _0x23dbbb=await API['createPasswordToken'](_0x428a3a['id'],'setup',![]);state['setupLinkData']={'username':_0x428a3a['username'],'email':_0x428a3a['email'],'url':_0x23dbbb['token']['setupUrl'],'userId':_0x428a3a['id'],'type':'setup','emailSent':![]},await loadUsers(),hideUserModal(),showSetupLinkModal();}else Utils['showToast']('User\x20created\x20successfully','success'),await loadUsers(),hideUserModal();}}catch(_0x57aa16){console['error']('Failed\x20to\x20save\x20user',_0x57aa16),_0x28d640['textContent']=_0x57aa16['message'],_0x28d640['hidden']=![],_0x28d640['scrollIntoView']({'behavior':'smooth','block':'nearest'}),_0x34ca78['disabled']=![],_0x34ca78['textContent']=_0xcf982e;}}window['deleteUser']=function(_0x14029f){const _0xcd4c20=state['users']['find'](_0x3dd8ad=>_0x3dd8ad['id']===_0x14029f);if(!_0xcd4c20)return;state['deletingUserId']=_0x14029f;const _0x58f450=document['getElementById']('delete-modal'),_0x471c17=document['getElementById']('delete-user-name');_0x471c17['textContent']=_0xcd4c20['name']+'\x20('+_0xcd4c20['username']+')',_0x58f450['hidden']=![];};function hideDeleteModal(){const _0x614db9=document['getElementById']('delete-modal');_0x614db9['hidden']=!![],state['deletingUserId']=null;const _0x2771b9=document['getElementById']('confirm-delete');_0x2771b9&&(_0x2771b9['disabled']=![],_0x2771b9['textContent']='Delete\x20User');}async function confirmDeleteUser(){if(!state['deletingUserId'])return;const _0x5664e8=document['getElementById']('confirm-delete'),_0x5a74ab=_0x5664e8['textContent'];_0x5664e8['disabled']=!![],_0x5664e8['textContent']='Deleting...';try{await API['deleteUser'](state['deletingUserId']),Utils['showToast']('User\x20deleted\x20successfully','success'),await loadUsers(),hideDeleteModal();}catch(_0x2bd84c){console['error']('Failed\x20to\x20delete\x20user',_0x2bd84c),Utils['showToast']('Failed\x20to\x20delete\x20user:\x20'+_0x2bd84c['message'],'error'),_0x5664e8['disabled']=![],_0x5664e8['textContent']=_0x5a74ab;}}function handleSkipPasswordChange(_0x32d8da){const _0x25362d=document['getElementById']('user-password'),_0x369f20=_0x32d8da['target']['checked'];_0x369f20?(_0x25362d['required']=![],_0x25362d['disabled']=!![],_0x25362d['value']=''):(_0x25362d['required']=!![],_0x25362d['disabled']=![]);}function showSetupLinkModal(){if(!state['setupLinkData'])return;const _0x21842f=document['getElementById']('setup-link-modal'),_0x4446ab=document['getElementById']('setup-link-username'),_0x20a1d0=document['getElementById']('setup-link-url'),_0x5c6e5a=document['getElementById']('email-setup-link'),_0x2fa3fa=document['getElementById']('email-button-text');_0x4446ab['textContent']=state['setupLinkData']['username'],_0x20a1d0['textContent']=state['setupLinkData']['url'],_0x5c6e5a&&_0x2fa3fa&&(_0x5c6e5a['disabled']=![],_0x2fa3fa['textContent']='Email\x20Link',state['emailConfigured']&&!state['setupLinkData']['emailSent']?_0x5c6e5a['style']['display']='inline-block':_0x5c6e5a['style']['display']='none'),_0x21842f['hidden']=![];}function hideSetupLinkModal(){const _0x549d30=document['getElementById']('setup-link-modal');_0x549d30['hidden']=!![],state['setupLinkData']=null;}async function copySetupLink(){if(!state['setupLinkData'])return;try{await navigator['clipboard']['writeText'](state['setupLinkData']['url']),Utils['showToast']('Link\x20copied\x20to\x20clipboard','success');}catch(_0x154e25){console['error']('Failed\x20to\x20copy\x20link',_0x154e25),Utils['showToast']('Failed\x20to\x20copy\x20link','error');}}async function emailSetupLink(){if(!state['setupLinkData'])return;const _0xd332c=document['getElementById']('email-setup-link'),_0x47cfc9=document['getElementById']('email-button-text');if(!_0xd332c||!_0x47cfc9)return;const _0x4c2a71=_0x47cfc9['textContent'];_0xd332c['disabled']=!![],_0x47cfc9['textContent']='Sending...';try{const _0x5c170a=await API['createPasswordToken'](state['setupLinkData']['userId'],state['setupLinkData']['type'],!![]);if(_0x5c170a['emailSent'])Utils['showToast']('Email\x20sent\x20to\x20'+state['setupLinkData']['email'],'success'),state['setupLinkData']['emailSent']=!![],_0xd332c['style']['display']='none';else{const _0x13c8fa=_0x5c170a['emailError']||'Failed\x20to\x20send\x20email';Utils['showToast'](_0x13c8fa,'error'),_0xd332c['disabled']=![],_0x47cfc9['textContent']=_0x4c2a71;}}catch(_0x14316d){console['error']('Failed\x20to\x20email\x20setup\x20link',_0x14316d),Utils['showToast']('Failed\x20to\x20send\x20email:\x20'+_0x14316d['message'],'error'),_0xd332c['disabled']=![],_0x47cfc9['textContent']=_0x4c2a71;}}window['resetPassword']=async function(_0x1eb3cd){const _0x23a123=state['users']['find'](_0x1c2057=>_0x1c2057['id']===_0x1eb3cd);if(!_0x23a123)return;const _0x5924fe=await ConfirmDialog['show']({'title':'Reset\x20Password','message':'Generate\x20a\x20password\x20reset\x20link\x20for\x20'+_0x23a123['name']+'\x20('+_0x23a123['username']+')?\x20This\x20will\x20create\x20a\x20link\x20that\x20they\x20can\x20use\x20to\x20set\x20a\x20new\x20password.','confirmText':'Generate\x20Link'});if(!_0x5924fe)return;try{const _0x42bc88=await API['createPasswordToken'](_0x1eb3cd,'reset',![]);state['setupLinkData']={'username':_0x23a123['username'],'email':_0x23a123['email'],'url':_0x42bc88['token']['setupUrl'],'userId':_0x23a123['id'],'type':'reset','emailSent':![]},showSetupLinkModal();}catch(_0x50f62e){console['error']('Failed\x20to\x20create\x20password\x20reset\x20link',_0x50f62e),Utils['showToast']('Failed\x20to\x20create\x20password\x20reset\x20link:\x20'+_0x50f62e['message'],'error');}};
@@ -1 +1 @@
1
- import{API,Utils}from'./app.js';const dragState={'draggedTaskId':null,'draggedElement':null,'sourceGroupId':null,'isDragging':![]},AGENT_ICONS={'claude':'<svg\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20aria-hidden=\x22true\x22><path\x20d=\x22m3.127\x2010.604\x203.135-1.76.053-.153-.053-.085H6.11l-.525-.032-1.791-.048-1.554-.065-1.505-.08-.38-.081L0\x207.832l.036-.234.32-.214.455.04\x201.009.069\x201.513.105\x201.097.064\x201.626.17h.259l.036-.105-.089-.065-.068-.064-1.566-1.062-1.695-1.121-.887-.646-.48-.327-.243-.306-.104-.67.435-.48.585.04.15.04.593.456\x201.267.981\x201.654\x201.218.242.202.097-.068.012-.049-.109-.181-.9-1.626-.96-1.655-.428-.686-.113-.411a2\x202\x200\x200\x201-.068-.484l.496-.674L4.446\x200l.662.089.279.242.411.94.666\x201.48\x201.033\x202.014.302.597.162.553.06.17h.105v-.097l.085-1.134.157-1.392.154-1.792.052-.504.25-.605.497-.327.387.186.319.456-.045.294-.19\x201.23-.37\x201.93-.243\x201.29h.142l.161-.16.654-.868\x201.097-1.372.484-.545.565-.601.363-.287h.686l.505.751-.226.775-.707.895-.585.759-.839\x201.13-.524.904.048.072.125-.012\x201.897-.403\x201.024-.186\x201.223-.21.553.258.06.263-.218.536-1.307.323-1.533.307-2.284.54-.028.02.032.04\x201.029.098.44.024h1.077l2.005.15.525.346.315.424-.053.323-.807.411-3.631-.863-.872-.218h-.12v.073l.726.71\x201.331\x201.202\x201.667\x201.55.084.383-.214.302-.226-.032-1.464-1.101-.565-.497-1.28-1.077h-.084v.113l.295.432\x201.557\x202.34.08.718-.112.234-.404.141-.444-.08-.911-1.28-.94-1.44-.759-1.291-.093.053-.448\x204.821-.21.246-.484.186-.403-.307-.214-.496.214-.98.258-1.28.21-1.016.19-1.263.112-.42-.008-.028-.092.012-.953\x201.307-1.448\x201.957-1.146\x201.227-.274.109-.477-.247.045-.44.266-.39\x201.586-2.018.956-1.25.617-.723-.004-.105h-.036l-4.212\x204.212z\x22/></svg>','codex':'<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22currentColor\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20aria-hidden=\x22true\x22><path\x20d=\x22M22.2819\x209.8211a5.9847\x205.9847\x200\x200\x200-.5157-4.9108\x206.0462\x206.0462\x200\x200\x200-6.5098-2.9A6.0651\x206.0651\x200\x200\x200\x204.9807\x204.1818a5.9847\x205.9847\x200\x200\x200-3.9977\x202.9\x206.0462\x206.0462\x200\x200\x200\x20.7427\x207.0966\x205.98\x205.98\x200\x200\x200\x20.511\x204.9107\x206.051\x206.051\x200\x200\x200\x206.5146\x202.9001A5.9847\x205.9847\x200\x200\x200\x2013.2599\x2024a6.0557\x206.0557\x200\x200\x200\x205.7718-4.2058\x205.9894\x205.9894\x200\x200\x200\x203.9977-2.9001\x206.0557\x206.0557\x200\x200\x200-.7475-7.0729zm-9.022\x2012.6081a4.4755\x204.4755\x200\x200\x201-2.8764-1.0408l.1419-.0804\x204.7783-2.7582a.7948.7948\x200\x200\x200\x20.3927-.6813v-6.7369l2.02\x201.1686a.071.071\x200\x200\x201\x20.038.052v5.5826a4.504\x204.504\x200\x200\x201-4.4945\x204.4944zm-9.6607-4.1254a4.4708\x204.4708\x200\x200\x201-.5346-3.0137l.142.0852\x204.783\x202.7582a.7712.7712\x200\x200\x200\x20.7806\x200l5.8428-3.3685v2.3324a.0804.0804\x200\x200\x201-.0332.0615L9.74\x2019.9502a4.4992\x204.4992\x200\x200\x201-6.1408-1.6464zM2.3408\x207.8956a4.485\x204.485\x200\x200\x201\x202.3655-1.9728V11.6a.7664.7664\x200\x200\x200\x20.3879.6765l5.8144\x203.3543-2.0201\x201.1685a.0757.0757\x200\x200\x201-.071\x200l-4.8303-2.7865A4.504\x204.504\x200\x200\x201\x202.3408\x207.872zm16.5963\x203.8558L13.1038\x208.364\x2015.1192\x207.2a.0757.0757\x200\x200\x201\x20.071\x200l4.8303\x202.7913a4.4944\x204.4944\x200\x200\x201-.6765\x208.1042v-5.6772a.79.79\x200\x200\x200-.407-.667zm2.0107-3.0231l-.142-.0852-4.7735-2.7818a.7759.7759\x200\x200\x200-.7854\x200L9.409\x209.2297V6.8974a.0662.0662\x200\x200\x201\x20.0284-.0615l4.8303-2.7866a4.4992\x204.4992\x200\x200\x201\x206.6802\x204.66zM8.3065\x2012.863l-2.02-1.1638a.0804.0804\x200\x200\x201-.038-.0567V6.0742a4.4992\x204.4992\x200\x200\x201\x207.3757-3.4537l-.142.0805L8.704\x205.459a.7948.7948\x200\x200\x200-.3927.6813zm1.0976-2.3654l2.602-1.4998\x202.6069\x201.4998v2.9994l-2.5974\x201.4997-2.6067-1.4997Z\x22/></svg>','gemini':'<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20aria-hidden=\x22true\x22><path\x20d=\x22M12\x202L9.09\x209.09\x202\x2012l7.09\x202.91L12\x2022l2.91-7.09L22\x2012l-7.09-2.91z\x22></path></svg>','unknown':'<svg\x20viewBox=\x220\x200\x2020\x2020\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x221.6\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20aria-hidden=\x22true\x22><circle\x20cx=\x2210\x22\x20cy=\x2210\x22\x20r=\x228\x22\x20/><path\x20d=\x22M10\x206.5c-1.1\x200-2\x20.7-2\x201.6\x200\x20.9\x201\x201.3\x201.6\x201.7.7.4\x201\x20.8\x201\x201.5V12\x22\x20/><circle\x20cx=\x2210\x22\x20cy=\x2214\x22\x20r=\x220.8\x22\x20fill=\x22currentColor\x22\x20stroke=\x22none\x22\x20/></svg>'},STATUS_ICONS={'running':'<svg\x20class=\x22status-icon\x20spinning\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222.5\x22\x20stroke-linecap=\x22round\x22><path\x20d=\x22M8\x202\x20A6\x206\x200\x200\x201\x2014\x208\x22/></svg>','completed':'<svg\x20class=\x22status-icon\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22><path\x20d=\x22M8\x200a8\x208\x200\x201\x200\x200\x2016A8\x208\x200\x200\x200\x208\x200zm3.78\x206.28-4.5\x204.5a.75.75\x200\x200\x201-1.06\x200l-2-2a.75.75\x200\x201\x201\x201.06-1.06L6.75\x209.19l3.97-3.97a.75.75\x200\x201\x201\x201.06\x201.06z\x22/></svg>','failed':'<svg\x20class=\x22status-icon\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22><path\x20d=\x22M8\x200a8\x208\x200\x201\x200\x200\x2016A8\x208\x200\x200\x200\x208\x200zm3.53\x2010.47a.75.75\x200\x201\x201-1.06\x201.06L8\x209.06l-2.47\x202.47a.75.75\x200\x200\x201-1.06-1.06L6.94\x208\x204.47\x205.53a.75.75\x200\x200\x201\x201.06-1.06L8\x206.94l2.47-2.47a.75.75\x200\x200\x201\x201.06\x201.06L9.06\x208l2.47\x202.47z\x22/></svg>','staged':'<svg\x20class=\x22status-icon\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22><path\x20d=\x22M8\x200a8\x208\x200\x201\x200\x200\x2016A8\x208\x200\x200\x200\x208\x200zm0\x2012a.75.75\x200\x201\x201\x200-1.5.75.75\x200\x200\x201\x200\x201.5zm1-3.5a1\x201\x200\x200\x201-2\x200V5a1\x201\x200\x201\x201\x202\x200v3.5z\x22/></svg>','queued':'<svg\x20class=\x22status-icon\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22><path\x20d=\x22M8\x200a8\x208\x200\x201\x200\x200\x2016A8\x208\x200\x200\x200\x208\x200zm0\x2014A6\x206\x200\x201\x201\x208\x202a6\x206\x200\x200\x201\x200\x2012zm1-6.5V4H7v4.5l3\x203\x201.5-1.5-2.5-2.5z\x22/></svg>'};export class VariantGroupingManager{constructor(_0x1ab23c={}){this['container']=null,this['onGroupChange']=_0x1ab23c['onGroupChange']||(()=>{}),this['onAddExistingTask']=_0x1ab23c['onAddExistingTask']||(()=>{}),this['tasks']=[],this['groupId']=_0x1ab23c['groupId']||null,this['isOpen']=![];}['init'](_0x15adfd,_0x3a1a37,_0x13694b){this['container']=_0x15adfd,this['tasks']=_0x3a1a37,this['groupId']=_0x13694b,this['render'](),this['attachEventListeners']();}['updateTasks'](_0x1513db){const _0x74f17d=new Map(_0x1513db['map'](_0x4dd763=>[_0x4dd763['taskId'],_0x4dd763]));this['tasks']=this['tasks']['map'](_0x2b0fe7=>{const _0x4ba834=_0x74f17d['get'](_0x2b0fe7['taskId']);return _0x4ba834?_0x4ba834:_0x2b0fe7;});for(const _0x255e21 of _0x1513db){!this['tasks']['some'](_0x1b4506=>_0x1b4506['taskId']===_0x255e21['taskId'])&&this['tasks']['push'](_0x255e21);}this['isOpen']&&this['render']();}['open'](){this['isOpen']=!![],this['render'](),this['container']?.['classList']['add']('open');}['close'](){this['isOpen']=![],this['container']?.['classList']['remove']('open');}['render'](){if(!this['container'])return;const _0x49f8c0=this['tasks']['filter'](_0x5c5ad5=>_0x5c5ad5['groupId']===this['groupId'])['sort']((_0xb5bf84,_0x236f6b)=>(_0xb5bf84['groupIndex']??0x0)-(_0x236f6b['groupIndex']??0x0)),_0x471386=this['tasks']['filter'](_0x3f4f0a=>!_0x3f4f0a['groupId']||_0x3f4f0a['groupId']!==this['groupId']);this['container']['innerHTML']='\x0a\x20\x20\x20\x20\x20\x20<div\x20class=\x22variant-grouping-manager\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-header\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<h3>Manage\x20Task\x20Group</h3>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22vgm-close-btn\x22\x20title=\x22Close\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2220\x22\x20height=\x2220\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M18\x206L6\x2018M6\x206l12\x2012\x22/>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-content\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-section\x20vgm-grouped\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-section-header\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-section-title\x22>Grouped\x20Variants</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-section-count\x22>'+_0x49f8c0['length']+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-drop-zone\x20vgm-grouped-zone\x22\x20data-zone=\x22grouped\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x49f8c0['length']===0x0?'<div\x20class=\x22vgm-empty-hint\x22>Drag\x20variants\x20here\x20to\x20add\x20to\x20group</div>':_0x49f8c0['map']((_0x5205cf,_0x1710ee)=>this['renderVariantCard'](_0x5205cf,_0x1710ee,!![]))['join'](''))+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-divider\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-divider-line\x22></div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-divider-text\x22>Drag\x20to\x20move</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-divider-line\x22></div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-section\x20vgm-ungrouped\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-section-header\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-section-title\x22>Ungrouped\x20Tasks</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-section-count\x22>'+_0x471386['length']+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-drop-zone\x20vgm-ungrouped-zone\x22\x20data-zone=\x22ungrouped\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x471386['length']===0x0?'<div\x20class=\x22vgm-empty-hint\x22>Drag\x20variants\x20here\x20to\x20remove\x20from\x20group</div>':_0x471386['map']((_0x35efdd,_0x2ded1d)=>this['renderVariantCard'](_0x35efdd,_0x2ded1d,![]))['join'](''))+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-actions\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22vgm-btn\x20vgm-btn-secondary\x20vgm-add-existing-btn\x22\x20title=\x22Add\x20an\x20existing\x20task\x20to\x20this\x20group\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M12\x205v14M5\x2012h14\x22/>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20Add\x20Existing\x20Task\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20';}['renderVariantCard'](_0x449f76,_0x24ac9b,_0x5e938c){const _0x2cec5f=_0x449f76['envVars']?.['CODER_AGENT']||_0x449f76['envVars']?.['default_agent']||'claude',_0x363aed=AGENT_ICONS[_0x2cec5f['toLowerCase']()]||AGENT_ICONS['claude'],_0x35dfcc=STATUS_ICONS[_0x449f76['status']]||'',_0x4caf56='status-'+(_0x449f76['status']||'unknown'),_0x28f8b6=_0x449f76['isWinner']?'<span\x20class=\x22vgm-badge\x20vgm-badge-winner\x22\x20title=\x22Winner\x22>★</span>':'',_0x84a7dc=_0x449f76['approvedBy']?'<span\x20class=\x22vgm-badge\x20vgm-badge-approved\x22\x20title=\x22Approved\x22><svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22\x20width=\x2212\x22\x20height=\x2212\x22><path\x20d=\x22M14\x209V5a3\x203\x200\x200\x200-3-3l-4\x209v11h11.28a2\x202\x200\x200\x200\x202-1.7l1.38-9a2\x202\x200\x200\x200-2-2.3zM7\x2022H4a2\x202\x200\x200\x201-2-2v-7a2\x202\x200\x200\x201\x202-2h3\x22></path></svg></span>':'';return'\x0a\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-card\x20'+_0x4caf56+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20draggable=\x22true\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-task-id=\x22'+_0x449f76['taskId']+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-is-grouped=\x22'+_0x5e938c+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-index=\x22'+_0x24ac9b+'\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-drag-handle\x22\x20title=\x22Drag\x20to\x20reorder\x20or\x20move\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20fill=\x22currentColor\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M8\x206h2v2H8V6zm6\x200h2v2h-2V6zM8\x2011h2v2H8v-2zm6\x200h2v2h-2v-2zm-6\x205h2v2H8v-2zm6\x200h2v2h-2v-2z\x22/>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-icon\x22\x20data-agent=\x22'+_0x2cec5f['toLowerCase']()+'\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x363aed+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-info\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-variant-name\x22>'+Utils['escapeHtml'](_0x449f76['name']||'Task\x20'+_0x449f76['taskId']['slice'](-0x6))+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-variant-meta\x22>'+Utils['escapeHtml'](_0x2cec5f)+'\x20·\x20'+(_0x449f76['status']||'unknown')+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-badges\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x28f8b6+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x84a7dc+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-status\x20'+_0x4caf56+'\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x35dfcc+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22vgm-variant-remove-btn\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20title=\x22'+(_0x5e938c?'Remove\x20from\x20group':'Add\x20to\x20group')+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-task-id=\x22'+_0x449f76['taskId']+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-action=\x22'+(_0x5e938c?'remove':'add')+'\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x5e938c?'<path\x20d=\x22M18\x206L6\x2018M6\x206l12\x2012\x22/>':'<path\x20d=\x22M12\x205v14M5\x2012h14\x22/>')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20';}['attachEventListeners'](){if(!this['container'])return;this['container']['addEventListener']('click',_0x453b9b=>{_0x453b9b['target']['closest']('.vgm-close-btn')&&(this['close'](),this['onGroupChange'](this['tasks'])),_0x453b9b['target']['closest']('.vgm-add-existing-btn')&&(this['close'](),this['onAddExistingTask'](this['groupId']));}),this['container']['addEventListener']('click',async _0x44424e=>{const _0x36b5f0=_0x44424e['target']['closest']('.vgm-variant-remove-btn');if(!_0x36b5f0)return;const _0x58d626=_0x36b5f0['dataset']['taskId'],_0x415949=_0x36b5f0['dataset']['action'];try{_0x415949==='remove'?await this['removeFromGroup'](_0x58d626):await this['addToGroup'](_0x58d626);}catch(_0xa084e){console['error']('Failed\x20to\x20update\x20group:',_0xa084e),Utils['showToast'](_0xa084e['message']||'Failed\x20to\x20update\x20group','error');}}),this['container']['addEventListener']('dragstart',_0xe4c25b=>this['handleDragStart'](_0xe4c25b)),this['container']['addEventListener']('dragend',_0x4ff367=>this['handleDragEnd'](_0x4ff367)),this['container']['addEventListener']('dragover',_0xc9bc9=>this['handleDragOver'](_0xc9bc9)),this['container']['addEventListener']('dragleave',_0x42ce10=>this['handleDragLeave'](_0x42ce10)),this['container']['addEventListener']('drop',_0x48ae46=>this['handleDrop'](_0x48ae46));}['handleDragStart'](_0xfa3946){const _0x467b05=_0xfa3946['target']['closest']('.vgm-variant-card');if(!_0x467b05)return;dragState['draggedTaskId']=_0x467b05['dataset']['taskId'],dragState['draggedElement']=_0x467b05,dragState['sourceGroupId']=_0x467b05['dataset']['isGrouped']==='true'?this['groupId']:null,dragState['isDragging']=!![],_0x467b05['classList']['add']('dragging'),_0xfa3946['dataTransfer']['effectAllowed']='move',_0xfa3946['dataTransfer']['setData']('text/plain',_0x467b05['dataset']['taskId']),this['container']['classList']['add']('is-dragging');}['handleDragEnd'](_0x2af541){const _0x3887fd=_0x2af541['target']['closest']('.vgm-variant-card');_0x3887fd&&_0x3887fd['classList']['remove']('dragging'),this['container']['querySelectorAll']('.drag-over,\x20.drop-indicator')['forEach'](_0x269cd2=>{_0x269cd2['classList']['remove']('drag-over','drop-indicator','drop-before','drop-after');}),this['container']['classList']['remove']('is-dragging'),dragState['isDragging']=![],dragState['draggedTaskId']=null,dragState['draggedElement']=null,dragState['sourceGroupId']=null;}['handleDragOver'](_0xde2036){_0xde2036['preventDefault'](),_0xde2036['dataTransfer']['dropEffect']='move';const _0x43b6a5=_0xde2036['target']['closest']('.vgm-drop-zone'),_0x59b66a=_0xde2036['target']['closest']('.vgm-variant-card');this['container']['querySelectorAll']('.vgm-drop-zone')['forEach'](_0xf3e6c0=>{_0xf3e6c0['classList']['toggle']('drag-over',_0xf3e6c0===_0x43b6a5);});if(_0x59b66a&&_0x59b66a!==dragState['draggedElement']){const _0x3ed7a9=_0x59b66a['getBoundingClientRect'](),_0x557fa7=_0x3ed7a9['top']+_0x3ed7a9['height']/0x2,_0x14c374=_0xde2036['clientY']<_0x557fa7;this['container']['querySelectorAll']('.vgm-variant-card')['forEach'](_0x36c80a=>{_0x36c80a!==_0x59b66a&&_0x36c80a['classList']['remove']('drop-before','drop-after');}),_0x59b66a['classList']['toggle']('drop-before',_0x14c374),_0x59b66a['classList']['toggle']('drop-after',!_0x14c374);}}['handleDragLeave'](_0x8bf01c){const _0x4414e7=_0x8bf01c['target']['closest']('.vgm-drop-zone');_0x4414e7&&!_0x4414e7['contains'](_0x8bf01c['relatedTarget'])&&_0x4414e7['classList']['remove']('drag-over');}async['handleDrop'](_0x53ec7a){_0x53ec7a['preventDefault']();const _0x1792b1=_0x53ec7a['target']['closest']('.vgm-drop-zone'),_0x4068d7=_0x53ec7a['target']['closest']('.vgm-variant-card'),_0x4d17d1=_0x53ec7a['dataTransfer']['getData']('text/plain');if(!_0x1792b1||!_0x4d17d1)return;this['container']['querySelectorAll']('.drag-over,\x20.drop-before,\x20.drop-after')['forEach'](_0x344d31=>{_0x344d31['classList']['remove']('drag-over','drop-before','drop-after');});const _0x415188=_0x1792b1['dataset']['zone'],_0x8c63ad=_0x415188==='grouped',_0x40fcfe=dragState['sourceGroupId']===this['groupId'];try{if(_0x8c63ad&&!_0x40fcfe){let _0x2d4ece=this['tasks']['filter'](_0x3ef4e0=>_0x3ef4e0['groupId']===this['groupId'])['length'];if(_0x4068d7){const _0x40c735=_0x4068d7['dataset']['taskId'],_0x15ae5f=this['tasks']['find'](_0x22d093=>_0x22d093['taskId']===_0x40c735);if(_0x15ae5f?.['groupIndex']!==undefined){const _0x4ee04a=_0x4068d7['getBoundingClientRect'](),_0x28c1a7=_0x53ec7a['clientY']<_0x4ee04a['top']+_0x4ee04a['height']/0x2;_0x2d4ece=_0x15ae5f['groupIndex']+(_0x28c1a7?0x0:0x1);}}await this['addToGroup'](_0x4d17d1,_0x2d4ece);}else{if(!_0x8c63ad&&_0x40fcfe)await this['removeFromGroup'](_0x4d17d1);else _0x8c63ad&&_0x40fcfe&&_0x4068d7&&await this['reorderInGroup'](_0x4d17d1,_0x4068d7,_0x53ec7a['clientY']);}}catch(_0x52a3d8){console['error']('Drop\x20operation\x20failed:',_0x52a3d8),Utils['showToast'](_0x52a3d8['message']||'Failed\x20to\x20move\x20variant','error');}}async['addToGroup'](_0x141b95,_0x3cca8e){const _0x4620d4=this['tasks']['find'](_0x7af0c8=>_0x7af0c8['taskId']===_0x141b95);if(!_0x4620d4)return;_0x3cca8e===undefined&&(_0x3cca8e=this['tasks']['filter'](_0x1ae537=>_0x1ae537['groupId']===this['groupId'])['length']),!this['groupId']&&(this['groupId']=API['generateGroupId'](),await API['createTaskGroup']({'groupId':this['groupId']})),await API['updateTaskGroup'](_0x141b95,this['groupId'],_0x3cca8e),_0x4620d4['groupId']=this['groupId'],_0x4620d4['groupIndex']=_0x3cca8e,this['render'](),this['onGroupChange'](this['tasks']);}async['removeFromGroup'](_0x12c6aa){const _0x3af1ce=this['tasks']['find'](_0x7dcfc1=>_0x7dcfc1['taskId']===_0x12c6aa);if(!_0x3af1ce)return;await API['removeTaskFromGroup'](_0x12c6aa),_0x3af1ce['groupId']=null,_0x3af1ce['groupIndex']=null,this['render'](),this['onGroupChange'](this['tasks']);}async['reorderInGroup'](_0x32e526,_0x18481c,_0x45026c){const _0x491a99=this['tasks']['find'](_0x53ac53=>_0x53ac53['taskId']===_0x32e526),_0x50ac3d=_0x18481c['dataset']['taskId'],_0xbb80cc=this['tasks']['find'](_0x456a3b=>_0x456a3b['taskId']===_0x50ac3d);if(!_0x491a99||!_0xbb80cc||_0x491a99['taskId']===_0xbb80cc['taskId'])return;const _0x2e0392=_0x18481c['getBoundingClientRect'](),_0x2e69bd=_0x45026c<_0x2e0392['top']+_0x2e0392['height']/0x2;let _0x280974=_0xbb80cc['groupIndex']??0x0;if(!_0x2e69bd)_0x280974++;if((_0x491a99['groupIndex']??0x0)<(_0xbb80cc['groupIndex']??0x0))_0x280974--;if(_0x280974===_0x491a99['groupIndex'])return;await API['updateTaskGroup'](_0x32e526,this['groupId'],_0x280974);const _0x3e3646=this['tasks']['filter'](_0x5bae02=>_0x5bae02['groupId']===this['groupId'])['sort']((_0x2ce262,_0x37bfaa)=>(_0x2ce262['groupIndex']??0x0)-(_0x37bfaa['groupIndex']??0x0)),_0x1f8520=_0x3e3646['findIndex'](_0x36ed1b=>_0x36ed1b['taskId']===_0x32e526);_0x1f8520!==-0x1&&_0x3e3646['splice'](_0x1f8520,0x1),_0x3e3646['splice'](_0x280974,0x0,_0x491a99),_0x3e3646['forEach']((_0x543100,_0x156efe)=>{_0x543100['groupIndex']=_0x156efe;}),this['render'](),this['onGroupChange'](this['tasks']);}}export function initializeTabDragDrop(_0x45fcff,_0x53553f,_0x279892,_0x3b97fc={}){if(!_0x45fcff)return;const {onReorder:_0x2cfb83,onRemove:_0x3bce3b}=_0x3b97fc,_0x576305=_0x45fcff['querySelectorAll']('.task-tab');_0x576305['forEach']((_0x3bbad3,_0x4abc1f)=>{_0x3bbad3['setAttribute']('draggable','true'),_0x3bbad3['dataset']['index']=_0x4abc1f,_0x3bbad3['addEventListener']('dragstart',_0x1291f3=>{_0x1291f3['dataTransfer']['effectAllowed']='move',_0x1291f3['dataTransfer']['setData']('text/plain',_0x3bbad3['dataset']['taskId']),_0x3bbad3['classList']['add']('dragging'),_0x45fcff['classList']['add']('is-dragging');}),_0x3bbad3['addEventListener']('dragend',()=>{_0x3bbad3['classList']['remove']('dragging'),_0x45fcff['classList']['remove']('is-dragging'),_0x45fcff['querySelectorAll']('.drop-indicator')['forEach'](_0x19f1cc=>{_0x19f1cc['classList']['remove']('drop-indicator','drop-before','drop-after');});}),_0x3bbad3['addEventListener']('dragover',_0x54fd71=>{_0x54fd71['preventDefault'](),_0x54fd71['dataTransfer']['dropEffect']='move';if(_0x3bbad3['classList']['contains']('dragging'))return;const _0x49e1c5=_0x3bbad3['getBoundingClientRect'](),_0x29aa46=_0x49e1c5['left']+_0x49e1c5['width']/0x2,_0x13438c=_0x54fd71['clientX']<_0x29aa46;_0x576305['forEach'](_0xb111ff=>_0xb111ff['classList']['remove']('drop-before','drop-after')),_0x3bbad3['classList']['add'](_0x13438c?'drop-before':'drop-after');}),_0x3bbad3['addEventListener']('dragleave',()=>{_0x3bbad3['classList']['remove']('drop-before','drop-after');}),_0x3bbad3['addEventListener']('drop',async _0x22e109=>{_0x22e109['preventDefault']();const _0x1f0f08=_0x22e109['dataTransfer']['getData']('text/plain'),_0x1be32b=_0x3bbad3['dataset']['taskId'];if(_0x1f0f08===_0x1be32b)return;const _0xfac8c2=_0x3bbad3['getBoundingClientRect'](),_0x23403e=_0x22e109['clientX']<_0xfac8c2['left']+_0xfac8c2['width']/0x2;_0x3bbad3['classList']['remove']('drop-before','drop-after'),_0x2cfb83&&await _0x2cfb83(_0x1f0f08,_0x1be32b,_0x23403e);});});const _0x2605b5=document['createElement']('div');return _0x2605b5['className']='task-tab-remove-zone',_0x2605b5['innerHTML']='\x0a\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20<path\x20d=\x22M18\x206L6\x2018M6\x206l12\x2012\x22/>\x0a\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20<span>Remove\x20from\x20group</span>\x0a\x20\x20',_0x2605b5['style']['display']='none',_0x2605b5['addEventListener']('dragover',_0x27cc95=>{_0x27cc95['preventDefault'](),_0x2605b5['classList']['add']('drag-over');}),_0x2605b5['addEventListener']('dragleave',()=>{_0x2605b5['classList']['remove']('drag-over');}),_0x2605b5['addEventListener']('drop',async _0x4b5afd=>{_0x4b5afd['preventDefault'](),_0x2605b5['classList']['remove']('drag-over');const _0x3715f2=_0x4b5afd['dataTransfer']['getData']('text/plain');_0x3bce3b&&await _0x3bce3b(_0x3715f2);}),_0x45fcff['addEventListener']('dragstart',()=>{_0x53553f['length']>0x1&&(_0x2605b5['style']['display']='flex');}),_0x45fcff['addEventListener']('dragend',()=>{_0x2605b5['style']['display']='none';}),_0x45fcff['appendChild'](_0x2605b5),{'destroy':()=>{_0x2605b5['remove']();}};}export default VariantGroupingManager;
1
+ import{API,Utils}from'./app.js';const dragState={'draggedTaskId':null,'draggedElement':null,'sourceGroupId':null,'isDragging':![]},AGENT_ICONS={'claude':'<svg\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20aria-hidden=\x22true\x22><path\x20d=\x22m3.127\x2010.604\x203.135-1.76.053-.153-.053-.085H6.11l-.525-.032-1.791-.048-1.554-.065-1.505-.08-.38-.081L0\x207.832l.036-.234.32-.214.455.04\x201.009.069\x201.513.105\x201.097.064\x201.626.17h.259l.036-.105-.089-.065-.068-.064-1.566-1.062-1.695-1.121-.887-.646-.48-.327-.243-.306-.104-.67.435-.48.585.04.15.04.593.456\x201.267.981\x201.654\x201.218.242.202.097-.068.012-.049-.109-.181-.9-1.626-.96-1.655-.428-.686-.113-.411a2\x202\x200\x200\x201-.068-.484l.496-.674L4.446\x200l.662.089.279.242.411.94.666\x201.48\x201.033\x202.014.302.597.162.553.06.17h.105v-.097l.085-1.134.157-1.392.154-1.792.052-.504.25-.605.497-.327.387.186.319.456-.045.294-.19\x201.23-.37\x201.93-.243\x201.29h.142l.161-.16.654-.868\x201.097-1.372.484-.545.565-.601.363-.287h.686l.505.751-.226.775-.707.895-.585.759-.839\x201.13-.524.904.048.072.125-.012\x201.897-.403\x201.024-.186\x201.223-.21.553.258.06.263-.218.536-1.307.323-1.533.307-2.284.54-.028.02.032.04\x201.029.098.44.024h1.077l2.005.15.525.346.315.424-.053.323-.807.411-3.631-.863-.872-.218h-.12v.073l.726.71\x201.331\x201.202\x201.667\x201.55.084.383-.214.302-.226-.032-1.464-1.101-.565-.497-1.28-1.077h-.084v.113l.295.432\x201.557\x202.34.08.718-.112.234-.404.141-.444-.08-.911-1.28-.94-1.44-.759-1.291-.093.053-.448\x204.821-.21.246-.484.186-.403-.307-.214-.496.214-.98.258-1.28.21-1.016.19-1.263.112-.42-.008-.028-.092.012-.953\x201.307-1.448\x201.957-1.146\x201.227-.274.109-.477-.247.045-.44.266-.39\x201.586-2.018.956-1.25.617-.723-.004-.105h-.036l-4.212\x204.212z\x22/></svg>','codex':'<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22currentColor\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20aria-hidden=\x22true\x22><path\x20d=\x22M22.2819\x209.8211a5.9847\x205.9847\x200\x200\x200-.5157-4.9108\x206.0462\x206.0462\x200\x200\x200-6.5098-2.9A6.0651\x206.0651\x200\x200\x200\x204.9807\x204.1818a5.9847\x205.9847\x200\x200\x200-3.9977\x202.9\x206.0462\x206.0462\x200\x200\x200\x20.7427\x207.0966\x205.98\x205.98\x200\x200\x200\x20.511\x204.9107\x206.051\x206.051\x200\x200\x200\x206.5146\x202.9001A5.9847\x205.9847\x200\x200\x200\x2013.2599\x2024a6.0557\x206.0557\x200\x200\x200\x205.7718-4.2058\x205.9894\x205.9894\x200\x200\x200\x203.9977-2.9001\x206.0557\x206.0557\x200\x200\x200-.7475-7.0729zm-9.022\x2012.6081a4.4755\x204.4755\x200\x200\x201-2.8764-1.0408l.1419-.0804\x204.7783-2.7582a.7948.7948\x200\x200\x200\x20.3927-.6813v-6.7369l2.02\x201.1686a.071.071\x200\x200\x201\x20.038.052v5.5826a4.504\x204.504\x200\x200\x201-4.4945\x204.4944zm-9.6607-4.1254a4.4708\x204.4708\x200\x200\x201-.5346-3.0137l.142.0852\x204.783\x202.7582a.7712.7712\x200\x200\x200\x20.7806\x200l5.8428-3.3685v2.3324a.0804.0804\x200\x200\x201-.0332.0615L9.74\x2019.9502a4.4992\x204.4992\x200\x200\x201-6.1408-1.6464zM2.3408\x207.8956a4.485\x204.485\x200\x200\x201\x202.3655-1.9728V11.6a.7664.7664\x200\x200\x200\x20.3879.6765l5.8144\x203.3543-2.0201\x201.1685a.0757.0757\x200\x200\x201-.071\x200l-4.8303-2.7865A4.504\x204.504\x200\x200\x201\x202.3408\x207.872zm16.5963\x203.8558L13.1038\x208.364\x2015.1192\x207.2a.0757.0757\x200\x200\x201\x20.071\x200l4.8303\x202.7913a4.4944\x204.4944\x200\x200\x201-.6765\x208.1042v-5.6772a.79.79\x200\x200\x200-.407-.667zm2.0107-3.0231l-.142-.0852-4.7735-2.7818a.7759.7759\x200\x200\x200-.7854\x200L9.409\x209.2297V6.8974a.0662.0662\x200\x200\x201\x20.0284-.0615l4.8303-2.7866a4.4992\x204.4992\x200\x200\x201\x206.6802\x204.66zM8.3065\x2012.863l-2.02-1.1638a.0804.0804\x200\x200\x201-.038-.0567V6.0742a4.4992\x204.4992\x200\x200\x201\x207.3757-3.4537l-.142.0805L8.704\x205.459a.7948.7948\x200\x200\x200-.3927.6813zm1.0976-2.3654l2.602-1.4998\x202.6069\x201.4998v2.9994l-2.5974\x201.4997-2.6067-1.4997Z\x22/></svg>','gemini':'<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20aria-hidden=\x22true\x22><path\x20d=\x22M12\x202L9.09\x209.09\x202\x2012l7.09\x202.91L12\x2022l2.91-7.09L22\x2012l-7.09-2.91z\x22></path></svg>','unknown':'<svg\x20viewBox=\x220\x200\x2020\x2020\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x221.6\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20aria-hidden=\x22true\x22><circle\x20cx=\x2210\x22\x20cy=\x2210\x22\x20r=\x228\x22\x20/><path\x20d=\x22M10\x206.5c-1.1\x200-2\x20.7-2\x201.6\x200\x20.9\x201\x201.3\x201.6\x201.7.7.4\x201\x20.8\x201\x201.5V12\x22\x20/><circle\x20cx=\x2210\x22\x20cy=\x2214\x22\x20r=\x220.8\x22\x20fill=\x22currentColor\x22\x20stroke=\x22none\x22\x20/></svg>'},STATUS_ICONS={'running':'<svg\x20class=\x22status-icon\x20spinning\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222.5\x22\x20stroke-linecap=\x22round\x22><path\x20d=\x22M8\x202\x20A6\x206\x200\x200\x201\x2014\x208\x22/></svg>','completed':'<svg\x20class=\x22status-icon\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22><path\x20d=\x22M8\x200a8\x208\x200\x201\x200\x200\x2016A8\x208\x200\x200\x200\x208\x200zm3.78\x206.28-4.5\x204.5a.75.75\x200\x200\x201-1.06\x200l-2-2a.75.75\x200\x201\x201\x201.06-1.06L6.75\x209.19l3.97-3.97a.75.75\x200\x201\x201\x201.06\x201.06z\x22/></svg>','failed':'<svg\x20class=\x22status-icon\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22><path\x20d=\x22M8\x200a8\x208\x200\x201\x200\x200\x2016A8\x208\x200\x200\x200\x208\x200zm3.53\x2010.47a.75.75\x200\x201\x201-1.06\x201.06L8\x209.06l-2.47\x202.47a.75.75\x200\x200\x201-1.06-1.06L6.94\x208\x204.47\x205.53a.75.75\x200\x200\x201\x201.06-1.06L8\x206.94l2.47-2.47a.75.75\x200\x200\x201\x201.06\x201.06L9.06\x208l2.47\x202.47z\x22/></svg>','staged':'<svg\x20class=\x22status-icon\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22><path\x20d=\x22M8\x200a8\x208\x200\x201\x200\x200\x2016A8\x208\x200\x200\x200\x208\x200zm0\x2012a.75.75\x200\x201\x201\x200-1.5.75.75\x200\x200\x201\x200\x201.5zm1-3.5a1\x201\x200\x200\x201-2\x200V5a1\x201\x200\x201\x201\x202\x200v3.5z\x22/></svg>','queued':'<svg\x20class=\x22status-icon\x22\x20viewBox=\x220\x200\x2016\x2016\x22\x20fill=\x22currentColor\x22><path\x20d=\x22M8\x200a8\x208\x200\x201\x200\x200\x2016A8\x208\x200\x200\x200\x208\x200zm0\x2014A6\x206\x200\x201\x201\x208\x202a6\x206\x200\x200\x201\x200\x2012zm1-6.5V4H7v4.5l3\x203\x201.5-1.5-2.5-2.5z\x22/></svg>'};export class VariantGroupingManager{constructor(_0x8baf74={}){this['container']=null,this['onGroupChange']=_0x8baf74['onGroupChange']||(()=>{}),this['onAddExistingTask']=_0x8baf74['onAddExistingTask']||(()=>{}),this['tasks']=[],this['groupId']=_0x8baf74['groupId']||null,this['isOpen']=![];}['init'](_0x597528,_0x15ef6d,_0x29e435){this['container']=_0x597528,this['tasks']=_0x15ef6d,this['groupId']=_0x29e435,this['render'](),this['attachEventListeners']();}['updateTasks'](_0xa22eb9){const _0x88ad9=new Map(_0xa22eb9['map'](_0x373d84=>[_0x373d84['taskId'],_0x373d84]));this['tasks']=this['tasks']['map'](_0x2518c4=>{const _0x12cb88=_0x88ad9['get'](_0x2518c4['taskId']);return _0x12cb88?_0x12cb88:_0x2518c4;});for(const _0x229bf7 of _0xa22eb9){!this['tasks']['some'](_0x35155c=>_0x35155c['taskId']===_0x229bf7['taskId'])&&this['tasks']['push'](_0x229bf7);}this['isOpen']&&this['render']();}['open'](){this['isOpen']=!![],this['render'](),this['container']?.['classList']['add']('open');}['close'](){this['isOpen']=![],this['container']?.['classList']['remove']('open');}['render'](){if(!this['container'])return;const _0x555e9b=this['tasks']['filter'](_0x5ad2f4=>_0x5ad2f4['groupId']===this['groupId'])['sort']((_0x5a3e20,_0x37d149)=>(_0x5a3e20['groupIndex']??0x0)-(_0x37d149['groupIndex']??0x0)),_0x30ef80=this['tasks']['filter'](_0x4df5b6=>!_0x4df5b6['groupId']||_0x4df5b6['groupId']!==this['groupId']);this['container']['innerHTML']='\x0a\x20\x20\x20\x20\x20\x20<div\x20class=\x22variant-grouping-manager\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-header\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<h3>Manage\x20Task\x20Group</h3>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22vgm-close-btn\x22\x20title=\x22Close\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2220\x22\x20height=\x2220\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M18\x206L6\x2018M6\x206l12\x2012\x22/>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-content\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-section\x20vgm-grouped\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-section-header\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-section-title\x22>Grouped\x20Variants</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-section-count\x22>'+_0x555e9b['length']+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-drop-zone\x20vgm-grouped-zone\x22\x20data-zone=\x22grouped\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x555e9b['length']===0x0?'<div\x20class=\x22vgm-empty-hint\x22>Drag\x20variants\x20here\x20to\x20add\x20to\x20group</div>':_0x555e9b['map']((_0x4edb7c,_0x3bc4ff)=>this['renderVariantCard'](_0x4edb7c,_0x3bc4ff,!![]))['join'](''))+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-divider\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-divider-line\x22></div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-divider-text\x22>Drag\x20to\x20move</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-divider-line\x22></div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-section\x20vgm-ungrouped\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-section-header\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-section-title\x22>Ungrouped\x20Tasks</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-section-count\x22>'+_0x30ef80['length']+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-drop-zone\x20vgm-ungrouped-zone\x22\x20data-zone=\x22ungrouped\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0x30ef80['length']===0x0?'<div\x20class=\x22vgm-empty-hint\x22>Drag\x20variants\x20here\x20to\x20remove\x20from\x20group</div>':_0x30ef80['map']((_0x29e29f,_0x39f595)=>this['renderVariantCard'](_0x29e29f,_0x39f595,![]))['join'](''))+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-actions\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22vgm-btn\x20vgm-btn-secondary\x20vgm-add-existing-btn\x22\x20title=\x22Add\x20an\x20existing\x20task\x20to\x20this\x20group\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M12\x205v14M5\x2012h14\x22/>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20Add\x20Existing\x20Task\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20';}['renderVariantCard'](_0xcebedb,_0xd71836,_0xe3f7f2){const _0x1f344c=_0xcebedb['envVars']?.['CODER_AGENT']||_0xcebedb['envVars']?.['default_agent']||'claude',_0x33371a=AGENT_ICONS[_0x1f344c['toLowerCase']()]||AGENT_ICONS['claude'],_0xbc285f=STATUS_ICONS[_0xcebedb['status']]||'',_0x3d64bd='status-'+(_0xcebedb['status']||'unknown'),_0x31c06a=_0xcebedb['isWinner']?'<span\x20class=\x22vgm-badge\x20vgm-badge-winner\x22\x20title=\x22Winner\x22>★</span>':'',_0x25e255=_0xcebedb['approvedBy']?'<span\x20class=\x22vgm-badge\x20vgm-badge-approved\x22\x20title=\x22Approved\x22><svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22\x20stroke-linecap=\x22round\x22\x20stroke-linejoin=\x22round\x22\x20width=\x2212\x22\x20height=\x2212\x22><path\x20d=\x22M14\x209V5a3\x203\x200\x200\x200-3-3l-4\x209v11h11.28a2\x202\x200\x200\x200\x202-1.7l1.38-9a2\x202\x200\x200\x200-2-2.3zM7\x2022H4a2\x202\x200\x200\x201-2-2v-7a2\x202\x200\x200\x201\x202-2h3\x22></path></svg></span>':'';return'\x0a\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-card\x20'+_0x3d64bd+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20draggable=\x22true\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-task-id=\x22'+_0xcebedb['taskId']+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-is-grouped=\x22'+_0xe3f7f2+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-index=\x22'+_0xd71836+'\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-drag-handle\x22\x20title=\x22Drag\x20to\x20reorder\x20or\x20move\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20fill=\x22currentColor\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<path\x20d=\x22M8\x206h2v2H8V6zm6\x200h2v2h-2V6zM8\x2011h2v2H8v-2zm6\x200h2v2h-2v-2zm-6\x205h2v2H8v-2zm6\x200h2v2h-2v-2z\x22/>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-icon\x22\x20data-agent=\x22'+_0x1f344c['toLowerCase']()+'\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x33371a+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-info\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-variant-name\x22>'+Utils['escapeHtml'](_0xcebedb['name']||'Task\x20'+_0xcebedb['taskId']['slice'](-0x6))+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<span\x20class=\x22vgm-variant-meta\x22>'+Utils['escapeHtml'](_0x1f344c)+'\x20·\x20'+(_0xcebedb['status']||'unknown')+'</span>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-badges\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x31c06a+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0x25e255+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<div\x20class=\x22vgm-variant-status\x20'+_0x3d64bd+'\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+_0xbc285f+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20\x20\x20\x20\x20<button\x20class=\x22vgm-variant-remove-btn\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20title=\x22'+(_0xe3f7f2?'Remove\x20from\x20group':'Add\x20to\x20group')+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-task-id=\x22'+_0xcebedb['taskId']+'\x22\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20data-action=\x22'+(_0xe3f7f2?'remove':'add')+'\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20'+(_0xe3f7f2?'<path\x20d=\x22M18\x206L6\x2018M6\x206l12\x2012\x22/>':'<path\x20d=\x22M12\x205v14M5\x2012h14\x22/>')+'\x0a\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20\x20\x20\x20\x20</button>\x0a\x20\x20\x20\x20\x20\x20</div>\x0a\x20\x20\x20\x20';}['attachEventListeners'](){if(!this['container'])return;this['container']['addEventListener']('click',_0x590c62=>{_0x590c62['target']['closest']('.vgm-close-btn')&&(this['close'](),this['onGroupChange'](this['tasks'])),_0x590c62['target']['closest']('.vgm-add-existing-btn')&&(this['close'](),this['onAddExistingTask'](this['groupId']));}),this['container']['addEventListener']('click',async _0x101e0c=>{const _0x5c1583=_0x101e0c['target']['closest']('.vgm-variant-remove-btn');if(!_0x5c1583)return;const _0x7c71d4=_0x5c1583['dataset']['taskId'],_0x12f1a1=_0x5c1583['dataset']['action'];try{_0x12f1a1==='remove'?await this['removeFromGroup'](_0x7c71d4):await this['addToGroup'](_0x7c71d4);}catch(_0x2793dd){console['error']('Failed\x20to\x20update\x20group:',_0x2793dd),Utils['showToast'](_0x2793dd['message']||'Failed\x20to\x20update\x20group','error');}}),this['container']['addEventListener']('dragstart',_0x57cc78=>this['handleDragStart'](_0x57cc78)),this['container']['addEventListener']('dragend',_0x1bb7b8=>this['handleDragEnd'](_0x1bb7b8)),this['container']['addEventListener']('dragover',_0x6e70fb=>this['handleDragOver'](_0x6e70fb)),this['container']['addEventListener']('dragleave',_0x3694fe=>this['handleDragLeave'](_0x3694fe)),this['container']['addEventListener']('drop',_0x298afd=>this['handleDrop'](_0x298afd));}['handleDragStart'](_0x40626d){const _0x1b71bb=_0x40626d['target']['closest']('.vgm-variant-card');if(!_0x1b71bb)return;dragState['draggedTaskId']=_0x1b71bb['dataset']['taskId'],dragState['draggedElement']=_0x1b71bb,dragState['sourceGroupId']=_0x1b71bb['dataset']['isGrouped']==='true'?this['groupId']:null,dragState['isDragging']=!![],_0x1b71bb['classList']['add']('dragging'),_0x40626d['dataTransfer']['effectAllowed']='move',_0x40626d['dataTransfer']['setData']('text/plain',_0x1b71bb['dataset']['taskId']),this['container']['classList']['add']('is-dragging');}['handleDragEnd'](_0x39db00){const _0x432371=_0x39db00['target']['closest']('.vgm-variant-card');_0x432371&&_0x432371['classList']['remove']('dragging'),this['container']['querySelectorAll']('.drag-over,\x20.drop-indicator')['forEach'](_0x3017e6=>{_0x3017e6['classList']['remove']('drag-over','drop-indicator','drop-before','drop-after');}),this['container']['classList']['remove']('is-dragging'),dragState['isDragging']=![],dragState['draggedTaskId']=null,dragState['draggedElement']=null,dragState['sourceGroupId']=null;}['handleDragOver'](_0x4c4bfd){_0x4c4bfd['preventDefault'](),_0x4c4bfd['dataTransfer']['dropEffect']='move';const _0x594d18=_0x4c4bfd['target']['closest']('.vgm-drop-zone'),_0x3309e4=_0x4c4bfd['target']['closest']('.vgm-variant-card');this['container']['querySelectorAll']('.vgm-drop-zone')['forEach'](_0x123f0e=>{_0x123f0e['classList']['toggle']('drag-over',_0x123f0e===_0x594d18);});if(_0x3309e4&&_0x3309e4!==dragState['draggedElement']){const _0xa1eec9=_0x3309e4['getBoundingClientRect'](),_0x15416e=_0xa1eec9['top']+_0xa1eec9['height']/0x2,_0x151e2b=_0x4c4bfd['clientY']<_0x15416e;this['container']['querySelectorAll']('.vgm-variant-card')['forEach'](_0x2aa849=>{_0x2aa849!==_0x3309e4&&_0x2aa849['classList']['remove']('drop-before','drop-after');}),_0x3309e4['classList']['toggle']('drop-before',_0x151e2b),_0x3309e4['classList']['toggle']('drop-after',!_0x151e2b);}}['handleDragLeave'](_0x21c74c){const _0x4658fa=_0x21c74c['target']['closest']('.vgm-drop-zone');_0x4658fa&&!_0x4658fa['contains'](_0x21c74c['relatedTarget'])&&_0x4658fa['classList']['remove']('drag-over');}async['handleDrop'](_0x47e25f){_0x47e25f['preventDefault']();const _0x481593=_0x47e25f['target']['closest']('.vgm-drop-zone'),_0x397392=_0x47e25f['target']['closest']('.vgm-variant-card'),_0xda5127=_0x47e25f['dataTransfer']['getData']('text/plain');if(!_0x481593||!_0xda5127)return;this['container']['querySelectorAll']('.drag-over,\x20.drop-before,\x20.drop-after')['forEach'](_0x54e08a=>{_0x54e08a['classList']['remove']('drag-over','drop-before','drop-after');});const _0x1f85c1=_0x481593['dataset']['zone'],_0x49a9bd=_0x1f85c1==='grouped',_0x270cfe=dragState['sourceGroupId']===this['groupId'];try{if(_0x49a9bd&&!_0x270cfe){let _0x3dc5b1=this['tasks']['filter'](_0x9164cb=>_0x9164cb['groupId']===this['groupId'])['length'];if(_0x397392){const _0xf04036=_0x397392['dataset']['taskId'],_0x502dd7=this['tasks']['find'](_0x126213=>_0x126213['taskId']===_0xf04036);if(_0x502dd7?.['groupIndex']!==undefined){const _0x2ed294=_0x397392['getBoundingClientRect'](),_0x5d50ca=_0x47e25f['clientY']<_0x2ed294['top']+_0x2ed294['height']/0x2;_0x3dc5b1=_0x502dd7['groupIndex']+(_0x5d50ca?0x0:0x1);}}await this['addToGroup'](_0xda5127,_0x3dc5b1);}else{if(!_0x49a9bd&&_0x270cfe)await this['removeFromGroup'](_0xda5127);else _0x49a9bd&&_0x270cfe&&_0x397392&&await this['reorderInGroup'](_0xda5127,_0x397392,_0x47e25f['clientY']);}}catch(_0x5e5104){console['error']('Drop\x20operation\x20failed:',_0x5e5104),Utils['showToast'](_0x5e5104['message']||'Failed\x20to\x20move\x20variant','error');}}async['addToGroup'](_0x54dcb7,_0x4746d9){const _0x6f2ddb=this['tasks']['find'](_0x566c0c=>_0x566c0c['taskId']===_0x54dcb7);if(!_0x6f2ddb)return;_0x4746d9===undefined&&(_0x4746d9=this['tasks']['filter'](_0x16f1f6=>_0x16f1f6['groupId']===this['groupId'])['length']),!this['groupId']&&(this['groupId']=API['generateGroupId'](),await API['createTaskGroup']({'groupId':this['groupId']})),await API['updateTaskGroup'](_0x54dcb7,this['groupId'],_0x4746d9),_0x6f2ddb['groupId']=this['groupId'],_0x6f2ddb['groupIndex']=_0x4746d9,this['render'](),this['onGroupChange'](this['tasks']);}async['removeFromGroup'](_0x396abc){const _0x36e3de=this['tasks']['find'](_0x5a7e8b=>_0x5a7e8b['taskId']===_0x396abc);if(!_0x36e3de)return;await API['removeTaskFromGroup'](_0x396abc),_0x36e3de['groupId']=null,_0x36e3de['groupIndex']=null,this['render'](),this['onGroupChange'](this['tasks']);}async['reorderInGroup'](_0x379ca9,_0x2758f,_0x269a4e){const _0x40c607=this['tasks']['find'](_0x486fd8=>_0x486fd8['taskId']===_0x379ca9),_0x546471=_0x2758f['dataset']['taskId'],_0x49136c=this['tasks']['find'](_0x38624d=>_0x38624d['taskId']===_0x546471);if(!_0x40c607||!_0x49136c||_0x40c607['taskId']===_0x49136c['taskId'])return;const _0x58878a=_0x2758f['getBoundingClientRect'](),_0x4d9077=_0x269a4e<_0x58878a['top']+_0x58878a['height']/0x2;let _0x1f60da=_0x49136c['groupIndex']??0x0;if(!_0x4d9077)_0x1f60da++;if((_0x40c607['groupIndex']??0x0)<(_0x49136c['groupIndex']??0x0))_0x1f60da--;if(_0x1f60da===_0x40c607['groupIndex'])return;await API['updateTaskGroup'](_0x379ca9,this['groupId'],_0x1f60da);const _0x4b5596=this['tasks']['filter'](_0x4b8403=>_0x4b8403['groupId']===this['groupId'])['sort']((_0x19ece2,_0x38a469)=>(_0x19ece2['groupIndex']??0x0)-(_0x38a469['groupIndex']??0x0)),_0x8198ca=_0x4b5596['findIndex'](_0x53931f=>_0x53931f['taskId']===_0x379ca9);_0x8198ca!==-0x1&&_0x4b5596['splice'](_0x8198ca,0x1),_0x4b5596['splice'](_0x1f60da,0x0,_0x40c607),_0x4b5596['forEach']((_0x4f8dcd,_0x414047)=>{_0x4f8dcd['groupIndex']=_0x414047;}),this['render'](),this['onGroupChange'](this['tasks']);}}export function initializeTabDragDrop(_0x149c0b,_0x555b5f,_0x10340a,_0x323a88={}){if(!_0x149c0b)return;const {onReorder:_0xf6d4ce,onRemove:_0x3b5248}=_0x323a88,_0x2bb81d=_0x149c0b['querySelectorAll']('.task-tab');_0x2bb81d['forEach']((_0x3087d6,_0x24e932)=>{_0x3087d6['setAttribute']('draggable','true'),_0x3087d6['dataset']['index']=_0x24e932,_0x3087d6['addEventListener']('dragstart',_0x5bd669=>{_0x5bd669['dataTransfer']['effectAllowed']='move',_0x5bd669['dataTransfer']['setData']('text/plain',_0x3087d6['dataset']['taskId']),_0x3087d6['classList']['add']('dragging'),_0x149c0b['classList']['add']('is-dragging');}),_0x3087d6['addEventListener']('dragend',()=>{_0x3087d6['classList']['remove']('dragging'),_0x149c0b['classList']['remove']('is-dragging'),_0x149c0b['querySelectorAll']('.drop-indicator')['forEach'](_0x498d9f=>{_0x498d9f['classList']['remove']('drop-indicator','drop-before','drop-after');});}),_0x3087d6['addEventListener']('dragover',_0x571342=>{_0x571342['preventDefault'](),_0x571342['dataTransfer']['dropEffect']='move';if(_0x3087d6['classList']['contains']('dragging'))return;const _0x1ba0d2=_0x3087d6['getBoundingClientRect'](),_0x135bf7=_0x1ba0d2['left']+_0x1ba0d2['width']/0x2,_0x3c8979=_0x571342['clientX']<_0x135bf7;_0x2bb81d['forEach'](_0x5181e6=>_0x5181e6['classList']['remove']('drop-before','drop-after')),_0x3087d6['classList']['add'](_0x3c8979?'drop-before':'drop-after');}),_0x3087d6['addEventListener']('dragleave',()=>{_0x3087d6['classList']['remove']('drop-before','drop-after');}),_0x3087d6['addEventListener']('drop',async _0x1e58c0=>{_0x1e58c0['preventDefault']();const _0x3c3f4c=_0x1e58c0['dataTransfer']['getData']('text/plain'),_0x57a34e=_0x3087d6['dataset']['taskId'];if(_0x3c3f4c===_0x57a34e)return;const _0x1c38a7=_0x3087d6['getBoundingClientRect'](),_0x525514=_0x1e58c0['clientX']<_0x1c38a7['left']+_0x1c38a7['width']/0x2;_0x3087d6['classList']['remove']('drop-before','drop-after'),_0xf6d4ce&&await _0xf6d4ce(_0x3c3f4c,_0x57a34e,_0x525514);});});const _0x188d40=document['createElement']('div');return _0x188d40['className']='task-tab-remove-zone',_0x188d40['innerHTML']='\x0a\x20\x20\x20\x20<svg\x20viewBox=\x220\x200\x2024\x2024\x22\x20width=\x2216\x22\x20height=\x2216\x22\x20fill=\x22none\x22\x20stroke=\x22currentColor\x22\x20stroke-width=\x222\x22>\x0a\x20\x20\x20\x20\x20\x20<path\x20d=\x22M18\x206L6\x2018M6\x206l12\x2012\x22/>\x0a\x20\x20\x20\x20</svg>\x0a\x20\x20\x20\x20<span>Remove\x20from\x20group</span>\x0a\x20\x20',_0x188d40['style']['display']='none',_0x188d40['addEventListener']('dragover',_0x10ec7d=>{_0x10ec7d['preventDefault'](),_0x188d40['classList']['add']('drag-over');}),_0x188d40['addEventListener']('dragleave',()=>{_0x188d40['classList']['remove']('drag-over');}),_0x188d40['addEventListener']('drop',async _0x15c642=>{_0x15c642['preventDefault'](),_0x188d40['classList']['remove']('drag-over');const _0x548d86=_0x15c642['dataTransfer']['getData']('text/plain');_0x3b5248&&await _0x3b5248(_0x548d86);}),_0x149c0b['addEventListener']('dragstart',()=>{_0x555b5f['length']>0x1&&(_0x188d40['style']['display']='flex');}),_0x149c0b['addEventListener']('dragend',()=>{_0x188d40['style']['display']='none';}),_0x149c0b['appendChild'](_0x188d40),{'destroy':()=>{_0x188d40['remove']();}};}export default VariantGroupingManager;