GroupEditPage.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. import { useState, useMemo } from 'react';
  2. import { useParams, useNavigate } from 'react-router-dom';
  3. import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
  4. import { useTranslation } from 'react-i18next';
  5. import { ArrowLeft, Save, Loader2, Search, Check, Minus, Shield, AlertTriangle } from 'lucide-react';
  6. import { api } from '../api/client';
  7. import type { Permission, PermissionCategory } from '../api/client';
  8. import { Button } from '../components/Button';
  9. import { Card } from '../components/Card';
  10. import { useToast } from '../contexts/ToastContext';
  11. export function GroupEditPage() {
  12. const { id } = useParams<{ id: string }>();
  13. const navigate = useNavigate();
  14. const queryClient = useQueryClient();
  15. const { t } = useTranslation();
  16. const { showToast } = useToast();
  17. const isEditing = Boolean(id);
  18. const [name, setName] = useState('');
  19. const [description, setDescription] = useState('');
  20. const [permissions, setPermissions] = useState<Permission[]>([]);
  21. const [search, setSearch] = useState('');
  22. const [initialized, setInitialized] = useState(false);
  23. const { data: groupData, isLoading: groupLoading } = useQuery({
  24. queryKey: ['group', id],
  25. queryFn: () => api.getGroup(Number(id)),
  26. enabled: isEditing,
  27. });
  28. const { data: permissionsData, isLoading: permissionsLoading } = useQuery({
  29. queryKey: ['permissions'],
  30. queryFn: () => api.getPermissions(),
  31. });
  32. // Initialize form from fetched group data (once)
  33. if (isEditing && groupData && !initialized) {
  34. setName(groupData.name);
  35. setDescription(groupData.description || '');
  36. setPermissions(groupData.permissions);
  37. setInitialized(true);
  38. }
  39. const createMutation = useMutation({
  40. mutationFn: (data: { name: string; description?: string; permissions: Permission[] }) =>
  41. api.createGroup(data),
  42. onSuccess: () => {
  43. queryClient.invalidateQueries({ queryKey: ['groups'] });
  44. showToast(t('groups.toast.created'));
  45. navigate('/settings?tab=users');
  46. },
  47. onError: (error: Error) => {
  48. showToast(error.message, 'error');
  49. },
  50. });
  51. const updateMutation = useMutation({
  52. mutationFn: (data: { name?: string; description?: string; permissions: Permission[] }) =>
  53. api.updateGroup(Number(id), data),
  54. onSuccess: () => {
  55. queryClient.invalidateQueries({ queryKey: ['groups'] });
  56. showToast(t('groups.toast.updated'));
  57. navigate('/settings?tab=users');
  58. },
  59. onError: (error: Error) => {
  60. showToast(error.message, 'error');
  61. },
  62. });
  63. const isSaving = createMutation.isPending || updateMutation.isPending;
  64. const handleSave = () => {
  65. if (!name.trim()) {
  66. showToast(t('groups.toast.enterGroupName'), 'error');
  67. return;
  68. }
  69. if (isEditing) {
  70. updateMutation.mutate({
  71. name: name !== groupData?.name ? name : undefined,
  72. description,
  73. permissions,
  74. });
  75. } else {
  76. createMutation.mutate({
  77. name,
  78. description: description || undefined,
  79. permissions,
  80. });
  81. }
  82. };
  83. const togglePermission = (perm: Permission) => {
  84. setPermissions((prev) =>
  85. prev.includes(perm) ? prev.filter((p) => p !== perm) : [...prev, perm]
  86. );
  87. };
  88. const toggleCategoryPermissions = (category: PermissionCategory, checked: boolean) => {
  89. const categoryPerms = category.permissions.map((p) => p.value);
  90. setPermissions((prev) => {
  91. const otherPerms = prev.filter((p) => !categoryPerms.includes(p));
  92. return checked ? [...otherPerms, ...categoryPerms] : otherPerms;
  93. });
  94. };
  95. const isCategoryFullySelected = (category: PermissionCategory) =>
  96. category.permissions.every((p) => permissions.includes(p.value));
  97. const isCategoryPartiallySelected = (category: PermissionCategory) => {
  98. const count = category.permissions.filter((p) => permissions.includes(p.value)).length;
  99. return count > 0 && count < category.permissions.length;
  100. };
  101. const selectAll = () => {
  102. if (permissionsData) {
  103. setPermissions(permissionsData.all_permissions);
  104. }
  105. };
  106. const clearAll = () => {
  107. setPermissions([]);
  108. };
  109. const searchLower = search.toLowerCase();
  110. const filteredCategories = useMemo(() => {
  111. if (!permissionsData) return [];
  112. if (!searchLower) return permissionsData.categories;
  113. return permissionsData.categories
  114. .map((cat) => ({
  115. ...cat,
  116. permissions: cat.permissions.filter((p) =>
  117. p.label.toLowerCase().includes(searchLower)
  118. ),
  119. }))
  120. .filter((cat) => cat.permissions.length > 0);
  121. }, [permissionsData, searchLower]);
  122. const totalPermissions = permissionsData?.all_permissions.length ?? 0;
  123. if (groupLoading || permissionsLoading) {
  124. return (
  125. <div className="flex items-center justify-center py-16">
  126. <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
  127. </div>
  128. );
  129. }
  130. return (
  131. <div className="space-y-6 max-w-5xl mx-auto">
  132. {/* Header */}
  133. <div className="flex items-center gap-3">
  134. <button
  135. onClick={() => navigate('/settings?tab=users')}
  136. className="p-2 rounded-lg hover:bg-bambu-dark-tertiary text-bambu-gray hover:text-white transition-colors"
  137. >
  138. <ArrowLeft className="w-5 h-5" />
  139. </button>
  140. <h1 className="text-xl font-bold text-white">
  141. {isEditing ? t('groups.editor.title') : t('groups.editor.createTitle')}
  142. </h1>
  143. </div>
  144. {/* System group warning */}
  145. {isEditing && groupData?.is_system && (
  146. <div className="flex items-center gap-3 px-4 py-3 rounded-lg bg-yellow-500/10 border border-yellow-500/20 text-yellow-400 text-sm">
  147. <AlertTriangle className="w-4 h-4 shrink-0" />
  148. {t('groups.form.systemGroupWarning')}
  149. </div>
  150. )}
  151. {/* Name + Description */}
  152. <div className="grid grid-cols-1 md:grid-cols-2 gap-4">
  153. <div>
  154. <label className="block text-sm font-medium text-white mb-2">{t('groups.form.groupName')}</label>
  155. <input
  156. type="text"
  157. value={name}
  158. onChange={(e) => setName(e.target.value)}
  159. disabled={isEditing && groupData?.is_system}
  160. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors disabled:opacity-50"
  161. placeholder={t('groups.form.groupNamePlaceholder')}
  162. />
  163. </div>
  164. <div>
  165. <label className="block text-sm font-medium text-white mb-2">{t('groups.form.description')}</label>
  166. <input
  167. type="text"
  168. value={description}
  169. onChange={(e) => setDescription(e.target.value)}
  170. className="w-full px-4 py-3 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors"
  171. placeholder={t('groups.form.descriptionPlaceholder')}
  172. />
  173. </div>
  174. </div>
  175. {/* Toolbar */}
  176. <div className="flex items-center justify-between flex-wrap gap-3">
  177. <div className="flex items-center gap-3">
  178. <span className="text-sm text-bambu-gray">
  179. {t('groups.editor.permissionsSelected', { count: permissions.length })} / {totalPermissions}
  180. </span>
  181. <Button size="sm" variant="ghost" onClick={selectAll}>
  182. {t('groups.editor.selectAll')}
  183. </Button>
  184. <Button size="sm" variant="ghost" onClick={clearAll}>
  185. {t('groups.editor.clearAll')}
  186. </Button>
  187. </div>
  188. <div className="relative">
  189. <Search className="w-4 h-4 absolute left-3 top-1/2 -translate-y-1/2 text-bambu-gray" />
  190. <input
  191. type="text"
  192. value={search}
  193. onChange={(e) => setSearch(e.target.value)}
  194. placeholder={t('groups.editor.search')}
  195. className="pl-9 pr-4 py-2 text-sm bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white placeholder-bambu-gray focus:outline-none focus:ring-2 focus:ring-bambu-green/50 focus:border-bambu-green transition-colors w-64"
  196. />
  197. </div>
  198. </div>
  199. {/* Permission grid */}
  200. {filteredCategories.length === 0 ? (
  201. <div className="text-center py-12 text-bambu-gray">
  202. {t('groups.editor.noResults')}
  203. </div>
  204. ) : (
  205. <div className="grid grid-cols-1 md:grid-cols-2 gap-4">
  206. {filteredCategories.map((category) => {
  207. // Use the full (unfiltered) category for selection logic
  208. const fullCategory = permissionsData!.categories.find((c) => c.name === category.name)!;
  209. const selectedCount = fullCategory.permissions.filter((p) => permissions.includes(p.value)).length;
  210. const totalCount = fullCategory.permissions.length;
  211. const fullySelected = isCategoryFullySelected(fullCategory);
  212. const partiallySelected = isCategoryPartiallySelected(fullCategory);
  213. return (
  214. <Card key={category.name}>
  215. <div className="sticky top-0 z-10 flex items-center justify-between px-4 py-3 bg-bambu-dark-secondary border-b border-bambu-dark-tertiary rounded-t-xl">
  216. <div className="flex items-center gap-3">
  217. <button
  218. type="button"
  219. onClick={() => toggleCategoryPermissions(fullCategory, !fullySelected)}
  220. className={`w-5 h-5 rounded border flex items-center justify-center transition-colors shrink-0 ${
  221. fullySelected
  222. ? 'bg-bambu-green border-bambu-green'
  223. : partiallySelected
  224. ? 'bg-bambu-green/50 border-bambu-green'
  225. : 'border-bambu-gray hover:border-white'
  226. }`}
  227. >
  228. {fullySelected && <Check className="w-3 h-3 text-white" />}
  229. {partiallySelected && !fullySelected && <Minus className="w-3 h-3 text-white" />}
  230. </button>
  231. <Shield className="w-4 h-4 text-bambu-gray shrink-0" />
  232. <span className="text-white font-medium text-sm">{category.name}</span>
  233. </div>
  234. <span className="text-xs text-bambu-gray tabular-nums">
  235. {selectedCount}/{totalCount}
  236. </span>
  237. </div>
  238. <div className="p-3 space-y-1">
  239. {category.permissions.map((perm) => (
  240. <label
  241. key={perm.value}
  242. className="flex items-center gap-3 px-2 py-1.5 rounded hover:bg-bambu-dark-secondary cursor-pointer"
  243. >
  244. <input
  245. type="checkbox"
  246. checked={permissions.includes(perm.value)}
  247. onChange={() => togglePermission(perm.value)}
  248. className="w-4 h-4 rounded border-bambu-gray text-bambu-green focus:ring-bambu-green focus:ring-offset-0 bg-bambu-dark-secondary"
  249. />
  250. <span className="text-sm text-bambu-gray">{perm.label}</span>
  251. </label>
  252. ))}
  253. </div>
  254. </Card>
  255. );
  256. })}
  257. </div>
  258. )}
  259. {/* Spacer for fixed bottom bar */}
  260. <div className="h-16" />
  261. {/* Fixed bottom bar */}
  262. <div className="fixed bottom-0 left-0 right-0 z-20 px-6 py-3 bg-bambu-dark-secondary border-t border-bambu-dark-tertiary flex items-center justify-center gap-3">
  263. <Button variant="secondary" onClick={() => navigate('/settings?tab=users')}>
  264. {t('common.cancel')}
  265. </Button>
  266. <Button onClick={handleSave} disabled={isSaving || !name.trim()}>
  267. {isSaving ? (
  268. <>
  269. <Loader2 className="w-4 h-4 animate-spin" />
  270. {t('common.saving')}
  271. </>
  272. ) : (
  273. <>
  274. <Save className="w-4 h-4" />
  275. {t('common.save')}
  276. </>
  277. )}
  278. </Button>
  279. </div>
  280. </div>
  281. );
  282. }