172 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			172 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| // File: frontend/app/(protected)/admin/users/_components/user-form-dialog.jsx
 | |
| 'use client';
 | |
| 
 | |
| import { useState, useEffect } from 'react';
 | |
| import api from '@/lib/api';
 | |
| import { Button } from '@/components/ui/button';
 | |
| import {
 | |
|   Dialog,
 | |
|   DialogContent,
 | |
|   DialogHeader,
 | |
|   DialogTitle,
 | |
|   DialogDescription,
 | |
|   DialogFooter,
 | |
| } from '@/components/ui/dialog';
 | |
| import { Input } from '@/components/ui/input';
 | |
| import { Label } from '@/components/ui/label';
 | |
| import { Switch } from '@/components/ui/switch';
 | |
| import { Checkbox } from "@/components/ui/checkbox"
 | |
| 
 | |
| export function UserFormDialog({ user, isOpen, setIsOpen, onSuccess }) {
 | |
|   const [formData, setFormData] = useState({});
 | |
|   const [allRoles, setAllRoles] = useState([]);
 | |
|   const [selectedRoles, setSelectedRoles] = useState(new Set());
 | |
|   const [isLoading, setIsLoading] = useState(false);
 | |
|   const [error, setError] = useState('');
 | |
| 
 | |
|   const isEditMode = !!user;
 | |
| 
 | |
|   useEffect(() => {
 | |
|     // ดึงข้อมูล Role ทั้งหมดมาเตรียมไว้
 | |
|     const fetchRoles = async () => {
 | |
|       try {
 | |
|         const res = await api.get('/rbac/roles');
 | |
|         setAllRoles(res.data);
 | |
|       } catch (err) {
 | |
|         console.error('Failed to fetch roles', err);
 | |
|       }
 | |
|     };
 | |
|     fetchRoles();
 | |
|   }, []);
 | |
| 
 | |
|   useEffect(() => {
 | |
|     // เมื่อ user prop เปลี่ยน (เปิด dialog เพื่อแก้ไข) ให้ตั้งค่าฟอร์ม
 | |
|     if (isEditMode) {
 | |
|       setFormData({
 | |
|         username: user.username,
 | |
|         email: user.email,
 | |
|         first_name: user.first_name || '',
 | |
|         last_name: user.last_name || '',
 | |
|         is_active: user.is_active,
 | |
|       });
 | |
|       setSelectedRoles(new Set(user.Roles?.map(role => role.id) || []));
 | |
|     } else {
 | |
|       // ถ้าเป็นการสร้างใหม่ ให้เคลียร์ฟอร์ม
 | |
|       setFormData({
 | |
|         username: '',
 | |
|         email: '',
 | |
|         password: '',
 | |
|         first_name: '',
 | |
|         last_name: '',
 | |
|         is_active: true,
 | |
|       });
 | |
|       setSelectedRoles(new Set());
 | |
|     }
 | |
|     setError('');
 | |
|   }, [user, isOpen]);
 | |
| 
 | |
|   const handleInputChange = (e) => {
 | |
|     const { id, value } = e.target;
 | |
|     setFormData((prev) => ({ ...prev, [id]: value }));
 | |
|   };
 | |
|   
 | |
|   const handleRoleChange = (roleId) => {
 | |
|     setSelectedRoles(prev => {
 | |
|         const newSet = new Set(prev);
 | |
|         if (newSet.has(roleId)) {
 | |
|             newSet.delete(roleId);
 | |
|         } else {
 | |
|             newSet.add(roleId);
 | |
|         }
 | |
|         return newSet;
 | |
|     });
 | |
|   };
 | |
| 
 | |
|   const handleSubmit = async (e) => {
 | |
|     e.preventDefault();
 | |
|     setIsLoading(true);
 | |
|     setError('');
 | |
| 
 | |
|     const payload = { ...formData, roles: Array.from(selectedRoles) };
 | |
|     
 | |
|     try {
 | |
|       if (isEditMode) {
 | |
|         await api.put(`/users/${user.id}`, payload);
 | |
|       } else {
 | |
|         await api.post('/users', payload);
 | |
|       }
 | |
|       onSuccess(); // บอกให้หน้าหลัก refresh ข้อมูล
 | |
|       setIsOpen(false); // ปิด Dialog
 | |
|     } catch (err) {
 | |
|       setError(err.response?.data?.message || 'An unexpected error occurred.');
 | |
|     } finally {
 | |
|       setIsLoading(false);
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return (
 | |
|     <Dialog open={isOpen} onOpenChange={setIsOpen}>
 | |
|       <DialogContent className="sm:max-w-[425px]">
 | |
|         <form onSubmit={handleSubmit}>
 | |
|           <DialogHeader>
 | |
|             <DialogTitle>{isEditMode ? 'Edit User' : 'Create New User'}</DialogTitle>
 | |
|             <DialogDescription>
 | |
|               {isEditMode ? `Editing ${user.username}` : 'Fill in the details for the new user.'}
 | |
|             </DialogDescription>
 | |
|           </DialogHeader>
 | |
|           <div className="grid gap-4 py-4">
 | |
|             <div className="grid grid-cols-4 items-center gap-4">
 | |
|               <Label htmlFor="username" className="text-right">Username</Label>
 | |
|               <Input id="username" value={formData.username || ''} onChange={handleInputChange} className="col-span-3" required disabled={isEditMode} />
 | |
|             </div>
 | |
|             <div className="grid grid-cols-4 items-center gap-4">
 | |
|               <Label htmlFor="email" className="text-right">Email</Label>
 | |
|               <Input id="email" type="email" value={formData.email || ''} onChange={handleInputChange} className="col-span-3" required />
 | |
|             </div>
 | |
|             {!isEditMode && (
 | |
|               <div className="grid grid-cols-4 items-center gap-4">
 | |
|                 <Label htmlFor="password" className="text-right">Password</Label>
 | |
|                 <Input id="password" type="password" value={formData.password || ''} onChange={handleInputChange} className="col-span-3" required />
 | |
|               </div>
 | |
|             )}
 | |
|             <div className="grid grid-cols-4 items-center gap-4">
 | |
|               <Label htmlFor="first_name" className="text-right">First Name</Label>
 | |
|               <Input id="first_name" value={formData.first_name || ''} onChange={handleInputChange} className="col-span-3" />
 | |
|             </div>
 | |
|              <div className="grid grid-cols-4 items-center gap-4">
 | |
|               <Label htmlFor="last_name" className="text-right">Last Name</Label>
 | |
|               <Input id="last_name" value={formData.last_name || ''} onChange={handleInputChange} className="col-span-3" />
 | |
|             </div>
 | |
|             <div className="grid grid-cols-4 items-center gap-4">
 | |
|               <Label htmlFor="roles" className="text-right">Roles</Label>
 | |
|               <div className="col-span-3 space-y-2">
 | |
|                  {allRoles.map(role => (
 | |
|                     <div key={role.id} className="flex items-center space-x-2">
 | |
|                         <Checkbox 
 | |
|                             id={`role-${role.id}`} 
 | |
|                             checked={selectedRoles.has(role.id)}
 | |
|                             onCheckedChange={() => handleRoleChange(role.id)}
 | |
|                         />
 | |
|                         <label htmlFor={`role-${role.id}`} className="text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70">
 | |
|                             {role.name}
 | |
|                         </label>
 | |
|                     </div>
 | |
|                  ))}
 | |
|               </div>
 | |
|             </div>
 | |
|             <div className="grid grid-cols-4 items-center gap-4">
 | |
|               <Label htmlFor="is_active" className="text-right">Active</Label>
 | |
|               <Switch id="is_active" checked={formData.is_active || false} onCheckedChange={(checked) => setFormData(prev => ({...prev, is_active: checked}))} />
 | |
|             </div>
 | |
|           </div>
 | |
|           {error && <p className="text-sm text-red-500 text-center">{error}</p>}
 | |
|           <DialogFooter>
 | |
|             <Button type="submit" disabled={isLoading}>
 | |
|               {isLoading ? 'Saving...' : 'Save Changes'}
 | |
|             </Button>
 | |
|           </DialogFooter>
 | |
|         </form>
 | |
|       </DialogContent>
 | |
|     </Dialog>
 | |
|   );
 | |
| } | 
