Add Companies page
This commit is contained in:
@@ -6,6 +6,8 @@ import RegisterPage from './pages/auth/RegisterPage';
|
|||||||
import { DashboardPage } from './pages/dashboard/DashboardPage';
|
import { DashboardPage } from './pages/dashboard/DashboardPage';
|
||||||
import { InvoicesPage } from './pages/invoices/InvoicesPage';
|
import { InvoicesPage } from './pages/invoices/InvoicesPage';
|
||||||
|
|
||||||
|
import { CompaniesPage } from './pages/companies/CompaniesPage';
|
||||||
|
|
||||||
// ── Protected Route Guard ─────────────────────────────────
|
// ── Protected Route Guard ─────────────────────────────────
|
||||||
const ProtectedRoute = ({ children }: { children: React.ReactNode }) => {
|
const ProtectedRoute = ({ children }: { children: React.ReactNode }) => {
|
||||||
const isAuthenticated = useAuthStore((state) => state.isAuthenticated);
|
const isAuthenticated = useAuthStore((state) => state.isAuthenticated);
|
||||||
@@ -25,7 +27,7 @@ export default function App() {
|
|||||||
<Route index element={<Navigate to="/dashboard" replace />} />
|
<Route index element={<Navigate to="/dashboard" replace />} />
|
||||||
<Route path="dashboard" element={<DashboardPage />} />
|
<Route path="dashboard" element={<DashboardPage />} />
|
||||||
<Route path="invoices" element={<InvoicesPage />} />
|
<Route path="invoices" element={<InvoicesPage />} />
|
||||||
<Route path="companies" element={<div className="text-3xl font-bold">إدارة الشركات</div>} />
|
<Route path="companies" element={<CompaniesPage />} />
|
||||||
<Route path="staff" element={<div className="text-3xl font-bold">إدارة الموظفين</div>} />
|
<Route path="staff" element={<div className="text-3xl font-bold">إدارة الموظفين</div>} />
|
||||||
<Route path="settings" element={<div className="text-3xl font-bold">الإعدادات</div>} />
|
<Route path="settings" element={<div className="text-3xl font-bold">الإعدادات</div>} />
|
||||||
</Route>
|
</Route>
|
||||||
|
|||||||
197
frontend/src/pages/companies/CompaniesPage.tsx
Normal file
197
frontend/src/pages/companies/CompaniesPage.tsx
Normal file
@@ -0,0 +1,197 @@
|
|||||||
|
/**
|
||||||
|
* ════════════════════════════════════════════════════════════
|
||||||
|
* مُصادَق (Musadaq) — Companies Management Page
|
||||||
|
* ════════════════════════════════════════════════════════════
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { useState, useEffect } from 'react';
|
||||||
|
import { Building2, Plus, Search, MoreVertical, ShieldCheck, Key } from 'lucide-react';
|
||||||
|
import apiClient from '../../api/client';
|
||||||
|
|
||||||
|
export const CompaniesPage = () => {
|
||||||
|
const [companies, setCompanies] = useState<any[]>([]);
|
||||||
|
const [isLoading, setIsLoading] = useState(true);
|
||||||
|
const [searchTerm, setSearchTerm] = useState('');
|
||||||
|
const [isAddModalOpen, setIsAddModalOpen] = useState(false);
|
||||||
|
|
||||||
|
// Form State
|
||||||
|
const [name, setName] = useState('');
|
||||||
|
const [tin, setTin] = useState('');
|
||||||
|
const [address, setAddress] = useState('');
|
||||||
|
|
||||||
|
const fetchCompanies = async () => {
|
||||||
|
try {
|
||||||
|
const { data } = await apiClient.get('/companies');
|
||||||
|
setCompanies(data);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to fetch companies', error);
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
fetchCompanies();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleCreateCompany = async (e: React.FormEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
try {
|
||||||
|
await apiClient.post('/companies', { name, tin, address });
|
||||||
|
setIsAddModalOpen(false);
|
||||||
|
setName('');
|
||||||
|
setTin('');
|
||||||
|
setAddress('');
|
||||||
|
fetchCompanies();
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to create company', error);
|
||||||
|
alert('حدث خطأ أثناء إضافة الشركة');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const filteredCompanies = companies.filter(c =>
|
||||||
|
c.name.includes(searchTerm) || c.tin?.includes(searchTerm)
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="space-y-8 h-full flex flex-col animate-in fade-in slide-in-from-bottom-4 duration-700">
|
||||||
|
<header className="flex items-center justify-between">
|
||||||
|
<div>
|
||||||
|
<h2 className="text-3xl font-bold text-slate-900">إدارة الشركات</h2>
|
||||||
|
<p className="text-slate-500 mt-1">أضف عملائك وشركاتك لربط فواتيرهم بنظام جو فوترة.</p>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
onClick={() => setIsAddModalOpen(true)}
|
||||||
|
className="btn-primary py-3 px-8 rounded-2xl flex items-center gap-2 shadow-xl shadow-primary-500/25 active:scale-95 transition-all"
|
||||||
|
>
|
||||||
|
<Plus className="w-5 h-5" />
|
||||||
|
إضافة شركة جديدة
|
||||||
|
</button>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
{/* ── Search Bar ──────────────────────────────── */}
|
||||||
|
<div className="flex gap-4">
|
||||||
|
<div className="flex-1 glass border-slate-200 rounded-2xl px-4 py-3 flex items-center gap-3">
|
||||||
|
<Search className="w-5 h-5 text-slate-400" />
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
placeholder="ابحث باسم الشركة أو الرقم الضريبي..."
|
||||||
|
className="bg-transparent border-none outline-none flex-1 text-slate-800 text-sm"
|
||||||
|
value={searchTerm}
|
||||||
|
onChange={(e) => setSearchTerm(e.target.value)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* ── Companies Grid ───────────────────────────────────── */}
|
||||||
|
{isLoading ? (
|
||||||
|
<div className="flex-1 flex justify-center items-center">
|
||||||
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-primary-600"></div>
|
||||||
|
</div>
|
||||||
|
) : filteredCompanies.length === 0 ? (
|
||||||
|
<div className="flex-1 flex flex-col items-center justify-center p-20 text-center bg-white rounded-3xl border border-slate-100 shadow-sm">
|
||||||
|
<div className="w-24 h-24 bg-slate-50 rounded-full flex items-center justify-center mb-6 border border-slate-100">
|
||||||
|
<Building2 className="w-10 h-10 text-slate-300" />
|
||||||
|
</div>
|
||||||
|
<h3 className="text-xl font-bold text-slate-900 mb-2">لا توجد شركات مسجلة</h3>
|
||||||
|
<p className="text-slate-500 max-w-sm mb-8">ابدأ بإضافة أول شركة لكي تتمكن من رفع فواتيرها ومعالجتها ضريبياً.</p>
|
||||||
|
<button onClick={() => setIsAddModalOpen(true)} className="btn-primary py-3 px-8 rounded-2xl flex items-center gap-2">
|
||||||
|
<Plus className="w-5 h-5" />
|
||||||
|
إضافة شركتك الأولى
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
|
||||||
|
{filteredCompanies.map((company, idx) => (
|
||||||
|
<div key={company.id} className="card-premium p-6 hover:shadow-lg transition-all border border-slate-100">
|
||||||
|
<div className="flex items-start justify-between mb-4">
|
||||||
|
<div className="w-12 h-12 rounded-xl bg-primary-50 text-primary-600 flex items-center justify-center">
|
||||||
|
<Building2 className="w-6 h-6" />
|
||||||
|
</div>
|
||||||
|
<button className="p-2 text-slate-400 hover:text-slate-600 hover:bg-slate-100 rounded-lg transition-all">
|
||||||
|
<MoreVertical className="w-5 h-5" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<h3 className="text-xl font-bold text-slate-900 mb-1">{company.name}</h3>
|
||||||
|
<p className="text-sm text-slate-500 mb-6">الرقم الضريبي: <span className="font-mono font-bold text-slate-700">{company.tin || 'غير محدد'}</span></p>
|
||||||
|
|
||||||
|
<div className="pt-4 border-t border-slate-100 flex items-center justify-between">
|
||||||
|
<div className="flex items-center gap-2 text-xs font-bold text-slate-500">
|
||||||
|
{company.jofotara_client_id ? (
|
||||||
|
<span className="flex items-center gap-1 text-emerald-600 bg-emerald-50 px-2 py-1 rounded-md">
|
||||||
|
<ShieldCheck className="w-3.5 h-3.5" /> مربوط بجو فوترة
|
||||||
|
</span>
|
||||||
|
) : (
|
||||||
|
<span className="flex items-center gap-1 text-amber-600 bg-amber-50 px-2 py-1 rounded-md">
|
||||||
|
<Key className="w-3.5 h-3.5" /> غير مربوط
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<button className="text-primary-600 text-sm font-bold hover:underline">
|
||||||
|
التفاصيل
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* ── Add Company Modal ───────────────────────────────── */}
|
||||||
|
{isAddModalOpen && (
|
||||||
|
<div className="fixed inset-0 z-50 flex items-center justify-center p-4 bg-slate-900/50 backdrop-blur-sm animate-in fade-in duration-200">
|
||||||
|
<div className="bg-white rounded-3xl p-8 w-full max-w-md shadow-2xl animate-in zoom-in-95 duration-200">
|
||||||
|
<h3 className="text-2xl font-bold text-slate-900 mb-6">إضافة شركة جديدة</h3>
|
||||||
|
<form onSubmit={handleCreateCompany} className="space-y-4">
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-bold text-slate-700 mb-1">اسم الشركة / العميل *</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
required
|
||||||
|
value={name}
|
||||||
|
onChange={e => setName(e.target.value)}
|
||||||
|
className="w-full bg-slate-50 border border-slate-200 rounded-xl px-4 py-3 outline-none focus:border-primary-500 focus:ring-2 focus:ring-primary-500/20 transition-all"
|
||||||
|
placeholder="مثال: صيدلية النجاح"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-bold text-slate-700 mb-1">الرقم الضريبي (TIN)</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={tin}
|
||||||
|
onChange={e => setTin(e.target.value)}
|
||||||
|
className="w-full bg-slate-50 border border-slate-200 rounded-xl px-4 py-3 outline-none focus:border-primary-500 focus:ring-2 focus:ring-primary-500/20 transition-all font-mono"
|
||||||
|
placeholder="مثال: 123456789"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-bold text-slate-700 mb-1">العنوان</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={address}
|
||||||
|
onChange={e => setAddress(e.target.value)}
|
||||||
|
className="w-full bg-slate-50 border border-slate-200 rounded-xl px-4 py-3 outline-none focus:border-primary-500 focus:ring-2 focus:ring-primary-500/20 transition-all"
|
||||||
|
placeholder="مثال: عمان، شارع مكة"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex gap-3 pt-4">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => setIsAddModalOpen(false)}
|
||||||
|
className="flex-1 bg-slate-100 text-slate-700 font-bold py-3 rounded-xl hover:bg-slate-200 transition-all"
|
||||||
|
>
|
||||||
|
إلغاء
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
className="flex-1 btn-primary py-3 rounded-xl"
|
||||||
|
>
|
||||||
|
حفظ الشركة
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
Reference in New Issue
Block a user