create-ishvexa-app 1.0.0

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 (484) hide show
  1. package/index.js +62 -0
  2. package/package.json +14 -0
  3. package/templates/epms-app/.env +7 -0
  4. package/templates/epms-app/README.md +113 -0
  5. package/templates/epms-app/backend-mongodb/.env +3 -0
  6. package/templates/epms-app/backend-mongodb/.env.example +3 -0
  7. package/templates/epms-app/backend-mongodb/config/db.js +29 -0
  8. package/templates/epms-app/backend-mongodb/controllers/authController.js +93 -0
  9. package/templates/epms-app/backend-mongodb/controllers/departmentController.js +24 -0
  10. package/templates/epms-app/backend-mongodb/controllers/employeeController.js +39 -0
  11. package/templates/epms-app/backend-mongodb/controllers/reportController.js +57 -0
  12. package/templates/epms-app/backend-mongodb/controllers/salaryController.js +101 -0
  13. package/templates/epms-app/backend-mongodb/middleware/auth.js +16 -0
  14. package/templates/epms-app/backend-mongodb/models/Counter.js +13 -0
  15. package/templates/epms-app/backend-mongodb/models/Department.js +11 -0
  16. package/templates/epms-app/backend-mongodb/models/Employee.js +18 -0
  17. package/templates/epms-app/backend-mongodb/models/Salary.js +19 -0
  18. package/templates/epms-app/backend-mongodb/models/User.js +9 -0
  19. package/templates/epms-app/backend-mongodb/package-lock.json +1571 -0
  20. package/templates/epms-app/backend-mongodb/package.json +22 -0
  21. package/templates/epms-app/backend-mongodb/routes/authRoutes.js +8 -0
  22. package/templates/epms-app/backend-mongodb/routes/departmentRoutes.js +6 -0
  23. package/templates/epms-app/backend-mongodb/routes/employeeRoutes.js +6 -0
  24. package/templates/epms-app/backend-mongodb/routes/reportRoutes.js +5 -0
  25. package/templates/epms-app/backend-mongodb/routes/salaryRoutes.js +8 -0
  26. package/templates/epms-app/backend-mongodb/server.js +39 -0
  27. package/templates/epms-app/backend-mysql/.env +7 -0
  28. package/templates/epms-app/backend-mysql/.env.example +7 -0
  29. package/templates/epms-app/backend-mysql/config/db.js +33 -0
  30. package/templates/epms-app/backend-mysql/controllers/authController.js +98 -0
  31. package/templates/epms-app/backend-mysql/controllers/departmentController.js +25 -0
  32. package/templates/epms-app/backend-mysql/controllers/employeeController.js +39 -0
  33. package/templates/epms-app/backend-mysql/controllers/reportController.js +41 -0
  34. package/templates/epms-app/backend-mysql/controllers/salaryController.js +93 -0
  35. package/templates/epms-app/backend-mysql/database/schema.sql +7 -0
  36. package/templates/epms-app/backend-mysql/middleware/auth.js +16 -0
  37. package/templates/epms-app/backend-mysql/package-lock.json +1486 -0
  38. package/templates/epms-app/backend-mysql/package.json +23 -0
  39. package/templates/epms-app/backend-mysql/routes/authRoutes.js +8 -0
  40. package/templates/epms-app/backend-mysql/routes/departmentRoutes.js +6 -0
  41. package/templates/epms-app/backend-mysql/routes/employeeRoutes.js +6 -0
  42. package/templates/epms-app/backend-mysql/routes/reportRoutes.js +5 -0
  43. package/templates/epms-app/backend-mysql/routes/salaryRoutes.js +8 -0
  44. package/templates/epms-app/backend-mysql/server.js +39 -0
  45. package/templates/epms-app/frontend/README.md +16 -0
  46. package/templates/epms-app/frontend/eslint.config.js +21 -0
  47. package/templates/epms-app/frontend/index.html +12 -0
  48. package/templates/epms-app/frontend/package-lock.json +3033 -0
  49. package/templates/epms-app/frontend/package.json +23 -0
  50. package/templates/epms-app/frontend/public/favicon.svg +1 -0
  51. package/templates/epms-app/frontend/public/icons.svg +24 -0
  52. package/templates/epms-app/frontend/src/App.css +184 -0
  53. package/templates/epms-app/frontend/src/App.jsx +31 -0
  54. package/templates/epms-app/frontend/src/api/authApi.js +7 -0
  55. package/templates/epms-app/frontend/src/api/client.js +11 -0
  56. package/templates/epms-app/frontend/src/api/departmentApi.js +5 -0
  57. package/templates/epms-app/frontend/src/api/employeeApi.js +4 -0
  58. package/templates/epms-app/frontend/src/api/reportApi.js +4 -0
  59. package/templates/epms-app/frontend/src/api/salaryApi.js +6 -0
  60. package/templates/epms-app/frontend/src/api/sparePartsApi.js +3 -0
  61. package/templates/epms-app/frontend/src/api/usersApi.js +4 -0
  62. package/templates/epms-app/frontend/src/assets/hero.png +0 -0
  63. package/templates/epms-app/frontend/src/assets/react.svg +1 -0
  64. package/templates/epms-app/frontend/src/assets/vite.svg +1 -0
  65. package/templates/epms-app/frontend/src/components/AppLayout.jsx +49 -0
  66. package/templates/epms-app/frontend/src/context/AuthContext.jsx +41 -0
  67. package/templates/epms-app/frontend/src/hooks/.gitkeep +0 -0
  68. package/templates/epms-app/frontend/src/index.css +2 -0
  69. package/templates/epms-app/frontend/src/main.jsx +16 -0
  70. package/templates/epms-app/frontend/src/pages/DepartmentPage.jsx +165 -0
  71. package/templates/epms-app/frontend/src/pages/DepartmentsPage.jsx +119 -0
  72. package/templates/epms-app/frontend/src/pages/EmployeePage.jsx +212 -0
  73. package/templates/epms-app/frontend/src/pages/EmployeesPage.jsx +217 -0
  74. package/templates/epms-app/frontend/src/pages/ForgotPassword.jsx +103 -0
  75. package/templates/epms-app/frontend/src/pages/LoginPage.jsx +105 -0
  76. package/templates/epms-app/frontend/src/pages/RegisterPage.jsx +84 -0
  77. package/templates/epms-app/frontend/src/pages/ReportsPage.jsx +192 -0
  78. package/templates/epms-app/frontend/src/pages/ResetPasswordPage.jsx +83 -0
  79. package/templates/epms-app/frontend/src/pages/SalariesPage.jsx +274 -0
  80. package/templates/epms-app/frontend/src/pages/SalaryPage.jsx +254 -0
  81. package/templates/epms-app/frontend/vite.config.js +8 -0
  82. package/templates/lms-app/.env +9 -0
  83. package/templates/lms-app/README.md +89 -0
  84. package/templates/lms-app/backend-mongodb/.env +5 -0
  85. package/templates/lms-app/backend-mongodb/.env.example +5 -0
  86. package/templates/lms-app/backend-mongodb/package-lock.json +1583 -0
  87. package/templates/lms-app/backend-mongodb/package.json +26 -0
  88. package/templates/lms-app/backend-mongodb/src/config/db.js +10 -0
  89. package/templates/lms-app/backend-mongodb/src/config/env.js +28 -0
  90. package/templates/lms-app/backend-mongodb/src/controllers/authController.js +86 -0
  91. package/templates/lms-app/backend-mongodb/src/controllers/bookController.js +101 -0
  92. package/templates/lms-app/backend-mongodb/src/controllers/borrowController.js +106 -0
  93. package/templates/lms-app/backend-mongodb/src/controllers/dashboardController.js +40 -0
  94. package/templates/lms-app/backend-mongodb/src/controllers/reportController.js +47 -0
  95. package/templates/lms-app/backend-mongodb/src/controllers/studentController.js +92 -0
  96. package/templates/lms-app/backend-mongodb/src/ensureSeedData.js +72 -0
  97. package/templates/lms-app/backend-mongodb/src/middleware/auth.js +29 -0
  98. package/templates/lms-app/backend-mongodb/src/models/Book.js +14 -0
  99. package/templates/lms-app/backend-mongodb/src/models/Borrow.js +16 -0
  100. package/templates/lms-app/backend-mongodb/src/models/Student.js +14 -0
  101. package/templates/lms-app/backend-mongodb/src/models/User.js +13 -0
  102. package/templates/lms-app/backend-mongodb/src/routes/authRoutes.js +12 -0
  103. package/templates/lms-app/backend-mongodb/src/routes/bookRoutes.js +13 -0
  104. package/templates/lms-app/backend-mongodb/src/routes/borrowRoutes.js +11 -0
  105. package/templates/lms-app/backend-mongodb/src/routes/dashboardRoutes.js +9 -0
  106. package/templates/lms-app/backend-mongodb/src/routes/reportRoutes.js +12 -0
  107. package/templates/lms-app/backend-mongodb/src/routes/studentRoutes.js +13 -0
  108. package/templates/lms-app/backend-mongodb/src/seed.js +16 -0
  109. package/templates/lms-app/backend-mongodb/src/server.js +66 -0
  110. package/templates/lms-app/backend-mysql/.env +9 -0
  111. package/templates/lms-app/backend-mysql/.env.example +9 -0
  112. package/templates/lms-app/backend-mysql/database/schema.sql +45 -0
  113. package/templates/lms-app/backend-mysql/package-lock.json +1462 -0
  114. package/templates/lms-app/backend-mysql/package.json +23 -0
  115. package/templates/lms-app/backend-mysql/src/config/db.js +33 -0
  116. package/templates/lms-app/backend-mysql/src/config/env.js +21 -0
  117. package/templates/lms-app/backend-mysql/src/controllers/authController.js +87 -0
  118. package/templates/lms-app/backend-mysql/src/controllers/bookController.js +106 -0
  119. package/templates/lms-app/backend-mysql/src/controllers/borrowController.js +113 -0
  120. package/templates/lms-app/backend-mysql/src/controllers/dashboardController.js +33 -0
  121. package/templates/lms-app/backend-mysql/src/controllers/reportController.js +40 -0
  122. package/templates/lms-app/backend-mysql/src/controllers/studentController.js +95 -0
  123. package/templates/lms-app/backend-mysql/src/ensureSeedData.js +54 -0
  124. package/templates/lms-app/backend-mysql/src/middleware/auth.js +28 -0
  125. package/templates/lms-app/backend-mysql/src/routes/authRoutes.js +12 -0
  126. package/templates/lms-app/backend-mysql/src/routes/bookRoutes.js +13 -0
  127. package/templates/lms-app/backend-mysql/src/routes/borrowRoutes.js +11 -0
  128. package/templates/lms-app/backend-mysql/src/routes/dashboardRoutes.js +9 -0
  129. package/templates/lms-app/backend-mysql/src/routes/reportRoutes.js +12 -0
  130. package/templates/lms-app/backend-mysql/src/routes/studentRoutes.js +13 -0
  131. package/templates/lms-app/backend-mysql/src/server.js +69 -0
  132. package/templates/lms-app/backend-mysql/src/utils/mappers.js +73 -0
  133. package/templates/lms-app/frontend/.env.example +5 -0
  134. package/templates/lms-app/frontend/index.html +13 -0
  135. package/templates/lms-app/frontend/package-lock.json +1592 -0
  136. package/templates/lms-app/frontend/package.json +23 -0
  137. package/templates/lms-app/frontend/public/favicon.svg +4 -0
  138. package/templates/lms-app/frontend/src/App.jsx +107 -0
  139. package/templates/lms-app/frontend/src/api/authApi.js +5 -0
  140. package/templates/lms-app/frontend/src/api/booksApi.js +6 -0
  141. package/templates/lms-app/frontend/src/api/borrowsApi.js +5 -0
  142. package/templates/lms-app/frontend/src/api/client.js +8 -0
  143. package/templates/lms-app/frontend/src/api/dashboardApi.js +3 -0
  144. package/templates/lms-app/frontend/src/api/reportsApi.js +6 -0
  145. package/templates/lms-app/frontend/src/api/studentsApi.js +6 -0
  146. package/templates/lms-app/frontend/src/components/AppLayout.jsx +63 -0
  147. package/templates/lms-app/frontend/src/index.css +34 -0
  148. package/templates/lms-app/frontend/src/main.jsx +13 -0
  149. package/templates/lms-app/frontend/src/pages/BooksPage.jsx +206 -0
  150. package/templates/lms-app/frontend/src/pages/BorrowPage.jsx +134 -0
  151. package/templates/lms-app/frontend/src/pages/DashboardPage.jsx +42 -0
  152. package/templates/lms-app/frontend/src/pages/ForgotPassword.jsx +112 -0
  153. package/templates/lms-app/frontend/src/pages/LoginPage.jsx +71 -0
  154. package/templates/lms-app/frontend/src/pages/ReportsPage.jsx +176 -0
  155. package/templates/lms-app/frontend/src/pages/ReturnPage.jsx +75 -0
  156. package/templates/lms-app/frontend/src/pages/SearchPage.jsx +156 -0
  157. package/templates/lms-app/frontend/src/pages/StudentsPage.jsx +204 -0
  158. package/templates/lms-app/frontend/vite.config.js +26 -0
  159. package/templates/scms-app/.env +7 -0
  160. package/templates/scms-app/README.md +80 -0
  161. package/templates/scms-app/backend-mongodb/.env +3 -0
  162. package/templates/scms-app/backend-mongodb/.env.example +3 -0
  163. package/templates/scms-app/backend-mongodb/config/db.js +29 -0
  164. package/templates/scms-app/backend-mongodb/controllers/authController.js +93 -0
  165. package/templates/scms-app/backend-mongodb/controllers/deliveryController.js +65 -0
  166. package/templates/scms-app/backend-mongodb/controllers/reportController.js +51 -0
  167. package/templates/scms-app/backend-mongodb/controllers/shipmentController.js +65 -0
  168. package/templates/scms-app/backend-mongodb/controllers/supplierController.js +27 -0
  169. package/templates/scms-app/backend-mongodb/middleware/auth.js +16 -0
  170. package/templates/scms-app/backend-mongodb/models/Delivery.js +14 -0
  171. package/templates/scms-app/backend-mongodb/models/Shipment.js +14 -0
  172. package/templates/scms-app/backend-mongodb/models/Supplier.js +14 -0
  173. package/templates/scms-app/backend-mongodb/models/User.js +9 -0
  174. package/templates/scms-app/backend-mongodb/package-lock.json +1571 -0
  175. package/templates/scms-app/backend-mongodb/package.json +22 -0
  176. package/templates/scms-app/backend-mongodb/routes/authRoutes.js +8 -0
  177. package/templates/scms-app/backend-mongodb/routes/deliveryRoutes.js +8 -0
  178. package/templates/scms-app/backend-mongodb/routes/reportRoutes.js +5 -0
  179. package/templates/scms-app/backend-mongodb/routes/shipmentRoutes.js +8 -0
  180. package/templates/scms-app/backend-mongodb/routes/supplierRoutes.js +6 -0
  181. package/templates/scms-app/backend-mongodb/server.js +39 -0
  182. package/templates/scms-app/backend-mysql/.env +7 -0
  183. package/templates/scms-app/backend-mysql/.env.example +7 -0
  184. package/templates/scms-app/backend-mysql/config/db.js +33 -0
  185. package/templates/scms-app/backend-mysql/controllers/authController.js +98 -0
  186. package/templates/scms-app/backend-mysql/controllers/deliveryController.js +62 -0
  187. package/templates/scms-app/backend-mysql/controllers/reportController.js +39 -0
  188. package/templates/scms-app/backend-mysql/controllers/shipmentController.js +62 -0
  189. package/templates/scms-app/backend-mysql/controllers/supplierController.js +28 -0
  190. package/templates/scms-app/backend-mysql/database/schema.sql +7 -0
  191. package/templates/scms-app/backend-mysql/middleware/auth.js +16 -0
  192. package/templates/scms-app/backend-mysql/package-lock.json +1486 -0
  193. package/templates/scms-app/backend-mysql/package.json +23 -0
  194. package/templates/scms-app/backend-mysql/routes/authRoutes.js +8 -0
  195. package/templates/scms-app/backend-mysql/routes/deliveryRoutes.js +8 -0
  196. package/templates/scms-app/backend-mysql/routes/reportRoutes.js +5 -0
  197. package/templates/scms-app/backend-mysql/routes/shipmentRoutes.js +8 -0
  198. package/templates/scms-app/backend-mysql/routes/supplierRoutes.js +6 -0
  199. package/templates/scms-app/backend-mysql/server.js +39 -0
  200. package/templates/scms-app/frontend/index.html +12 -0
  201. package/templates/scms-app/frontend/package-lock.json +1634 -0
  202. package/templates/scms-app/frontend/package.json +23 -0
  203. package/templates/scms-app/frontend/src/App.jsx +31 -0
  204. package/templates/scms-app/frontend/src/api/client.js +11 -0
  205. package/templates/scms-app/frontend/src/components/AppLayout.jsx +49 -0
  206. package/templates/scms-app/frontend/src/context/AuthContext.jsx +41 -0
  207. package/templates/scms-app/frontend/src/hooks/.gitkeep +0 -0
  208. package/templates/scms-app/frontend/src/index.css +2 -0
  209. package/templates/scms-app/frontend/src/main.jsx +16 -0
  210. package/templates/scms-app/frontend/src/pages/DeliveriesPage.jsx +265 -0
  211. package/templates/scms-app/frontend/src/pages/ForgotPassword.jsx +103 -0
  212. package/templates/scms-app/frontend/src/pages/LoginPage.jsx +105 -0
  213. package/templates/scms-app/frontend/src/pages/ReportsPage.jsx +192 -0
  214. package/templates/scms-app/frontend/src/pages/ShipmentsPage.jsx +259 -0
  215. package/templates/scms-app/frontend/src/pages/SuppliersPage.jsx +168 -0
  216. package/templates/scms-app/frontend/vite.config.js +8 -0
  217. package/templates/sfms-app/.env +7 -0
  218. package/templates/sfms-app/README.md +72 -0
  219. package/templates/sfms-app/backend-mongodb/.env +3 -0
  220. package/templates/sfms-app/backend-mongodb/.env.example +3 -0
  221. package/templates/sfms-app/backend-mongodb/package-lock.json +1580 -0
  222. package/templates/sfms-app/backend-mongodb/package.json +23 -0
  223. package/templates/sfms-app/backend-mongodb/src/config/database.js +7 -0
  224. package/templates/sfms-app/backend-mongodb/src/config/env.js +35 -0
  225. package/templates/sfms-app/backend-mongodb/src/middleware/authMiddleware.js +32 -0
  226. package/templates/sfms-app/backend-mongodb/src/models/Payment.js +12 -0
  227. package/templates/sfms-app/backend-mongodb/src/models/Student.js +12 -0
  228. package/templates/sfms-app/backend-mongodb/src/models/User.js +14 -0
  229. package/templates/sfms-app/backend-mongodb/src/routes/authRoutes.js +140 -0
  230. package/templates/sfms-app/backend-mongodb/src/routes/paymentRoutes.js +117 -0
  231. package/templates/sfms-app/backend-mongodb/src/routes/reportRoutes.js +59 -0
  232. package/templates/sfms-app/backend-mongodb/src/routes/studentRoutes.js +79 -0
  233. package/templates/sfms-app/backend-mongodb/src/server.js +34 -0
  234. package/templates/sfms-app/backend-mysql/.env +7 -0
  235. package/templates/sfms-app/backend-mysql/.env.example +7 -0
  236. package/templates/sfms-app/backend-mysql/database/schema.sql +29 -0
  237. package/templates/sfms-app/backend-mysql/package-lock.json +1467 -0
  238. package/templates/sfms-app/backend-mysql/package.json +24 -0
  239. package/templates/sfms-app/backend-mysql/src/config/.gitkeep +0 -0
  240. package/templates/sfms-app/backend-mysql/src/config/db.js +31 -0
  241. package/templates/sfms-app/backend-mysql/src/config/env.js +20 -0
  242. package/templates/sfms-app/backend-mysql/src/middleware/.gitkeep +0 -0
  243. package/templates/sfms-app/backend-mysql/src/middleware/authMiddleware.js +26 -0
  244. package/templates/sfms-app/backend-mysql/src/models/.gitkeep +0 -0
  245. package/templates/sfms-app/backend-mysql/src/routes/.gitkeep +0 -0
  246. package/templates/sfms-app/backend-mysql/src/routes/authRoutes.js +131 -0
  247. package/templates/sfms-app/backend-mysql/src/routes/paymentRoutes.js +92 -0
  248. package/templates/sfms-app/backend-mysql/src/routes/reportRoutes.js +41 -0
  249. package/templates/sfms-app/backend-mysql/src/routes/studentRoutes.js +75 -0
  250. package/templates/sfms-app/backend-mysql/src/server.js +39 -0
  251. package/templates/sfms-app/backend-mysql/src/utils/mappers.js +43 -0
  252. package/templates/sfms-app/frontend/.env.example +9 -0
  253. package/templates/sfms-app/frontend/index.html +19 -0
  254. package/templates/sfms-app/frontend/package-lock.json +2667 -0
  255. package/templates/sfms-app/frontend/package.json +23 -0
  256. package/templates/sfms-app/frontend/postcss.config.js +6 -0
  257. package/templates/sfms-app/frontend/public/favicon.svg +4 -0
  258. package/templates/sfms-app/frontend/src/App.jsx +38 -0
  259. package/templates/sfms-app/frontend/src/api/apiClient.js +54 -0
  260. package/templates/sfms-app/frontend/src/components/AppLayout.jsx +61 -0
  261. package/templates/sfms-app/frontend/src/context/AuthContext.jsx +87 -0
  262. package/templates/sfms-app/frontend/src/index.css +7 -0
  263. package/templates/sfms-app/frontend/src/main.jsx +16 -0
  264. package/templates/sfms-app/frontend/src/pages/DashboardPage.jsx +78 -0
  265. package/templates/sfms-app/frontend/src/pages/ForgotPassword.jsx +114 -0
  266. package/templates/sfms-app/frontend/src/pages/LoginPage.jsx +141 -0
  267. package/templates/sfms-app/frontend/src/pages/PaymentsPage.jsx +309 -0
  268. package/templates/sfms-app/frontend/src/pages/ReportsPage.jsx +123 -0
  269. package/templates/sfms-app/frontend/src/pages/StudentsPage.jsx +281 -0
  270. package/templates/sfms-app/frontend/tailwind.config.js +21 -0
  271. package/templates/sfms-app/frontend/vite.config.js +61 -0
  272. package/templates/sims-app/README.md +138 -0
  273. package/templates/sims-app/backend/.env +4 -0
  274. package/templates/sims-app/backend/.env.example +4 -0
  275. package/templates/sims-app/backend/package.json +22 -0
  276. package/templates/sims-app/backend/src/config/db.js +9 -0
  277. package/templates/sims-app/backend/src/controllers/authController.js +115 -0
  278. package/templates/sims-app/backend/src/controllers/simsReportController.js +94 -0
  279. package/templates/sims-app/backend/src/controllers/sparePartController.js +41 -0
  280. package/templates/sims-app/backend/src/controllers/stockInController.js +45 -0
  281. package/templates/sims-app/backend/src/controllers/stockOutController.js +123 -0
  282. package/templates/sims-app/backend/src/middleware/auth.js +8 -0
  283. package/templates/sims-app/backend/src/models/SparePart.js +17 -0
  284. package/templates/sims-app/backend/src/models/StockIn.js +16 -0
  285. package/templates/sims-app/backend/src/models/StockOut.js +18 -0
  286. package/templates/sims-app/backend/src/models/User.js +12 -0
  287. package/templates/sims-app/backend/src/routes/authRoutes.js +12 -0
  288. package/templates/sims-app/backend/src/routes/simsReportRoutes.js +8 -0
  289. package/templates/sims-app/backend/src/routes/sparePartRoutes.js +8 -0
  290. package/templates/sims-app/backend/src/routes/stockInRoutes.js +8 -0
  291. package/templates/sims-app/backend/src/routes/stockOutRoutes.js +10 -0
  292. package/templates/sims-app/backend/src/server.js +62 -0
  293. package/templates/sims-app/backend/src/utils/passwordPolicy.js +10 -0
  294. package/templates/sims-app/backend/src/utils/sparePartHelpers.js +5 -0
  295. package/templates/sims-app/frontend/index.html +13 -0
  296. package/templates/sims-app/frontend/package.json +31 -0
  297. package/templates/sims-app/frontend/src/App.jsx +110 -0
  298. package/templates/sims-app/frontend/src/api/authApi.js +7 -0
  299. package/templates/sims-app/frontend/src/api/client.js +8 -0
  300. package/templates/sims-app/frontend/src/api/simsReportApi.js +5 -0
  301. package/templates/sims-app/frontend/src/api/sparePartsApi.js +4 -0
  302. package/templates/sims-app/frontend/src/api/stockInApi.js +4 -0
  303. package/templates/sims-app/frontend/src/api/stockOutApi.js +6 -0
  304. package/templates/sims-app/frontend/src/api/usersApi.js +3 -0
  305. package/templates/sims-app/frontend/src/components/AppLayout.jsx +47 -0
  306. package/templates/sims-app/frontend/src/index.css +7 -0
  307. package/templates/sims-app/frontend/src/main.jsx +13 -0
  308. package/templates/sims-app/frontend/src/pages/ForgotPassword.jsx +111 -0
  309. package/templates/sims-app/frontend/src/pages/LoginPage.jsx +71 -0
  310. package/templates/sims-app/frontend/src/pages/RegisterPage.jsx +99 -0
  311. package/templates/sims-app/frontend/src/pages/ReportsPage.jsx +120 -0
  312. package/templates/sims-app/frontend/src/pages/SparePartPage.jsx +148 -0
  313. package/templates/sims-app/frontend/src/pages/StockInPage.jsx +122 -0
  314. package/templates/sims-app/frontend/src/pages/StockOutPage.jsx +252 -0
  315. package/templates/sims-app/frontend/src/utils/passwordPolicy.js +8 -0
  316. package/templates/sims-app/frontend/vite.config.js +8 -0
  317. package/templates/smartshop-app/README.md +61 -0
  318. package/templates/smartshop-app/backend/.env +7 -0
  319. package/templates/smartshop-app/backend/.env.example +7 -0
  320. package/templates/smartshop-app/backend/database/schema.sql +46 -0
  321. package/templates/smartshop-app/backend/package-lock.json +1487 -0
  322. package/templates/smartshop-app/backend/package.json +26 -0
  323. package/templates/smartshop-app/backend/src/app.js +57 -0
  324. package/templates/smartshop-app/backend/src/config/db.js +52 -0
  325. package/templates/smartshop-app/backend/src/controllers/authController.js +98 -0
  326. package/templates/smartshop-app/backend/src/controllers/customerController.js +26 -0
  327. package/templates/smartshop-app/backend/src/controllers/productController.js +21 -0
  328. package/templates/smartshop-app/backend/src/controllers/reportController.js +29 -0
  329. package/templates/smartshop-app/backend/src/controllers/saleController.js +11 -0
  330. package/templates/smartshop-app/backend/src/middleware/authMiddleware.js +22 -0
  331. package/templates/smartshop-app/backend/src/models/authModel.js +24 -0
  332. package/templates/smartshop-app/backend/src/models/customerModel.js +43 -0
  333. package/templates/smartshop-app/backend/src/models/productModel.js +37 -0
  334. package/templates/smartshop-app/backend/src/models/reportModel.js +56 -0
  335. package/templates/smartshop-app/backend/src/models/saleModel.js +54 -0
  336. package/templates/smartshop-app/backend/src/routes/authRoutes.js +10 -0
  337. package/templates/smartshop-app/backend/src/routes/customerRoutes.js +16 -0
  338. package/templates/smartshop-app/backend/src/routes/productRoutes.js +16 -0
  339. package/templates/smartshop-app/backend/src/routes/reportRoutes.js +18 -0
  340. package/templates/smartshop-app/backend/src/routes/saleRoutes.js +9 -0
  341. package/templates/smartshop-app/backend/src/server.js +19 -0
  342. package/templates/smartshop-app/frontend/README.md +18 -0
  343. package/templates/smartshop-app/frontend/eslint.config.js +21 -0
  344. package/templates/smartshop-app/frontend/index.html +13 -0
  345. package/templates/smartshop-app/frontend/package-lock.json +3415 -0
  346. package/templates/smartshop-app/frontend/package.json +34 -0
  347. package/templates/smartshop-app/frontend/public/favicon.svg +1 -0
  348. package/templates/smartshop-app/frontend/public/icons.svg +24 -0
  349. package/templates/smartshop-app/frontend/src/App.css +184 -0
  350. package/templates/smartshop-app/frontend/src/App.jsx +41 -0
  351. package/templates/smartshop-app/frontend/src/assets/hero.png +0 -0
  352. package/templates/smartshop-app/frontend/src/assets/react.svg +1 -0
  353. package/templates/smartshop-app/frontend/src/assets/vite.svg +1 -0
  354. package/templates/smartshop-app/frontend/src/components/AppLayout.jsx +71 -0
  355. package/templates/smartshop-app/frontend/src/components/FormCard.jsx +12 -0
  356. package/templates/smartshop-app/frontend/src/components/StatCard.jsx +10 -0
  357. package/templates/smartshop-app/frontend/src/index.css +28 -0
  358. package/templates/smartshop-app/frontend/src/main.jsx +13 -0
  359. package/templates/smartshop-app/frontend/src/pages/CustomersPage.jsx +175 -0
  360. package/templates/smartshop-app/frontend/src/pages/DashboardPage.jsx +30 -0
  361. package/templates/smartshop-app/frontend/src/pages/ForgotPassword.jsx +102 -0
  362. package/templates/smartshop-app/frontend/src/pages/LoginPage.jsx +142 -0
  363. package/templates/smartshop-app/frontend/src/pages/ProductsPage.jsx +165 -0
  364. package/templates/smartshop-app/frontend/src/pages/ReportsPage.jsx +204 -0
  365. package/templates/smartshop-app/frontend/src/pages/SalesPage.jsx +153 -0
  366. package/templates/smartshop-app/frontend/src/services/api.js +15 -0
  367. package/templates/smartshop-app/frontend/vite.config.js +13 -0
  368. package/templates/srms-app/.env +7 -0
  369. package/templates/srms-app/README.md +82 -0
  370. package/templates/srms-app/backend-mongodb/.env +3 -0
  371. package/templates/srms-app/backend-mongodb/.env.example +3 -0
  372. package/templates/srms-app/backend-mongodb/config/db.js +29 -0
  373. package/templates/srms-app/backend-mongodb/controllers/authController.js +93 -0
  374. package/templates/srms-app/backend-mongodb/controllers/customerController.js +27 -0
  375. package/templates/srms-app/backend-mongodb/controllers/productController.js +26 -0
  376. package/templates/srms-app/backend-mongodb/controllers/reportController.js +44 -0
  377. package/templates/srms-app/backend-mongodb/controllers/saleController.js +72 -0
  378. package/templates/srms-app/backend-mongodb/middleware/auth.js +16 -0
  379. package/templates/srms-app/backend-mongodb/models/Customer.js +14 -0
  380. package/templates/srms-app/backend-mongodb/models/Product.js +13 -0
  381. package/templates/srms-app/backend-mongodb/models/Sale.js +15 -0
  382. package/templates/srms-app/backend-mongodb/models/User.js +9 -0
  383. package/templates/srms-app/backend-mongodb/package-lock.json +1571 -0
  384. package/templates/srms-app/backend-mongodb/package.json +22 -0
  385. package/templates/srms-app/backend-mongodb/routes/authRoutes.js +8 -0
  386. package/templates/srms-app/backend-mongodb/routes/customerRoutes.js +6 -0
  387. package/templates/srms-app/backend-mongodb/routes/productRoutes.js +6 -0
  388. package/templates/srms-app/backend-mongodb/routes/reportRoutes.js +5 -0
  389. package/templates/srms-app/backend-mongodb/routes/saleRoutes.js +8 -0
  390. package/templates/srms-app/backend-mongodb/server.js +39 -0
  391. package/templates/srms-app/backend-mysql/.env +7 -0
  392. package/templates/srms-app/backend-mysql/.env.example +7 -0
  393. package/templates/srms-app/backend-mysql/config/db.js +33 -0
  394. package/templates/srms-app/backend-mysql/controllers/authController.js +98 -0
  395. package/templates/srms-app/backend-mysql/controllers/customerController.js +28 -0
  396. package/templates/srms-app/backend-mysql/controllers/productController.js +27 -0
  397. package/templates/srms-app/backend-mysql/controllers/reportController.js +29 -0
  398. package/templates/srms-app/backend-mysql/controllers/saleController.js +68 -0
  399. package/templates/srms-app/backend-mysql/database/schema.sql +7 -0
  400. package/templates/srms-app/backend-mysql/middleware/auth.js +16 -0
  401. package/templates/srms-app/backend-mysql/package-lock.json +1486 -0
  402. package/templates/srms-app/backend-mysql/package.json +23 -0
  403. package/templates/srms-app/backend-mysql/routes/authRoutes.js +8 -0
  404. package/templates/srms-app/backend-mysql/routes/customerRoutes.js +6 -0
  405. package/templates/srms-app/backend-mysql/routes/productRoutes.js +6 -0
  406. package/templates/srms-app/backend-mysql/routes/reportRoutes.js +5 -0
  407. package/templates/srms-app/backend-mysql/routes/saleRoutes.js +8 -0
  408. package/templates/srms-app/backend-mysql/server.js +39 -0
  409. package/templates/srms-app/frontend/index.html +12 -0
  410. package/templates/srms-app/frontend/package-lock.json +1634 -0
  411. package/templates/srms-app/frontend/package.json +23 -0
  412. package/templates/srms-app/frontend/src/App.jsx +31 -0
  413. package/templates/srms-app/frontend/src/api/client.js +11 -0
  414. package/templates/srms-app/frontend/src/components/AppLayout.jsx +40 -0
  415. package/templates/srms-app/frontend/src/context/AuthContext.jsx +41 -0
  416. package/templates/srms-app/frontend/src/hooks/.gitkeep +0 -0
  417. package/templates/srms-app/frontend/src/index.css +2 -0
  418. package/templates/srms-app/frontend/src/main.jsx +16 -0
  419. package/templates/srms-app/frontend/src/pages/CustomersPage.jsx +160 -0
  420. package/templates/srms-app/frontend/src/pages/ForgotPassword.jsx +103 -0
  421. package/templates/srms-app/frontend/src/pages/LoginPage.jsx +105 -0
  422. package/templates/srms-app/frontend/src/pages/ProductsPage.jsx +158 -0
  423. package/templates/srms-app/frontend/src/pages/ReportsPage.jsx +192 -0
  424. package/templates/srms-app/frontend/src/pages/SalesPage.jsx +310 -0
  425. package/templates/srms-app/frontend/vite.config.js +8 -0
  426. package/templates/stockhub-sms-app/.env +7 -0
  427. package/templates/stockhub-sms-app/README.md +82 -0
  428. package/templates/stockhub-sms-app/backend-mongodb/.env +3 -0
  429. package/templates/stockhub-sms-app/backend-mongodb/.env.example +3 -0
  430. package/templates/stockhub-sms-app/backend-mongodb/config/db.js +29 -0
  431. package/templates/stockhub-sms-app/backend-mongodb/controllers/authController.js +93 -0
  432. package/templates/stockhub-sms-app/backend-mongodb/controllers/productController.js +29 -0
  433. package/templates/stockhub-sms-app/backend-mongodb/controllers/reportController.js +53 -0
  434. package/templates/stockhub-sms-app/backend-mongodb/controllers/transactionController.js +74 -0
  435. package/templates/stockhub-sms-app/backend-mongodb/controllers/warehouseController.js +25 -0
  436. package/templates/stockhub-sms-app/backend-mongodb/middleware/auth.js +16 -0
  437. package/templates/stockhub-sms-app/backend-mongodb/models/Counter.js +13 -0
  438. package/templates/stockhub-sms-app/backend-mongodb/models/Product.js +16 -0
  439. package/templates/stockhub-sms-app/backend-mongodb/models/StockTransaction.js +19 -0
  440. package/templates/stockhub-sms-app/backend-mongodb/models/User.js +9 -0
  441. package/templates/stockhub-sms-app/backend-mongodb/models/Warehouse.js +12 -0
  442. package/templates/stockhub-sms-app/backend-mongodb/package-lock.json +1571 -0
  443. package/templates/stockhub-sms-app/backend-mongodb/package.json +22 -0
  444. package/templates/stockhub-sms-app/backend-mongodb/routes/authRoutes.js +8 -0
  445. package/templates/stockhub-sms-app/backend-mongodb/routes/productRoutes.js +6 -0
  446. package/templates/stockhub-sms-app/backend-mongodb/routes/reportRoutes.js +5 -0
  447. package/templates/stockhub-sms-app/backend-mongodb/routes/transactionRoutes.js +8 -0
  448. package/templates/stockhub-sms-app/backend-mongodb/routes/warehouseRoutes.js +6 -0
  449. package/templates/stockhub-sms-app/backend-mongodb/server.js +39 -0
  450. package/templates/stockhub-sms-app/backend-mysql/.env +7 -0
  451. package/templates/stockhub-sms-app/backend-mysql/.env.example +7 -0
  452. package/templates/stockhub-sms-app/backend-mysql/config/db.js +33 -0
  453. package/templates/stockhub-sms-app/backend-mysql/controllers/authController.js +98 -0
  454. package/templates/stockhub-sms-app/backend-mysql/controllers/productController.js +30 -0
  455. package/templates/stockhub-sms-app/backend-mysql/controllers/reportController.js +53 -0
  456. package/templates/stockhub-sms-app/backend-mysql/controllers/transactionController.js +70 -0
  457. package/templates/stockhub-sms-app/backend-mysql/controllers/warehouseController.js +26 -0
  458. package/templates/stockhub-sms-app/backend-mysql/database/schema.sql +40 -0
  459. package/templates/stockhub-sms-app/backend-mysql/middleware/auth.js +16 -0
  460. package/templates/stockhub-sms-app/backend-mysql/package-lock.json +1486 -0
  461. package/templates/stockhub-sms-app/backend-mysql/package.json +23 -0
  462. package/templates/stockhub-sms-app/backend-mysql/routes/authRoutes.js +8 -0
  463. package/templates/stockhub-sms-app/backend-mysql/routes/productRoutes.js +6 -0
  464. package/templates/stockhub-sms-app/backend-mysql/routes/reportRoutes.js +5 -0
  465. package/templates/stockhub-sms-app/backend-mysql/routes/transactionRoutes.js +8 -0
  466. package/templates/stockhub-sms-app/backend-mysql/routes/warehouseRoutes.js +6 -0
  467. package/templates/stockhub-sms-app/backend-mysql/server.js +39 -0
  468. package/templates/stockhub-sms-app/frontend/index.html +12 -0
  469. package/templates/stockhub-sms-app/frontend/package-lock.json +1634 -0
  470. package/templates/stockhub-sms-app/frontend/package.json +23 -0
  471. package/templates/stockhub-sms-app/frontend/src/App.jsx +31 -0
  472. package/templates/stockhub-sms-app/frontend/src/api/client.js +11 -0
  473. package/templates/stockhub-sms-app/frontend/src/components/AppLayout.jsx +40 -0
  474. package/templates/stockhub-sms-app/frontend/src/context/AuthContext.jsx +41 -0
  475. package/templates/stockhub-sms-app/frontend/src/hooks/.gitkeep +0 -0
  476. package/templates/stockhub-sms-app/frontend/src/index.css +13 -0
  477. package/templates/stockhub-sms-app/frontend/src/main.jsx +16 -0
  478. package/templates/stockhub-sms-app/frontend/src/pages/ForgotPassword.jsx +103 -0
  479. package/templates/stockhub-sms-app/frontend/src/pages/LoginPage.jsx +105 -0
  480. package/templates/stockhub-sms-app/frontend/src/pages/ProductsPage.jsx +132 -0
  481. package/templates/stockhub-sms-app/frontend/src/pages/ReportsPage.jsx +177 -0
  482. package/templates/stockhub-sms-app/frontend/src/pages/TransactionsPage.jsx +270 -0
  483. package/templates/stockhub-sms-app/frontend/src/pages/WarehousesPage.jsx +116 -0
  484. package/templates/stockhub-sms-app/frontend/vite.config.js +8 -0
@@ -0,0 +1,175 @@
1
+ import { useEffect, useState } from "react";
2
+ import FormCard from "../components/FormCard";
3
+ import api from "../services/api";
4
+
5
+ const empty = { fullName: "", phoneNumber: "", email: "", address: "" };
6
+
7
+ const validateCustomer = (form) => {
8
+ const errors = {};
9
+ if (!form.fullName.trim()) errors.fullName = "Full name is required.";
10
+ if (!form.phoneNumber.trim()) errors.phoneNumber = "Phone number is required.";
11
+ else if (!/^[0-9+\-\s]{9,15}$/.test(form.phoneNumber.trim())) {
12
+ errors.phoneNumber = "Enter a valid phone number (9–15 digits).";
13
+ }
14
+ if (!form.email.trim()) errors.email = "Email is required.";
15
+ else if (!/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(form.email.trim())) {
16
+ errors.email = "Enter a valid email address.";
17
+ }
18
+ if (!form.address.trim()) errors.address = "Address is required.";
19
+ return errors;
20
+ };
21
+
22
+ function CustomersPage() {
23
+ const [customers, setCustomers] = useState([]);
24
+ const [form, setForm] = useState(empty);
25
+ const [search, setSearch] = useState("");
26
+ const [editId, setEditId] = useState(null);
27
+ const [fieldErrors, setFieldErrors] = useState({});
28
+ const [formMessage, setFormMessage] = useState("");
29
+
30
+ const loadCustomers = () =>
31
+ api.get("/customers", { params: { q: search } }).then((res) => setCustomers(res.data));
32
+
33
+ useEffect(() => {
34
+ loadCustomers();
35
+ }, [search]);
36
+
37
+ const setField = (key, value) => {
38
+ setForm((prev) => ({ ...prev, [key]: value }));
39
+ setFieldErrors((prev) => ({ ...prev, [key]: "" }));
40
+ };
41
+
42
+ const inputClass = (key) =>
43
+ `w-full border bg-white p-2.5 text-brand-900 ${fieldErrors[key] ? "border-red-500" : "border-brand-100"}`;
44
+
45
+ const submit = async (e) => {
46
+ e.preventDefault();
47
+ setFormMessage("");
48
+ const errors = validateCustomer(form);
49
+ if (Object.keys(errors).length) {
50
+ setFieldErrors(errors);
51
+ return;
52
+ }
53
+ try {
54
+ if (editId) await api.put(`/customers/${editId}`, form);
55
+ else await api.post("/customers", form);
56
+ setFormMessage(editId ? "Customer updated successfully." : "Customer saved successfully.");
57
+ setForm(empty);
58
+ setEditId(null);
59
+ setFieldErrors({});
60
+ loadCustomers();
61
+ } catch (error) {
62
+ setFormMessage(error.response?.data?.message || "Unable to save customer.");
63
+ }
64
+ };
65
+
66
+ return (
67
+ <div className="grid gap-6 lg:grid-cols-3">
68
+ <FormCard title={editId ? "Edit Customer" : "Add Customer"}>
69
+ <form onSubmit={submit} className="space-y-2">
70
+ <input
71
+ placeholder="Full Name"
72
+ value={form.fullName}
73
+ onChange={(e) => setField("fullName", e.target.value)}
74
+ className={inputClass("fullName")}
75
+ />
76
+ {fieldErrors.fullName && <p className="text-sm text-red-600">{fieldErrors.fullName}</p>}
77
+ <input
78
+ placeholder="Phone Number"
79
+ value={form.phoneNumber}
80
+ onChange={(e) => setField("phoneNumber", e.target.value)}
81
+ className={inputClass("phoneNumber")}
82
+ />
83
+ {fieldErrors.phoneNumber && <p className="text-sm text-red-600">{fieldErrors.phoneNumber}</p>}
84
+ <input
85
+ placeholder="Email"
86
+ value={form.email}
87
+ onChange={(e) => setField("email", e.target.value)}
88
+ className={inputClass("email")}
89
+ />
90
+ {fieldErrors.email && <p className="text-sm text-red-600">{fieldErrors.email}</p>}
91
+ <input
92
+ placeholder="Address"
93
+ value={form.address}
94
+ onChange={(e) => setField("address", e.target.value)}
95
+ className={inputClass("address")}
96
+ />
97
+ {fieldErrors.address && <p className="text-sm text-red-600">{fieldErrors.address}</p>}
98
+ {formMessage && <p className="text-sm text-brand-800">{formMessage}</p>}
99
+ <button className="w-full border border-brand-700 bg-brand-800 p-2.5 font-semibold text-white">
100
+ {editId ? "Update Customer" : "Save Customer"}
101
+ </button>
102
+ </form>
103
+ </FormCard>
104
+
105
+ <section className="border border-brand-100 bg-white p-5 shadow-sm lg:col-span-2 md:p-6">
106
+ <div className="mb-3 flex items-center justify-between gap-2">
107
+ <h2 className="border-l-4 border-accent-500 pl-3 text-lg font-semibold text-brand-900">
108
+ Customers
109
+ </h2>
110
+ <input
111
+ value={search}
112
+ onChange={(e) => setSearch(e.target.value)}
113
+ placeholder="Search customer"
114
+ className="border border-brand-100 bg-white p-2 text-sm text-brand-900"
115
+ />
116
+ </div>
117
+ <div className="overflow-auto">
118
+ <table className="w-full text-sm text-brand-900">
119
+ <thead>
120
+ <tr className="text-left">
121
+ <th>Name</th>
122
+ <th>Phone</th>
123
+ <th>Email</th>
124
+ <th>Address</th>
125
+ <th>Actions</th>
126
+ </tr>
127
+ </thead>
128
+ <tbody>
129
+ {customers.map((c) => (
130
+ <tr key={c.customer_id} className="border-t border-brand-100">
131
+ <td>{c.full_name}</td>
132
+ <td>{c.phone_number}</td>
133
+ <td>{c.email}</td>
134
+ <td>{c.address}</td>
135
+ <td className="space-x-2 py-2">
136
+ <button
137
+ onClick={() => {
138
+ setEditId(c.customer_id);
139
+ setForm({
140
+ fullName: c.full_name,
141
+ phoneNumber: c.phone_number,
142
+ email: c.email,
143
+ address: c.address,
144
+ });
145
+ setFieldErrors({});
146
+ }}
147
+ className="border border-brand-700 px-2 py-1 text-brand-900"
148
+ >
149
+ Edit
150
+ </button>
151
+ <button
152
+ onClick={async () => {
153
+ try {
154
+ await api.delete(`/customers/${c.customer_id}`);
155
+ loadCustomers();
156
+ } catch (error) {
157
+ alert(error.response?.data?.message || "Delete failed.");
158
+ }
159
+ }}
160
+ className="border border-red-500 bg-red-500 px-2 py-1 text-white"
161
+ >
162
+ Delete
163
+ </button>
164
+ </td>
165
+ </tr>
166
+ ))}
167
+ </tbody>
168
+ </table>
169
+ </div>
170
+ </section>
171
+ </div>
172
+ );
173
+ }
174
+
175
+ export default CustomersPage;
@@ -0,0 +1,30 @@
1
+ import { useEffect, useState } from "react";
2
+ import StatCard from "../components/StatCard";
3
+ import api from "../services/api";
4
+
5
+ function DashboardPage() {
6
+ const [summary, setSummary] = useState({
7
+ total_products: 0,
8
+ total_customers: 0,
9
+ total_sales: 0,
10
+ total_revenue: 0,
11
+ });
12
+
13
+ useEffect(() => {
14
+ api.get("/reports/dashboard").then((res) => setSummary(res.data));
15
+ }, []);
16
+
17
+ return (
18
+ <div>
19
+ <h2 className="mb-4 text-2xl font-bold">Dashboard Summary</h2>
20
+ <div className="grid gap-4 md:grid-cols-2 lg:grid-cols-4">
21
+ <StatCard title="Total Products" value={summary.total_products} />
22
+ <StatCard title="Total Customers" value={summary.total_customers} />
23
+ <StatCard title="Total Sales" value={summary.total_sales} />
24
+ <StatCard title="Revenue" value={`RWF ${Number(summary.total_revenue).toFixed(2)}`} />
25
+ </div>
26
+ </div>
27
+ );
28
+ }
29
+
30
+ export default DashboardPage;
@@ -0,0 +1,102 @@
1
+ import { useState } from "react";
2
+ import { Link, useNavigate } from "react-router-dom";
3
+ import api from "../services/api";
4
+
5
+ function ForgotPassword() {
6
+ const navigate = useNavigate();
7
+ const [form, setForm] = useState({ email: "", newPassword: "", confirmPassword: "" });
8
+ const [fieldErrors, setFieldErrors] = useState({});
9
+ const [error, setError] = useState("");
10
+ const [message, setMessage] = useState("");
11
+ const [loading, setLoading] = useState(false);
12
+
13
+ const validate = () => {
14
+ const errors = {};
15
+ const email = form.email.trim();
16
+ if (!email) errors.email = "Email is required.";
17
+ else if (!/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(email)) errors.email = "Enter a valid email address.";
18
+ if (!form.newPassword) errors.newPassword = "New password is required.";
19
+ else if (form.newPassword.length < 6) errors.newPassword = "Password must be at least 6 characters.";
20
+ if (!form.confirmPassword) errors.confirmPassword = "Confirm password is required.";
21
+ else if (form.newPassword && form.newPassword !== form.confirmPassword) {
22
+ errors.confirmPassword = "Passwords must match.";
23
+ }
24
+ setFieldErrors(errors);
25
+ return Object.keys(errors).length === 0;
26
+ };
27
+
28
+ const submit = async (event) => {
29
+ event.preventDefault();
30
+ setError("");
31
+ setMessage("");
32
+ if (!validate()) return;
33
+ setLoading(true);
34
+ try {
35
+ const { data } = await api.post("/auth/forgot-password", {
36
+ email: form.email.trim(),
37
+ newPassword: form.newPassword,
38
+ confirmPassword: form.confirmPassword,
39
+ });
40
+ setMessage(data.message || "Password reset successfully.");
41
+ setTimeout(() => navigate("/login", { replace: true }), 2000);
42
+ } catch (err) {
43
+ setError(err.response?.data?.message || "Request failed.");
44
+ } finally {
45
+ setLoading(false);
46
+ }
47
+ };
48
+
49
+ return (
50
+ <div className="flex min-h-screen items-center justify-center bg-brand-50 px-4 py-8">
51
+ <form
52
+ onSubmit={submit}
53
+ className="w-full max-w-md border border-brand-100 bg-white p-6 text-brand-900 shadow-sm md:p-8"
54
+ >
55
+ <h1 className="border-l-4 border-accent-500 pl-3 text-2xl font-bold">Forgot Password</h1>
56
+ <p className="mt-1 pl-3 text-sm text-gray-600">Enter your email and new password</p>
57
+ <div className="mt-4 space-y-3">
58
+ <input
59
+ className="w-full border border-brand-100 p-2.5"
60
+ type="email"
61
+ placeholder="Email"
62
+ value={form.email}
63
+ onChange={(e) => setForm({ ...form, email: e.target.value })}
64
+ />
65
+ {fieldErrors.email && <p className="text-sm text-red-600">{fieldErrors.email}</p>}
66
+ <input
67
+ className="w-full border border-brand-100 p-2.5"
68
+ type="password"
69
+ placeholder="New password"
70
+ value={form.newPassword}
71
+ onChange={(e) => setForm({ ...form, newPassword: e.target.value })}
72
+ />
73
+ {fieldErrors.newPassword && <p className="text-sm text-red-600">{fieldErrors.newPassword}</p>}
74
+ <input
75
+ className="w-full border border-brand-100 p-2.5"
76
+ type="password"
77
+ placeholder="Confirm password"
78
+ value={form.confirmPassword}
79
+ onChange={(e) => setForm({ ...form, confirmPassword: e.target.value })}
80
+ />
81
+ {fieldErrors.confirmPassword && (
82
+ <p className="text-sm text-red-600">{fieldErrors.confirmPassword}</p>
83
+ )}
84
+ </div>
85
+ {message && <p className="mt-3 text-sm text-green-700">{message}</p>}
86
+ {error && <p className="mt-3 text-sm text-red-600">{error}</p>}
87
+ <button
88
+ type="submit"
89
+ disabled={loading}
90
+ className="mt-5 w-full border border-brand-800 bg-brand-800 px-4 py-2.5 font-semibold text-white disabled:opacity-60"
91
+ >
92
+ {loading ? "Please wait…" : "Reset Password"}
93
+ </button>
94
+ <Link to="/login" className="mt-3 block w-full text-center text-sm text-brand-800 underline">
95
+ Back to login
96
+ </Link>
97
+ </form>
98
+ </div>
99
+ );
100
+ }
101
+
102
+ export default ForgotPassword;
@@ -0,0 +1,142 @@
1
+ import { useState } from "react";
2
+ import { Link, useNavigate } from "react-router-dom";
3
+ import api from "../services/api";
4
+
5
+ function LoginPage() {
6
+ const [mode, setMode] = useState("login");
7
+ const [form, setForm] = useState({ username: "", email: "", password: "" });
8
+ const [fieldErrors, setFieldErrors] = useState({});
9
+ const [error, setError] = useState("");
10
+ const [message, setMessage] = useState("");
11
+ const navigate = useNavigate();
12
+
13
+ const validate = () => {
14
+ const errors = {};
15
+ if (!form.username.trim()) errors.username = "Username is required.";
16
+ else if (form.username.trim().length < 3) {
17
+ errors.username = "Username must be at least 3 characters.";
18
+ }
19
+ if (mode === "register") {
20
+ if (!form.email.trim()) errors.email = "Email is required.";
21
+ else if (!/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(form.email.trim())) {
22
+ errors.email = "Enter a valid email address.";
23
+ }
24
+ }
25
+ if (!form.password) errors.password = "Password is required.";
26
+ else if (form.password.length < 6) {
27
+ errors.password = "Password must be at least 6 characters.";
28
+ }
29
+ setFieldErrors(errors);
30
+ return Object.keys(errors).length === 0;
31
+ };
32
+
33
+ const submit = async (event) => {
34
+ event.preventDefault();
35
+ setError("");
36
+ setMessage("");
37
+ if (!validate()) return;
38
+ try {
39
+ if (mode === "register") {
40
+ await api.post("/auth/register", {
41
+ username: form.username.trim(),
42
+ email: form.email.trim(),
43
+ password: form.password,
44
+ });
45
+ setMode("login");
46
+ setMessage("Registration successful. Please login.");
47
+ setForm({ username: form.username, email: "", password: "" });
48
+ return;
49
+ }
50
+
51
+ const { data } = await api.post("/auth/login", {
52
+ username: form.username.trim(),
53
+ password: form.password,
54
+ });
55
+ localStorage.setItem("smartshop_user", JSON.stringify(data.user));
56
+ localStorage.setItem("smartshop_token", data.token);
57
+ navigate("/");
58
+ } catch (err) {
59
+ setError(err.response?.data?.message || "Authentication failed");
60
+ }
61
+ };
62
+
63
+ return (
64
+ <div className="flex min-h-screen items-center justify-center bg-brand-50 px-4 py-8">
65
+ <form
66
+ onSubmit={submit}
67
+ className="w-full max-w-md border border-brand-100 bg-white p-6 text-brand-900 shadow-sm md:p-8"
68
+ >
69
+ <h1 className="border-l-4 border-accent-500 pl-3 text-2xl font-bold">
70
+ Sales Management System
71
+ </h1>
72
+ <p className="mt-1 pl-3 text-sm text-gray-600">
73
+ SMS — {mode === "login" ? "Sign In" : "Create Account"}
74
+ </p>
75
+ <div className="mt-4 space-y-3">
76
+ <input
77
+ className="w-full border border-brand-100 p-2.5"
78
+ placeholder="Username"
79
+ value={form.username}
80
+ onChange={(e) => setForm({ ...form, username: e.target.value })}
81
+ />
82
+ {fieldErrors.username && (
83
+ <p className="text-sm text-red-600">{fieldErrors.username}</p>
84
+ )}
85
+ {mode === "register" && (
86
+ <>
87
+ <input
88
+ className="w-full border border-brand-100 p-2.5"
89
+ type="email"
90
+ placeholder="Email"
91
+ value={form.email}
92
+ onChange={(e) => setForm({ ...form, email: e.target.value })}
93
+ />
94
+ {fieldErrors.email && (
95
+ <p className="text-sm text-red-600">{fieldErrors.email}</p>
96
+ )}
97
+ </>
98
+ )}
99
+ <input
100
+ className="w-full border border-brand-100 p-2.5"
101
+ type="password"
102
+ placeholder="Password"
103
+ value={form.password}
104
+ onChange={(e) => setForm({ ...form, password: e.target.value })}
105
+ />
106
+ {fieldErrors.password && (
107
+ <p className="text-sm text-red-600">{fieldErrors.password}</p>
108
+ )}
109
+ </div>
110
+ {message && <p className="mt-3 text-sm text-green-700">{message}</p>}
111
+ {error && <p className="mt-3 text-sm text-red-600">{error}</p>}
112
+ <button
113
+ type="submit"
114
+ className="mt-5 w-full border border-brand-800 bg-brand-800 px-4 py-2.5 font-semibold text-white"
115
+ >
116
+ {mode === "login" ? "Sign In" : "Create Account"}
117
+ </button>
118
+ {mode === "login" && (
119
+ <Link to="/forgot-password" className="mt-3 block w-full text-center text-sm text-brand-800 underline">
120
+ Forgot Password?
121
+ </Link>
122
+ )}
123
+ <button
124
+ type="button"
125
+ onClick={() => {
126
+ setMode(mode === "login" ? "register" : "login");
127
+ setError("");
128
+ setMessage("");
129
+ setForm({ username: form.username, email: "", password: "" });
130
+ }}
131
+ className="mt-3 w-full border border-brand-800 px-4 py-2.5 font-semibold text-brand-800"
132
+ >
133
+ {mode === "login"
134
+ ? "Need an account? Register"
135
+ : "Already have an account? Login"}
136
+ </button>
137
+ </form>
138
+ </div>
139
+ );
140
+ }
141
+
142
+ export default LoginPage;
@@ -0,0 +1,165 @@
1
+ import { useEffect, useState } from "react";
2
+ import FormCard from "../components/FormCard";
3
+ import api from "../services/api";
4
+
5
+ const empty = { productName: "", category: "", quantity: "", price: "" };
6
+
7
+ const validateProduct = (form) => {
8
+ const errors = {};
9
+ if (!form.productName.trim()) errors.productName = "Product name is required.";
10
+ if (!form.category.trim()) errors.category = "Category is required.";
11
+ if (form.quantity === "" || Number.isNaN(Number(form.quantity)) || Number(form.quantity) < 0) {
12
+ errors.quantity = "Quantity must be 0 or greater.";
13
+ }
14
+ if (!form.price || Number(form.price) <= 0) {
15
+ errors.price = "Price must be greater than 0.";
16
+ }
17
+ return errors;
18
+ };
19
+
20
+ function ProductsPage() {
21
+ const [products, setProducts] = useState([]);
22
+ const [form, setForm] = useState(empty);
23
+ const [editId, setEditId] = useState(null);
24
+ const [fieldErrors, setFieldErrors] = useState({});
25
+ const [formMessage, setFormMessage] = useState("");
26
+
27
+ const loadProducts = () => api.get("/products").then((res) => setProducts(res.data));
28
+
29
+ useEffect(() => {
30
+ loadProducts();
31
+ }, []);
32
+
33
+ const setField = (key, value) => {
34
+ setForm((prev) => ({ ...prev, [key]: value }));
35
+ setFieldErrors((prev) => ({ ...prev, [key]: "" }));
36
+ };
37
+
38
+ const inputClass = (key) =>
39
+ `w-full border bg-white p-2.5 text-brand-900 ${fieldErrors[key] ? "border-red-500" : "border-brand-100"}`;
40
+
41
+ const submit = async (e) => {
42
+ e.preventDefault();
43
+ setFormMessage("");
44
+ const errors = validateProduct(form);
45
+ if (Object.keys(errors).length) {
46
+ setFieldErrors(errors);
47
+ return;
48
+ }
49
+ try {
50
+ if (editId) await api.put(`/products/${editId}`, form);
51
+ else await api.post("/products", form);
52
+ setFormMessage(editId ? "Product updated successfully." : "Product saved successfully.");
53
+ setForm(empty);
54
+ setEditId(null);
55
+ setFieldErrors({});
56
+ loadProducts();
57
+ } catch (error) {
58
+ setFormMessage(error.response?.data?.message || "Unable to save product.");
59
+ }
60
+ };
61
+
62
+ return (
63
+ <div className="grid gap-6 lg:grid-cols-3">
64
+ <FormCard title={editId ? "Edit Product" : "Add Product"}>
65
+ <form onSubmit={submit} className="space-y-2">
66
+ <input
67
+ placeholder="Product Name"
68
+ value={form.productName}
69
+ onChange={(e) => setField("productName", e.target.value)}
70
+ className={inputClass("productName")}
71
+ />
72
+ {fieldErrors.productName && <p className="text-sm text-red-600">{fieldErrors.productName}</p>}
73
+ <input
74
+ placeholder="Category"
75
+ value={form.category}
76
+ onChange={(e) => setField("category", e.target.value)}
77
+ className={inputClass("category")}
78
+ />
79
+ {fieldErrors.category && <p className="text-sm text-red-600">{fieldErrors.category}</p>}
80
+ <input
81
+ type="number"
82
+ placeholder="Quantity"
83
+ value={form.quantity}
84
+ onChange={(e) => setField("quantity", e.target.value)}
85
+ className={inputClass("quantity")}
86
+ />
87
+ {fieldErrors.quantity && <p className="text-sm text-red-600">{fieldErrors.quantity}</p>}
88
+ <input
89
+ type="number"
90
+ placeholder="Price (RWF)"
91
+ value={form.price}
92
+ onChange={(e) => setField("price", e.target.value)}
93
+ className={inputClass("price")}
94
+ />
95
+ {fieldErrors.price && <p className="text-sm text-red-600">{fieldErrors.price}</p>}
96
+ {formMessage && <p className="text-sm text-brand-800">{formMessage}</p>}
97
+ <button className="w-full border border-brand-700 bg-brand-800 p-2.5 font-semibold text-white">
98
+ {editId ? "Update" : "Save"} Product
99
+ </button>
100
+ </form>
101
+ </FormCard>
102
+
103
+ <section className="border border-brand-100 bg-white p-5 shadow-sm lg:col-span-2 md:p-6">
104
+ <h2 className="mb-4 border-l-4 border-accent-500 pl-3 text-lg font-semibold text-brand-900">
105
+ All Products
106
+ </h2>
107
+ <div className="overflow-auto">
108
+ <table className="w-full text-sm text-brand-900">
109
+ <thead>
110
+ <tr className="text-left">
111
+ <th>Name</th>
112
+ <th>Category</th>
113
+ <th>Qty</th>
114
+ <th>Price</th>
115
+ <th>Actions</th>
116
+ </tr>
117
+ </thead>
118
+ <tbody>
119
+ {products.map((p) => (
120
+ <tr key={p.product_id} className="border-t border-brand-100">
121
+ <td>{p.product_name}</td>
122
+ <td>{p.category}</td>
123
+ <td>{p.quantity}</td>
124
+ <td>{p.price}</td>
125
+ <td className="space-x-2 py-2">
126
+ <button
127
+ onClick={() => {
128
+ setEditId(p.product_id);
129
+ setForm({
130
+ productName: p.product_name,
131
+ category: p.category,
132
+ quantity: p.quantity,
133
+ price: p.price,
134
+ });
135
+ setFieldErrors({});
136
+ }}
137
+ className="border border-brand-700 px-2 py-1 text-brand-900"
138
+ >
139
+ Edit
140
+ </button>
141
+ <button
142
+ onClick={async () => {
143
+ try {
144
+ await api.delete(`/products/${p.product_id}`);
145
+ loadProducts();
146
+ } catch (error) {
147
+ alert(error.response?.data?.message || "Delete failed.");
148
+ }
149
+ }}
150
+ className="border border-red-500 bg-red-500 px-2 py-1 text-white"
151
+ >
152
+ Delete
153
+ </button>
154
+ </td>
155
+ </tr>
156
+ ))}
157
+ </tbody>
158
+ </table>
159
+ </div>
160
+ </section>
161
+ </div>
162
+ );
163
+ }
164
+
165
+ export default ProductsPage;