@inventreedb/ui 0.0.1

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 (541) hide show
  1. package/.babelrc +8 -0
  2. package/.linguirc +55 -0
  3. package/README.md +17 -0
  4. package/index.html +15 -0
  5. package/lib/enums/ApiEndpoints.tsx +233 -0
  6. package/lib/enums/ModelType.tsx +38 -0
  7. package/lib/enums/Roles.tsx +25 -0
  8. package/lib/functions/Api.tsx +42 -0
  9. package/lib/functions/Plugins.tsx +27 -0
  10. package/lib/index.ts +11 -0
  11. package/lib/types/Auth.tsx +51 -0
  12. package/lib/types/Core.tsx +13 -0
  13. package/lib/types/Filters.tsx +69 -0
  14. package/lib/types/Forms.tsx +185 -0
  15. package/lib/types/Modals.tsx +17 -0
  16. package/lib/types/Plugins.tsx +68 -0
  17. package/lib/types/Server.tsx +8 -0
  18. package/lib/types/Settings.tsx +55 -0
  19. package/lib/types/Tables.tsx +69 -0
  20. package/lib/types/User.tsx +62 -0
  21. package/netlify.toml +22 -0
  22. package/package.json +125 -0
  23. package/playwright/global-setup.ts +45 -0
  24. package/playwright.config.ts +99 -0
  25. package/public/inventree.svg +291 -0
  26. package/src/App.tsx +35 -0
  27. package/src/assets/inventree.svg +1 -0
  28. package/src/components/Boundary.tsx +43 -0
  29. package/src/components/SplashScreen.tsx +35 -0
  30. package/src/components/barcodes/BarcodeCameraInput.tsx +207 -0
  31. package/src/components/barcodes/BarcodeInput.tsx +127 -0
  32. package/src/components/barcodes/BarcodeKeyboardInput.tsx +50 -0
  33. package/src/components/barcodes/BarcodeScanDialog.tsx +101 -0
  34. package/src/components/barcodes/BarcodeScanItem.tsx +23 -0
  35. package/src/components/barcodes/QRCode.tsx +211 -0
  36. package/src/components/buttons/ActionButton.tsx +61 -0
  37. package/src/components/buttons/AddItemButton.tsx +10 -0
  38. package/src/components/buttons/AdminButton.tsx +91 -0
  39. package/src/components/buttons/ButtonMenu.tsx +33 -0
  40. package/src/components/buttons/CopyButton.tsx +52 -0
  41. package/src/components/buttons/PrimaryActionButton.tsx +40 -0
  42. package/src/components/buttons/PrintingActions.tsx +191 -0
  43. package/src/components/buttons/RemoveRowButton.tsx +22 -0
  44. package/src/components/buttons/SSOButton.tsx +53 -0
  45. package/src/components/buttons/ScanButton.tsx +27 -0
  46. package/src/components/buttons/SegmentedIconControl.tsx +51 -0
  47. package/src/components/buttons/SplitButton.css.ts +19 -0
  48. package/src/components/buttons/SplitButton.tsx +111 -0
  49. package/src/components/buttons/SpotlightButton.tsx +22 -0
  50. package/src/components/buttons/StarredToggleButton.tsx +62 -0
  51. package/src/components/buttons/YesNoButton.tsx +42 -0
  52. package/src/components/calendar/Calendar.tsx +193 -0
  53. package/src/components/calendar/OrderCalendar.tsx +209 -0
  54. package/src/components/charts/colors.tsx +12 -0
  55. package/src/components/charts/tooltipFormatter.tsx +12 -0
  56. package/src/components/dashboard/DashboardLayout.tsx +327 -0
  57. package/src/components/dashboard/DashboardMenu.tsx +136 -0
  58. package/src/components/dashboard/DashboardWidget.tsx +82 -0
  59. package/src/components/dashboard/DashboardWidgetDrawer.tsx +130 -0
  60. package/src/components/dashboard/DashboardWidgetLibrary.tsx +189 -0
  61. package/src/components/dashboard/widgets/ColorToggleWidget.tsx +28 -0
  62. package/src/components/dashboard/widgets/GetStartedWidget.tsx +19 -0
  63. package/src/components/dashboard/widgets/LanguageSelectWidget.tsx +28 -0
  64. package/src/components/dashboard/widgets/NewsWidget.tsx +143 -0
  65. package/src/components/dashboard/widgets/QueryCountDashboardWidget.tsx +134 -0
  66. package/src/components/details/Details.tsx +514 -0
  67. package/src/components/details/DetailsBadge.tsx +20 -0
  68. package/src/components/details/DetailsImage.tsx +462 -0
  69. package/src/components/details/ItemDetails.tsx +17 -0
  70. package/src/components/editors/NotesEditor.tsx +232 -0
  71. package/src/components/editors/TemplateEditor/CodeEditor/CodeEditor.tsx +158 -0
  72. package/src/components/editors/TemplateEditor/CodeEditor/index.tsx +12 -0
  73. package/src/components/editors/TemplateEditor/PdfPreview/PdfPreview.tsx +90 -0
  74. package/src/components/editors/TemplateEditor/PdfPreview/index.tsx +12 -0
  75. package/src/components/editors/TemplateEditor/TemplateEditor.tsx +425 -0
  76. package/src/components/editors/TemplateEditor/index.tsx +3 -0
  77. package/src/components/errors/ClientError.tsx +28 -0
  78. package/src/components/errors/GenericErrorPage.tsx +73 -0
  79. package/src/components/errors/NotAuthenticated.tsx +12 -0
  80. package/src/components/errors/NotFound.tsx +12 -0
  81. package/src/components/errors/PermissionDenied.tsx +12 -0
  82. package/src/components/errors/ServerError.tsx +13 -0
  83. package/src/components/forms/ApiForm.tsx +718 -0
  84. package/src/components/forms/AuthFormOptions.tsx +39 -0
  85. package/src/components/forms/AuthenticationForm.tsx +332 -0
  86. package/src/components/forms/HostOptionsForm.tsx +96 -0
  87. package/src/components/forms/InstanceOptions.tsx +178 -0
  88. package/src/components/forms/StandaloneField.tsx +55 -0
  89. package/src/components/forms/fields/ApiFormField.tsx +281 -0
  90. package/src/components/forms/fields/ChoiceField.tsx +86 -0
  91. package/src/components/forms/fields/DateField.tsx +79 -0
  92. package/src/components/forms/fields/DependentField.tsx +91 -0
  93. package/src/components/forms/fields/IconField.tsx +353 -0
  94. package/src/components/forms/fields/NestedObjectField.tsx +45 -0
  95. package/src/components/forms/fields/RelatedModelField.tsx +339 -0
  96. package/src/components/forms/fields/TableField.tsx +275 -0
  97. package/src/components/forms/fields/TextField.tsx +81 -0
  98. package/src/components/images/ApiImage.tsx +35 -0
  99. package/src/components/images/Thumbnail.tsx +52 -0
  100. package/src/components/importer/ImportDataSelector.tsx +433 -0
  101. package/src/components/importer/ImporterColumnSelector.tsx +244 -0
  102. package/src/components/importer/ImporterDrawer.tsx +173 -0
  103. package/src/components/importer/ImporterImportProgress.tsx +45 -0
  104. package/src/components/items/ActionDropdown.tsx +290 -0
  105. package/src/components/items/ApiIcon.css.ts +13 -0
  106. package/src/components/items/ApiIcon.tsx +32 -0
  107. package/src/components/items/AttachmentLink.tsx +79 -0
  108. package/src/components/items/ColorToggle.tsx +34 -0
  109. package/src/components/items/DashboardItem.tsx +33 -0
  110. package/src/components/items/DocInfo.tsx +20 -0
  111. package/src/components/items/DocTooltip.tsx +98 -0
  112. package/src/components/items/ErrorItem.tsx +17 -0
  113. package/src/components/items/Expand.tsx +14 -0
  114. package/src/components/items/GettingStartedCarousel.css.ts +30 -0
  115. package/src/components/items/GettingStartedCarousel.tsx +49 -0
  116. package/src/components/items/InfoItem.tsx +64 -0
  117. package/src/components/items/InvenTreeLogo.tsx +24 -0
  118. package/src/components/items/LanguageSelect.tsx +43 -0
  119. package/src/components/items/LanguageToggle.tsx +37 -0
  120. package/src/components/items/MenuLinks.tsx +113 -0
  121. package/src/components/items/OnlyStaff.tsx +10 -0
  122. package/src/components/items/ProgressBar.tsx +45 -0
  123. package/src/components/items/StylishText.tsx +59 -0
  124. package/src/components/items/TitleWithDoc.tsx +26 -0
  125. package/src/components/items/UnavailableIndicator.tsx +5 -0
  126. package/src/components/items/inventree.svg +1 -0
  127. package/src/components/modals/AboutInvenTreeModal.tsx +203 -0
  128. package/src/components/modals/LicenseModal.tsx +116 -0
  129. package/src/components/modals/QrModal.tsx +21 -0
  130. package/src/components/modals/ServerInfoModal.tsx +136 -0
  131. package/src/components/nav/Alerts.tsx +130 -0
  132. package/src/components/nav/BreadcrumbList.tsx +84 -0
  133. package/src/components/nav/DetailDrawer.css.ts +6 -0
  134. package/src/components/nav/DetailDrawer.tsx +105 -0
  135. package/src/components/nav/Footer.tsx +11 -0
  136. package/src/components/nav/Header.tsx +226 -0
  137. package/src/components/nav/InstanceDetail.tsx +46 -0
  138. package/src/components/nav/Layout.tsx +85 -0
  139. package/src/components/nav/MainMenu.tsx +101 -0
  140. package/src/components/nav/NavHoverMenu.tsx +15 -0
  141. package/src/components/nav/NavigationDrawer.tsx +230 -0
  142. package/src/components/nav/NavigationTree.tsx +210 -0
  143. package/src/components/nav/NotificationDrawer.tsx +231 -0
  144. package/src/components/nav/PageDetail.tsx +171 -0
  145. package/src/components/nav/PageTitle.tsx +53 -0
  146. package/src/components/nav/SearchDrawer.tsx +598 -0
  147. package/src/components/nav/SettingsHeader.tsx +50 -0
  148. package/src/components/panels/AttachmentPanel.tsx +27 -0
  149. package/src/components/panels/NotesPanel.tsx +36 -0
  150. package/src/components/panels/Panel.tsx +15 -0
  151. package/src/components/panels/PanelGroup.css.ts +10 -0
  152. package/src/components/panels/PanelGroup.tsx +292 -0
  153. package/src/components/plugins/LocateItemButton.tsx +94 -0
  154. package/src/components/plugins/PluginContext.tsx +67 -0
  155. package/src/components/plugins/PluginDrawer.tsx +156 -0
  156. package/src/components/plugins/PluginInterface.tsx +34 -0
  157. package/src/components/plugins/PluginPanel.tsx +37 -0
  158. package/src/components/plugins/PluginSettingsPanel.tsx +34 -0
  159. package/src/components/plugins/PluginSource.tsx +48 -0
  160. package/src/components/plugins/PluginUIFeature.tsx +174 -0
  161. package/src/components/plugins/PluginUIFeatureTypes.ts +78 -0
  162. package/src/components/plugins/RemoteComponent.tsx +137 -0
  163. package/src/components/render/Build.tsx +54 -0
  164. package/src/components/render/Company.tsx +114 -0
  165. package/src/components/render/Generic.tsx +49 -0
  166. package/src/components/render/Instance.tsx +240 -0
  167. package/src/components/render/InstanceFromUrl.tsx +33 -0
  168. package/src/components/render/ModelType.tsx +293 -0
  169. package/src/components/render/Order.tsx +124 -0
  170. package/src/components/render/Part.tsx +109 -0
  171. package/src/components/render/Plugin.tsx +21 -0
  172. package/src/components/render/Report.tsx +29 -0
  173. package/src/components/render/StatusRenderer.tsx +178 -0
  174. package/src/components/render/Stock.tsx +84 -0
  175. package/src/components/render/User.tsx +42 -0
  176. package/src/components/settings/FactCollection.tsx +31 -0
  177. package/src/components/settings/FactItem.tsx +17 -0
  178. package/src/components/settings/SettingItem.tsx +176 -0
  179. package/src/components/settings/SettingList.tsx +217 -0
  180. package/src/components/wizards/OrderPartsWizard.tsx +473 -0
  181. package/src/components/wizards/WizardDrawer.tsx +188 -0
  182. package/src/contexts/ApiContext.tsx +31 -0
  183. package/src/contexts/LanguageContext.tsx +155 -0
  184. package/src/contexts/ThemeContext.tsx +58 -0
  185. package/src/contexts/colorSchema.tsx +67 -0
  186. package/src/defaults/actions.tsx +87 -0
  187. package/src/defaults/backendMappings.tsx +32 -0
  188. package/src/defaults/defaultHostList.tsx +4 -0
  189. package/src/defaults/defaults.tsx +38 -0
  190. package/src/defaults/formatters.tsx +176 -0
  191. package/src/defaults/links.tsx +193 -0
  192. package/src/defaults/templates.tsx +53 -0
  193. package/src/forms/BomForms.tsx +26 -0
  194. package/src/forms/BuildForms.tsx +615 -0
  195. package/src/forms/CommonForms.tsx +89 -0
  196. package/src/forms/CompanyForms.tsx +133 -0
  197. package/src/forms/ImporterForms.tsx +20 -0
  198. package/src/forms/PartForms.tsx +281 -0
  199. package/src/forms/PurchaseOrderForms.tsx +779 -0
  200. package/src/forms/ReturnOrderForms.tsx +263 -0
  201. package/src/forms/SalesOrderForms.tsx +408 -0
  202. package/src/forms/StockForms.tsx +1296 -0
  203. package/src/forms/selectionListFields.tsx +120 -0
  204. package/src/functions/api.tsx +61 -0
  205. package/src/functions/auth.tsx +574 -0
  206. package/src/functions/conversion.tsx +42 -0
  207. package/src/functions/events.tsx +6 -0
  208. package/src/functions/forms.tsx +173 -0
  209. package/src/functions/icons.tsx +294 -0
  210. package/src/functions/loading.tsx +29 -0
  211. package/src/functions/navigation.tsx +20 -0
  212. package/src/functions/notifications.tsx +101 -0
  213. package/src/functions/tables.tsx +26 -0
  214. package/src/functions/uid.tsx +15 -0
  215. package/src/functions/urls.tsx +58 -0
  216. package/src/hooks/UseCalendar.tsx +178 -0
  217. package/src/hooks/UseDashboardItems.tsx +118 -0
  218. package/src/hooks/UseDataExport.tsx +125 -0
  219. package/src/hooks/UseDataOutput.tsx +109 -0
  220. package/src/hooks/UseFilter.tsx +67 -0
  221. package/src/hooks/UseFilterSet.tsx +24 -0
  222. package/src/hooks/UseForm.tsx +161 -0
  223. package/src/hooks/UseGenerator.tsx +92 -0
  224. package/src/hooks/UseImportSession.tsx +139 -0
  225. package/src/hooks/UseInstance.tsx +137 -0
  226. package/src/hooks/UseInstanceName.tsx +14 -0
  227. package/src/hooks/UseModal.tsx +38 -0
  228. package/src/hooks/UsePlaceholder.tsx +66 -0
  229. package/src/hooks/UsePluginPanels.tsx +123 -0
  230. package/src/hooks/UsePluginUIFeature.tsx +95 -0
  231. package/src/hooks/UsePlugins.tsx +45 -0
  232. package/src/hooks/UseSelectedRows.tsx +37 -0
  233. package/src/hooks/UseStatusCodes.tsx +50 -0
  234. package/src/hooks/UseTable.tsx +145 -0
  235. package/src/hooks/UseWizard.tsx +133 -0
  236. package/src/locales/ar/messages.d.ts +4 -0
  237. package/src/locales/ar/messages.po +10695 -0
  238. package/src/locales/ar/messages.ts +1 -0
  239. package/src/locales/bg/messages.d.ts +4 -0
  240. package/src/locales/bg/messages.po +10695 -0
  241. package/src/locales/bg/messages.ts +1 -0
  242. package/src/locales/cs/messages.d.ts +4 -0
  243. package/src/locales/cs/messages.po +10695 -0
  244. package/src/locales/cs/messages.ts +1 -0
  245. package/src/locales/da/messages.d.ts +4 -0
  246. package/src/locales/da/messages.po +10695 -0
  247. package/src/locales/da/messages.ts +1 -0
  248. package/src/locales/de/messages.d.ts +4 -0
  249. package/src/locales/de/messages.po +10695 -0
  250. package/src/locales/de/messages.ts +1 -0
  251. package/src/locales/el/messages.d.ts +4 -0
  252. package/src/locales/el/messages.po +10695 -0
  253. package/src/locales/el/messages.ts +1 -0
  254. package/src/locales/en/messages.d.ts +4 -0
  255. package/src/locales/en/messages.po +10689 -0
  256. package/src/locales/en/messages.ts +1 -0
  257. package/src/locales/es/messages.d.ts +4 -0
  258. package/src/locales/es/messages.po +10695 -0
  259. package/src/locales/es/messages.ts +1 -0
  260. package/src/locales/es_MX/messages.d.ts +4 -0
  261. package/src/locales/es_MX/messages.po +10695 -0
  262. package/src/locales/es_MX/messages.ts +1 -0
  263. package/src/locales/et/messages.d.ts +4 -0
  264. package/src/locales/et/messages.po +10695 -0
  265. package/src/locales/et/messages.ts +1 -0
  266. package/src/locales/fa/messages.d.ts +4 -0
  267. package/src/locales/fa/messages.po +10695 -0
  268. package/src/locales/fa/messages.ts +1 -0
  269. package/src/locales/fi/messages.d.ts +4 -0
  270. package/src/locales/fi/messages.po +10695 -0
  271. package/src/locales/fi/messages.ts +1 -0
  272. package/src/locales/fr/messages.d.ts +4 -0
  273. package/src/locales/fr/messages.po +10695 -0
  274. package/src/locales/fr/messages.ts +1 -0
  275. package/src/locales/he/messages.d.ts +4 -0
  276. package/src/locales/he/messages.po +10695 -0
  277. package/src/locales/he/messages.ts +1 -0
  278. package/src/locales/hi/messages.d.ts +4 -0
  279. package/src/locales/hi/messages.po +10695 -0
  280. package/src/locales/hi/messages.ts +1 -0
  281. package/src/locales/hu/messages.d.ts +4 -0
  282. package/src/locales/hu/messages.po +10695 -0
  283. package/src/locales/hu/messages.ts +1 -0
  284. package/src/locales/id/messages.po +10695 -0
  285. package/src/locales/it/messages.d.ts +4 -0
  286. package/src/locales/it/messages.po +10695 -0
  287. package/src/locales/it/messages.ts +1 -0
  288. package/src/locales/ja/messages.d.ts +4 -0
  289. package/src/locales/ja/messages.po +10695 -0
  290. package/src/locales/ja/messages.ts +1 -0
  291. package/src/locales/ko/messages.d.ts +4 -0
  292. package/src/locales/ko/messages.po +10695 -0
  293. package/src/locales/ko/messages.ts +1 -0
  294. package/src/locales/lt/messages.d.ts +4 -0
  295. package/src/locales/lt/messages.po +10695 -0
  296. package/src/locales/lt/messages.ts +1 -0
  297. package/src/locales/lv/messages.d.ts +4 -0
  298. package/src/locales/lv/messages.po +10695 -0
  299. package/src/locales/lv/messages.ts +1 -0
  300. package/src/locales/nl/messages.d.ts +4 -0
  301. package/src/locales/nl/messages.po +10695 -0
  302. package/src/locales/nl/messages.ts +1 -0
  303. package/src/locales/no/messages.d.ts +4 -0
  304. package/src/locales/no/messages.po +10695 -0
  305. package/src/locales/no/messages.ts +1 -0
  306. package/src/locales/pl/messages.d.ts +4 -0
  307. package/src/locales/pl/messages.po +10695 -0
  308. package/src/locales/pl/messages.ts +1 -0
  309. package/src/locales/pseudo-LOCALE/messages.d.ts +4 -0
  310. package/src/locales/pseudo-LOCALE/messages.po +8003 -0
  311. package/src/locales/pseudo-LOCALE/messages.ts +1 -0
  312. package/src/locales/pt/messages.d.ts +4 -0
  313. package/src/locales/pt/messages.po +10696 -0
  314. package/src/locales/pt/messages.ts +1 -0
  315. package/src/locales/pt_BR/messages.d.ts +4 -0
  316. package/src/locales/pt_BR/messages.po +10695 -0
  317. package/src/locales/pt_BR/messages.ts +1 -0
  318. package/src/locales/ro/messages.po +10695 -0
  319. package/src/locales/ro/messages.ts +1 -0
  320. package/src/locales/ru/messages.d.ts +4 -0
  321. package/src/locales/ru/messages.po +10695 -0
  322. package/src/locales/ru/messages.ts +1 -0
  323. package/src/locales/sk/messages.d.ts +4 -0
  324. package/src/locales/sk/messages.po +10695 -0
  325. package/src/locales/sk/messages.ts +1 -0
  326. package/src/locales/sl/messages.d.ts +4 -0
  327. package/src/locales/sl/messages.po +10695 -0
  328. package/src/locales/sl/messages.ts +1 -0
  329. package/src/locales/sr/messages.d.ts +4 -0
  330. package/src/locales/sr/messages.po +10695 -0
  331. package/src/locales/sr/messages.ts +1 -0
  332. package/src/locales/sv/messages.d.ts +4 -0
  333. package/src/locales/sv/messages.po +10695 -0
  334. package/src/locales/sv/messages.ts +1 -0
  335. package/src/locales/th/messages.d.ts +4 -0
  336. package/src/locales/th/messages.po +10695 -0
  337. package/src/locales/th/messages.ts +1 -0
  338. package/src/locales/tr/messages.d.ts +4 -0
  339. package/src/locales/tr/messages.po +10695 -0
  340. package/src/locales/tr/messages.ts +1 -0
  341. package/src/locales/uk/messages.d.ts +3 -0
  342. package/src/locales/uk/messages.po +10695 -0
  343. package/src/locales/uk/messages.ts +1 -0
  344. package/src/locales/vi/messages.d.ts +4 -0
  345. package/src/locales/vi/messages.po +10695 -0
  346. package/src/locales/vi/messages.ts +1 -0
  347. package/src/locales/zh_Hans/messages.d.ts +4 -0
  348. package/src/locales/zh_Hans/messages.po +10695 -0
  349. package/src/locales/zh_Hans/messages.ts +1 -0
  350. package/src/locales/zh_Hant/messages.d.ts +4 -0
  351. package/src/locales/zh_Hant/messages.po +10695 -0
  352. package/src/locales/zh_Hant/messages.ts +1 -0
  353. package/src/main.css.ts +148 -0
  354. package/src/main.tsx +123 -0
  355. package/src/pages/Auth/ChangePassword.tsx +84 -0
  356. package/src/pages/Auth/Layout.tsx +74 -0
  357. package/src/pages/Auth/LoggedIn.tsx +21 -0
  358. package/src/pages/Auth/Login.tsx +133 -0
  359. package/src/pages/Auth/Logout.tsx +16 -0
  360. package/src/pages/Auth/MFA.tsx +36 -0
  361. package/src/pages/Auth/MFASetup.tsx +38 -0
  362. package/src/pages/Auth/Register.tsx +28 -0
  363. package/src/pages/Auth/Reset.tsx +30 -0
  364. package/src/pages/Auth/ResetPassword.tsx +48 -0
  365. package/src/pages/Auth/VerifyEmail.tsx +34 -0
  366. package/src/pages/ErrorPage.tsx +26 -0
  367. package/src/pages/Index/Home.tsx +12 -0
  368. package/src/pages/Index/Scan.tsx +258 -0
  369. package/src/pages/Index/Settings/AccountSettings/AccountDetailPanel.tsx +162 -0
  370. package/src/pages/Index/Settings/AccountSettings/QrRegistrationForm.tsx +38 -0
  371. package/src/pages/Index/Settings/AccountSettings/SecurityContent.tsx +713 -0
  372. package/src/pages/Index/Settings/AccountSettings/UserPanel.tsx +24 -0
  373. package/src/pages/Index/Settings/AccountSettings/UserThemePanel.tsx +207 -0
  374. package/src/pages/Index/Settings/AccountSettings/useConfirm.tsx +117 -0
  375. package/src/pages/Index/Settings/AdminCenter/CurrencyManagementPanel.tsx +111 -0
  376. package/src/pages/Index/Settings/AdminCenter/Index.tsx +251 -0
  377. package/src/pages/Index/Settings/AdminCenter/LabelTemplatePanel.tsx +23 -0
  378. package/src/pages/Index/Settings/AdminCenter/MachineManagementPanel.tsx +110 -0
  379. package/src/pages/Index/Settings/AdminCenter/PartParameterPanel.tsx +29 -0
  380. package/src/pages/Index/Settings/AdminCenter/PluginManagementPanel.tsx +84 -0
  381. package/src/pages/Index/Settings/AdminCenter/ReportTemplatePanel.tsx +38 -0
  382. package/src/pages/Index/Settings/AdminCenter/StocktakePanel.tsx +31 -0
  383. package/src/pages/Index/Settings/AdminCenter/TaskManagementPanel.tsx +73 -0
  384. package/src/pages/Index/Settings/AdminCenter/UnitManagementPanel.tsx +74 -0
  385. package/src/pages/Index/Settings/AdminCenter/UserManagementPanel.tsx +49 -0
  386. package/src/pages/Index/Settings/SystemSettings.tsx +339 -0
  387. package/src/pages/Index/Settings/UserSettings.tsx +143 -0
  388. package/src/pages/Notifications.tsx +134 -0
  389. package/src/pages/build/BuildDetail.tsx +579 -0
  390. package/src/pages/build/BuildIndex.tsx +100 -0
  391. package/src/pages/company/CompanyDetail.tsx +352 -0
  392. package/src/pages/company/CustomerDetail.tsx +13 -0
  393. package/src/pages/company/ManufacturerDetail.tsx +13 -0
  394. package/src/pages/company/ManufacturerPartDetail.tsx +307 -0
  395. package/src/pages/company/SupplierDetail.tsx +13 -0
  396. package/src/pages/company/SupplierPartDetail.tsx +433 -0
  397. package/src/pages/core/CoreIndex.tsx +50 -0
  398. package/src/pages/core/GroupDetail.tsx +97 -0
  399. package/src/pages/core/UserDetail.tsx +193 -0
  400. package/src/pages/part/CategoryDetail.tsx +369 -0
  401. package/src/pages/part/PartAllocationPanel.tsx +40 -0
  402. package/src/pages/part/PartDetail.tsx +1071 -0
  403. package/src/pages/part/PartPricingPanel.tsx +155 -0
  404. package/src/pages/part/PartSchedulingDetail.tsx +315 -0
  405. package/src/pages/part/PartStocktakeDetail.tsx +285 -0
  406. package/src/pages/part/PartSupplierDetail.tsx +31 -0
  407. package/src/pages/part/pricing/BomPricingPanel.tsx +269 -0
  408. package/src/pages/part/pricing/PriceBreakPanel.tsx +182 -0
  409. package/src/pages/part/pricing/PricingOverviewPanel.tsx +338 -0
  410. package/src/pages/part/pricing/PricingPanel.tsx +82 -0
  411. package/src/pages/part/pricing/PurchaseHistoryPanel.tsx +134 -0
  412. package/src/pages/part/pricing/SaleHistoryPanel.tsx +95 -0
  413. package/src/pages/part/pricing/SupplierPricingPanel.tsx +80 -0
  414. package/src/pages/part/pricing/VariantPricingPanel.tsx +112 -0
  415. package/src/pages/purchasing/PurchaseOrderDetail.tsx +547 -0
  416. package/src/pages/purchasing/PurchasingIndex.tsx +131 -0
  417. package/src/pages/sales/ReturnOrderDetail.tsx +529 -0
  418. package/src/pages/sales/SalesIndex.tsx +148 -0
  419. package/src/pages/sales/SalesOrderDetail.tsx +593 -0
  420. package/src/pages/sales/SalesOrderShipmentDetail.tsx +385 -0
  421. package/src/pages/stock/LocationDetail.tsx +413 -0
  422. package/src/pages/stock/StockDetail.tsx +951 -0
  423. package/src/router.tsx +213 -0
  424. package/src/states/ApiState.tsx +85 -0
  425. package/src/states/IconState.tsx +94 -0
  426. package/src/states/LocalState.tsx +181 -0
  427. package/src/states/SettingsState.tsx +206 -0
  428. package/src/states/StatusState.tsx +55 -0
  429. package/src/states/UserState.tsx +202 -0
  430. package/src/states/states.tsx +69 -0
  431. package/src/tables/Column.tsx +29 -0
  432. package/src/tables/ColumnRenderers.tsx +320 -0
  433. package/src/tables/ColumnSelect.tsx +39 -0
  434. package/src/tables/Filter.tsx +297 -0
  435. package/src/tables/FilterSelectDrawer.tsx +358 -0
  436. package/src/tables/InvenTreeTable.tsx +750 -0
  437. package/src/tables/InvenTreeTableHeader.tsx +257 -0
  438. package/src/tables/RowActions.tsx +176 -0
  439. package/src/tables/RowExpansionIcon.tsx +16 -0
  440. package/src/tables/Search.tsx +42 -0
  441. package/src/tables/TableHoverCard.tsx +91 -0
  442. package/src/tables/bom/BomTable.tsx +590 -0
  443. package/src/tables/bom/UsedInTable.tsx +114 -0
  444. package/src/tables/build/BuildAllocatedStockTable.tsx +233 -0
  445. package/src/tables/build/BuildLineTable.tsx +800 -0
  446. package/src/tables/build/BuildOrderTable.tsx +226 -0
  447. package/src/tables/build/BuildOrderTestTable.tsx +265 -0
  448. package/src/tables/build/BuildOutputTable.tsx +599 -0
  449. package/src/tables/company/AddressTable.tsx +211 -0
  450. package/src/tables/company/CompanyTable.tsx +176 -0
  451. package/src/tables/company/ContactTable.tsx +182 -0
  452. package/src/tables/core/UserTable.tsx +88 -0
  453. package/src/tables/general/AttachmentTable.tsx +404 -0
  454. package/src/tables/general/BarcodeScanTable.tsx +124 -0
  455. package/src/tables/general/ExtraLineItemTable.tsx +174 -0
  456. package/src/tables/machine/MachineListTable.tsx +634 -0
  457. package/src/tables/machine/MachineTypeTable.tsx +395 -0
  458. package/src/tables/notifications/NotificationTable.tsx +60 -0
  459. package/src/tables/part/ParametricPartTable.tsx +297 -0
  460. package/src/tables/part/PartBuildAllocationsTable.tsx +128 -0
  461. package/src/tables/part/PartCategoryTable.tsx +208 -0
  462. package/src/tables/part/PartCategoryTemplateTable.tsx +157 -0
  463. package/src/tables/part/PartParameterTable.tsx +226 -0
  464. package/src/tables/part/PartParameterTemplateTable.tsx +168 -0
  465. package/src/tables/part/PartPurchaseOrdersTable.tsx +163 -0
  466. package/src/tables/part/PartSalesAllocationsTable.tsx +130 -0
  467. package/src/tables/part/PartTable.tsx +423 -0
  468. package/src/tables/part/PartTestTemplateTable.tsx +290 -0
  469. package/src/tables/part/PartThumbTable.tsx +230 -0
  470. package/src/tables/part/PartVariantTable.tsx +52 -0
  471. package/src/tables/part/RelatedPartTable.tsx +182 -0
  472. package/src/tables/part/SelectionListTable.tsx +134 -0
  473. package/src/tables/plugin/PluginErrorTable.tsx +58 -0
  474. package/src/tables/plugin/PluginListTable.tsx +440 -0
  475. package/src/tables/purchasing/ManufacturerPartParameterTable.tsx +136 -0
  476. package/src/tables/purchasing/ManufacturerPartTable.tsx +158 -0
  477. package/src/tables/purchasing/PurchaseOrderLineItemTable.tsx +424 -0
  478. package/src/tables/purchasing/PurchaseOrderTable.tsx +193 -0
  479. package/src/tables/purchasing/SupplierPartTable.tsx +281 -0
  480. package/src/tables/purchasing/SupplierPriceBreakTable.tsx +222 -0
  481. package/src/tables/sales/ReturnOrderLineItemTable.tsx +271 -0
  482. package/src/tables/sales/ReturnOrderTable.tsx +201 -0
  483. package/src/tables/sales/SalesOrderAllocationTable.tsx +330 -0
  484. package/src/tables/sales/SalesOrderLineItemTable.tsx +508 -0
  485. package/src/tables/sales/SalesOrderShipmentTable.tsx +225 -0
  486. package/src/tables/sales/SalesOrderTable.tsx +212 -0
  487. package/src/tables/settings/ApiTokenTable.tsx +203 -0
  488. package/src/tables/settings/BarcodeScanHistoryTable.tsx +281 -0
  489. package/src/tables/settings/CustomStateTable.tsx +226 -0
  490. package/src/tables/settings/CustomUnitsTable.tsx +125 -0
  491. package/src/tables/settings/ErrorTable.tsx +170 -0
  492. package/src/tables/settings/ExportSessionTable.tsx +60 -0
  493. package/src/tables/settings/FailedTasksTable.tsx +103 -0
  494. package/src/tables/settings/GroupTable.tsx +240 -0
  495. package/src/tables/settings/ImportSessionTable.tsx +175 -0
  496. package/src/tables/settings/PendingTasksTable.tsx +63 -0
  497. package/src/tables/settings/ProjectCodeTable.tsx +115 -0
  498. package/src/tables/settings/ScheduledTasksTable.tsx +62 -0
  499. package/src/tables/settings/StocktakeReportTable.tsx +111 -0
  500. package/src/tables/settings/TemplateTable.tsx +404 -0
  501. package/src/tables/settings/UserTable.tsx +329 -0
  502. package/src/tables/stock/InstalledItemsTable.tsx +146 -0
  503. package/src/tables/stock/LocationTypesTable.tsx +132 -0
  504. package/src/tables/stock/StockItemTable.tsx +707 -0
  505. package/src/tables/stock/StockItemTestResultTable.tsx +487 -0
  506. package/src/tables/stock/StockLocationTable.tsx +208 -0
  507. package/src/tables/stock/StockTrackingTable.tsx +244 -0
  508. package/src/theme.ts +5 -0
  509. package/src/views/DesktopAppView.tsx +28 -0
  510. package/src/views/MainView.tsx +38 -0
  511. package/src/views/MobileAppView.tsx +43 -0
  512. package/tests/baseFixtures.ts +93 -0
  513. package/tests/defaults.ts +18 -0
  514. package/tests/helpers.ts +122 -0
  515. package/tests/login.ts +97 -0
  516. package/tests/pages/pui_build.spec.ts +373 -0
  517. package/tests/pages/pui_company.spec.ts +40 -0
  518. package/tests/pages/pui_core.spec.ts +26 -0
  519. package/tests/pages/pui_dashboard.spec.ts +64 -0
  520. package/tests/pages/pui_part.spec.ts +413 -0
  521. package/tests/pages/pui_purchase_order.spec.ts +388 -0
  522. package/tests/pages/pui_sales_order.spec.ts +224 -0
  523. package/tests/pages/pui_scan.spec.ts +114 -0
  524. package/tests/pages/pui_stock.spec.ts +259 -0
  525. package/tests/pui_exporting.spec.ts +124 -0
  526. package/tests/pui_forms.spec.ts +131 -0
  527. package/tests/pui_general.spec.ts +60 -0
  528. package/tests/pui_login.spec.ts +74 -0
  529. package/tests/pui_modals.spec.ts +144 -0
  530. package/tests/pui_plugins.spec.ts +216 -0
  531. package/tests/pui_printing.spec.ts +149 -0
  532. package/tests/pui_settings.spec.ts +285 -0
  533. package/tests/pui_tables.spec.ts +65 -0
  534. package/tests/settings/selectionList.spec.ts +103 -0
  535. package/tests/settings.ts +54 -0
  536. package/tsconfig.json +25 -0
  537. package/tsconfig.lib.json +13 -0
  538. package/tsconfig.node.json +11 -0
  539. package/vite-env.d.ts +11 -0
  540. package/vite.config.ts +98 -0
  541. package/vite.lib.config.ts +50 -0
@@ -0,0 +1,718 @@
1
+ import { t } from '@lingui/core/macro';
2
+ import {
3
+ Alert,
4
+ Button,
5
+ Divider,
6
+ Group,
7
+ LoadingOverlay,
8
+ Paper,
9
+ Stack,
10
+ Text
11
+ } from '@mantine/core';
12
+ import { useId } from '@mantine/hooks';
13
+ import { notifications } from '@mantine/notifications';
14
+ import { useQuery, useQueryClient } from '@tanstack/react-query';
15
+ import { useCallback, useEffect, useMemo, useState } from 'react';
16
+ import {
17
+ type FieldValues,
18
+ FormProvider,
19
+ type SubmitErrorHandler,
20
+ type SubmitHandler,
21
+ useForm
22
+ } from 'react-hook-form';
23
+ import { type NavigateFunction, useNavigate } from 'react-router-dom';
24
+
25
+ import type {
26
+ ApiFormFieldSet,
27
+ ApiFormFieldType,
28
+ ApiFormProps
29
+ } from '@lib/types/Forms';
30
+ import { useApi } from '../../contexts/ApiContext';
31
+ import {
32
+ type NestedDict,
33
+ constructField,
34
+ constructFormUrl,
35
+ extractAvailableFields,
36
+ mapFields
37
+ } from '../../functions/forms';
38
+ import {
39
+ invalidResponse,
40
+ showTimeoutNotification
41
+ } from '../../functions/notifications';
42
+ import { getDetailUrl } from '../../functions/urls';
43
+ import { Boundary } from '../Boundary';
44
+ import { ApiFormField } from './fields/ApiFormField';
45
+
46
+ export function OptionsApiForm({
47
+ props: _props,
48
+ id: pId
49
+ }: Readonly<{
50
+ props: ApiFormProps;
51
+ id?: string;
52
+ }>) {
53
+ const api = useApi();
54
+
55
+ const props = useMemo(
56
+ () => ({
57
+ ..._props,
58
+ method: _props.method || 'GET'
59
+ }),
60
+ [_props]
61
+ );
62
+
63
+ const id = useId(pId);
64
+
65
+ const url = useMemo(
66
+ () =>
67
+ constructFormUrl(
68
+ props.url,
69
+ props.pk,
70
+ props.pathParams,
71
+ props.queryParams
72
+ ),
73
+ [props.url, props.pk, props.pathParams, props.queryParams]
74
+ );
75
+
76
+ const optionsQuery = useQuery({
77
+ enabled: true,
78
+ refetchOnMount: false,
79
+ queryKey: [
80
+ 'form-options-data',
81
+ id,
82
+ props.method,
83
+ props.url,
84
+ props.pk,
85
+ props.pathParams
86
+ ],
87
+ queryFn: async () => {
88
+ const response = await api.options(url);
89
+ let fields: Record<string, ApiFormFieldType> | null = {};
90
+ if (!props.ignorePermissionCheck) {
91
+ fields = extractAvailableFields(response, props.method);
92
+ }
93
+
94
+ return fields;
95
+ },
96
+ throwOnError: (error: any) => {
97
+ if (error.response) {
98
+ invalidResponse(error.response.status);
99
+ } else {
100
+ notifications.hide('form-error');
101
+ notifications.show({
102
+ title: t`Form Error`,
103
+ message: error.message,
104
+ color: 'red',
105
+ id: 'form-error'
106
+ });
107
+ }
108
+ return false;
109
+ }
110
+ });
111
+
112
+ const formProps: ApiFormProps = useMemo(() => {
113
+ const _props = { ...props };
114
+
115
+ if (!_props.fields) return _props;
116
+
117
+ for (const [k, v] of Object.entries(_props.fields)) {
118
+ _props.fields[k] = constructField({
119
+ field: v,
120
+ definition: optionsQuery?.data?.[k]
121
+ });
122
+
123
+ // If the user has specified initial data, use that value here
124
+ const value = _props?.initialData?.[k];
125
+
126
+ if (value) {
127
+ _props.fields[k].value = value;
128
+ }
129
+ }
130
+
131
+ return _props;
132
+ }, [optionsQuery.data, props]);
133
+
134
+ return (
135
+ <ApiForm
136
+ id={id}
137
+ props={formProps}
138
+ optionsLoading={optionsQuery.isFetching || !optionsQuery.data}
139
+ />
140
+ );
141
+ }
142
+
143
+ /**
144
+ * An ApiForm component is a modal form which is rendered dynamically,
145
+ * based on an API endpoint.
146
+ */
147
+ export function ApiForm({
148
+ id,
149
+ props,
150
+ optionsLoading
151
+ }: Readonly<{
152
+ id: string;
153
+ props: ApiFormProps;
154
+ optionsLoading: boolean;
155
+ }>) {
156
+ const api = useApi();
157
+ const queryClient = useQueryClient();
158
+
159
+ // Accessor for the navigation function (which is used to redirect the user)
160
+ let navigate: NavigateFunction | null = null;
161
+
162
+ try {
163
+ navigate = useNavigate();
164
+ } catch (_error) {
165
+ // Note: If we launch a form within a plugin context, useNavigate() may not be available
166
+ navigate = null;
167
+ }
168
+
169
+ const [fields, setFields] = useState<ApiFormFieldSet>(
170
+ () => props.fields ?? {}
171
+ );
172
+
173
+ const defaultValues: FieldValues = useMemo(() => {
174
+ const defaultValuesMap = mapFields(fields ?? {}, (_path, field) => {
175
+ return field.value ?? field.default ?? undefined;
176
+ });
177
+
178
+ // If the user has specified initial data, that overrides default values
179
+ // But, *only* for the fields we have specified
180
+ if (props.initialData) {
181
+ Object.keys(props.initialData).forEach((key) => {
182
+ if (key in defaultValuesMap) {
183
+ defaultValuesMap[key] =
184
+ props?.initialData?.[key] ?? defaultValuesMap[key];
185
+ }
186
+ });
187
+ }
188
+
189
+ return defaultValuesMap;
190
+ }, [props.fields, props.initialData]);
191
+
192
+ // Form errors which are not associated with a specific field
193
+ const [nonFieldErrors, setNonFieldErrors] = useState<string[]>([]);
194
+
195
+ // Form state
196
+ const form = useForm({
197
+ criteriaMode: 'all',
198
+ defaultValues
199
+ });
200
+
201
+ const {
202
+ isValid,
203
+ isDirty,
204
+ isLoading: isFormLoading,
205
+ isSubmitting
206
+ } = form.formState;
207
+
208
+ // Cache URL
209
+ const url = useMemo(
210
+ () =>
211
+ constructFormUrl(
212
+ props.url,
213
+ props.pk,
214
+ props.pathParams,
215
+ props.queryParams
216
+ ),
217
+ [props.url, props.pk, props.pathParams]
218
+ );
219
+
220
+ // Define function to process API response
221
+ const processFields = (fields: ApiFormFieldSet, data: NestedDict) => {
222
+ const res: NestedDict = {};
223
+
224
+ for (const [k, field] of Object.entries(fields)) {
225
+ const dataValue = data[k];
226
+
227
+ if (
228
+ field.field_type === 'nested object' &&
229
+ field.children &&
230
+ typeof dataValue === 'object'
231
+ ) {
232
+ res[k] = processFields(field.children, dataValue);
233
+ } else {
234
+ res[k] = dataValue;
235
+
236
+ if (field.onValueChange) {
237
+ field.onValueChange(dataValue, data);
238
+ }
239
+ }
240
+ }
241
+
242
+ return res;
243
+ };
244
+
245
+ // Query manager for retrieving initial data from the server
246
+ const initialDataQuery = useQuery({
247
+ enabled: false,
248
+ queryKey: [
249
+ 'form-initial-data',
250
+ id,
251
+ props.method,
252
+ props.url,
253
+ props.pk,
254
+ props.pathParams
255
+ ],
256
+ queryFn: async () => {
257
+ return await api
258
+ .get(url)
259
+ .then((response: any) => {
260
+ // Process API response
261
+ const fetchedData: any = processFields(fields, response.data);
262
+
263
+ // Update form values, but only for the fields specified for this form
264
+ form.reset(fetchedData);
265
+ return fetchedData;
266
+ })
267
+ .catch(() => {
268
+ return {};
269
+ });
270
+ }
271
+ });
272
+
273
+ useEffect(() => {
274
+ const _fields: any = props.fields || {};
275
+ const _initialData: any = props.initialData || {};
276
+ const _fetchedData: any = initialDataQuery.data || {};
277
+
278
+ for (const k of Object.keys(_fields)) {
279
+ // Ensure default values override initial field spec
280
+ if (k in defaultValues) {
281
+ _fields[k].value = defaultValues[k];
282
+ }
283
+
284
+ // Ensure initial data overrides default values
285
+ if (_initialData && k in _initialData) {
286
+ _fields[k].value = _initialData[k];
287
+ }
288
+
289
+ // Ensure fetched data overrides also
290
+ if (_fetchedData && k in _fetchedData) {
291
+ _fields[k].value = _fetchedData[k];
292
+ }
293
+ }
294
+
295
+ setFields(_fields);
296
+ }, [props.fields, props.initialData, defaultValues, initialDataQuery.data]);
297
+
298
+ // Fetch initial data on form load
299
+ useEffect(() => {
300
+ // Fetch initial data if the fetchInitialData property is set
301
+ if (!optionsLoading && props.fetchInitialData) {
302
+ queryClient.removeQueries({
303
+ queryKey: [
304
+ 'form-initial-data',
305
+ id,
306
+ props.method,
307
+ props.url,
308
+ props.pk,
309
+ props.pathParams
310
+ ]
311
+ });
312
+ initialDataQuery.refetch();
313
+ }
314
+ }, [props.fetchInitialData, optionsLoading]);
315
+
316
+ const isLoading: boolean = useMemo(
317
+ () =>
318
+ isFormLoading ||
319
+ initialDataQuery.isFetching ||
320
+ optionsLoading ||
321
+ isSubmitting ||
322
+ !fields,
323
+ [isFormLoading, initialDataQuery, isSubmitting, fields, optionsLoading]
324
+ );
325
+
326
+ const [initialFocus, setInitialFocus] = useState<string>('');
327
+
328
+ // Update field focus when the form is loaded
329
+ useEffect(() => {
330
+ let focusField = props.focus ?? '';
331
+
332
+ if (!focusField) {
333
+ // If a focus field is not specified, then focus on the first available field
334
+ Object.entries(fields).forEach(([fieldName, field]) => {
335
+ if (focusField || field.read_only || field.disabled || field.hidden) {
336
+ return;
337
+ }
338
+
339
+ // Do not auto-focus on a 'choice' field
340
+ if (field.field_type == 'choice') {
341
+ return;
342
+ }
343
+
344
+ focusField = fieldName;
345
+ });
346
+ }
347
+
348
+ if (isLoading) {
349
+ return;
350
+ }
351
+
352
+ form.setFocus(focusField);
353
+ setInitialFocus(focusField);
354
+ }, [props.focus, form.setFocus, isLoading, initialFocus]);
355
+
356
+ const submitForm: SubmitHandler<FieldValues> = async (data) => {
357
+ setNonFieldErrors([]);
358
+
359
+ const method = props.method?.toLowerCase() ?? 'get';
360
+
361
+ let hasFiles = false;
362
+
363
+ // Optionally pre-process the data before submitting it
364
+ if (props.processFormData) {
365
+ data = props.processFormData(data);
366
+ }
367
+
368
+ const jsonData = { ...data };
369
+ const formData = new FormData();
370
+
371
+ Object.keys(data).forEach((key: string) => {
372
+ let value: any = data[key];
373
+ const field_type = fields[key]?.field_type;
374
+ const exclude = fields[key]?.exclude;
375
+
376
+ if (field_type == 'file upload' && !!value) {
377
+ hasFiles = true;
378
+ }
379
+
380
+ // Stringify any JSON objects
381
+ if (typeof value === 'object') {
382
+ switch (field_type) {
383
+ case 'file upload':
384
+ break;
385
+ default:
386
+ value = JSON.stringify(value);
387
+ break;
388
+ }
389
+ }
390
+
391
+ if (exclude) {
392
+ // Remove the field from the data
393
+ delete jsonData[key];
394
+ } else if (value != undefined) {
395
+ formData.append(key, value);
396
+ }
397
+ });
398
+
399
+ /* Set the timeout for the request:
400
+ * - If a timeout is provided in the props, use that
401
+ * - If the form contains files, use a longer timeout
402
+ * - Otherwise, use the default timeout
403
+ */
404
+ const timeout = props.timeout ?? (hasFiles ? 30000 : undefined);
405
+
406
+ return api({
407
+ method: method,
408
+ url: url,
409
+ params: method.toLowerCase() == 'get' ? jsonData : undefined,
410
+ data: hasFiles ? formData : jsonData,
411
+ timeout: timeout,
412
+ headers: {
413
+ 'Content-Type': hasFiles ? 'multipart/form-data' : 'application/json'
414
+ }
415
+ })
416
+ .then((response) => {
417
+ switch (response.status) {
418
+ case 200:
419
+ case 201:
420
+ case 204:
421
+ // Form was submitted successfully
422
+
423
+ if (props.onFormSuccess) {
424
+ // A custom callback hook is provided
425
+ props.onFormSuccess(response.data);
426
+ }
427
+
428
+ if (props.follow && props.modelType && response.data?.pk) {
429
+ // If we want to automatically follow the returned data
430
+ if (!!navigate) {
431
+ navigate(getDetailUrl(props.modelType, response.data?.pk));
432
+ }
433
+ } else if (props.table) {
434
+ // If we want to automatically update or reload a linked table
435
+ const pk_field = props.pk_field ?? 'pk';
436
+
437
+ if (props.pk && response?.data[pk_field]) {
438
+ props.table.updateRecord(response.data);
439
+ } else {
440
+ props.table.refreshTable();
441
+ }
442
+ }
443
+
444
+ // Optionally show a success message
445
+ if (props.successMessage) {
446
+ notifications.hide('form-success');
447
+
448
+ notifications.show({
449
+ id: 'form-success',
450
+ title: t`Success`,
451
+ message: props.successMessage,
452
+ color: 'green'
453
+ });
454
+ }
455
+
456
+ break;
457
+ default:
458
+ // Unexpected state on form success
459
+ invalidResponse(response.status);
460
+ props.onFormError?.(response);
461
+ break;
462
+ }
463
+
464
+ return response;
465
+ })
466
+ .catch((error) => {
467
+ if (error.response) {
468
+ switch (error.response.status) {
469
+ case 400:
470
+ // Data validation errors
471
+ const _nonFieldErrors: string[] = [];
472
+
473
+ const processErrors = (errors: any, _path?: string) => {
474
+ // Handle an array of errors
475
+ if (Array.isArray(errors)) {
476
+ errors.forEach((error: any) => {
477
+ _nonFieldErrors.push(error.toString());
478
+ });
479
+ return;
480
+ }
481
+
482
+ // Handle simple string
483
+ if (typeof errors === 'string') {
484
+ _nonFieldErrors.push(errors);
485
+ return;
486
+ }
487
+
488
+ for (const [k, v] of Object.entries(errors)) {
489
+ const path = _path ? `${_path}.${k}` : k;
490
+
491
+ // Determine if field "k" is valid (exists and is visible)
492
+ const field = fields[k];
493
+ const valid = field && !field.hidden;
494
+
495
+ if (!valid || k == 'non_field_errors' || k == '__all__') {
496
+ processErrors(v);
497
+ continue;
498
+ }
499
+
500
+ if (typeof v === 'object' && Array.isArray(v)) {
501
+ if (field?.field_type == 'table') {
502
+ // Special handling for "table" fields - they have nested errors
503
+ v.forEach((item: any, idx: number) => {
504
+ for (const [key, value] of Object.entries(item)) {
505
+ const path: string = `${k}.${idx}.${key}`;
506
+ if (Array.isArray(value)) {
507
+ form.setError(path, { message: value.join(', ') });
508
+ }
509
+ }
510
+ });
511
+ } else {
512
+ // Standard error handling for other fields
513
+ form.setError(path, { message: v.join(', ') });
514
+ }
515
+ } else if (typeof v === 'string') {
516
+ form.setError(path, { message: v });
517
+ } else {
518
+ processErrors(v, path);
519
+ }
520
+ }
521
+ };
522
+
523
+ processErrors(error.response.data);
524
+ setNonFieldErrors(_nonFieldErrors);
525
+ props.onFormError?.(error);
526
+
527
+ break;
528
+ default:
529
+ // Unexpected state on form error
530
+ invalidResponse(error.response.status);
531
+ props.onFormError?.(error);
532
+ break;
533
+ }
534
+ } else {
535
+ showTimeoutNotification();
536
+ props.onFormError?.(error);
537
+ }
538
+
539
+ return error;
540
+ });
541
+ };
542
+
543
+ const onFormError = useCallback<SubmitErrorHandler<FieldValues>>(
544
+ (error: any) => {
545
+ props.onFormError?.(error);
546
+ },
547
+ [props.onFormError]
548
+ );
549
+
550
+ if (optionsLoading || initialDataQuery.isFetching) {
551
+ return (
552
+ <Paper mah={'65vh'}>
553
+ <LoadingOverlay visible zIndex={1010} />
554
+ </Paper>
555
+ );
556
+ }
557
+
558
+ return (
559
+ <Stack>
560
+ <Boundary label={`ApiForm-${id}`}>
561
+ {/* Show loading overlay while fetching fields */}
562
+ {/* zIndex used to force overlay on top of modal header bar */}
563
+ <LoadingOverlay visible={isLoading} zIndex={1010} />
564
+
565
+ {/* Attempt at making fixed footer with scroll area */}
566
+ <Paper mah={'65vh'} style={{ overflowY: 'auto' }}>
567
+ <div>
568
+ {/* Form Fields */}
569
+ <Stack gap='sm'>
570
+ {(!isValid || nonFieldErrors.length > 0) && (
571
+ <Alert radius='sm' color='red' title={t`Form Error`}>
572
+ {nonFieldErrors.length > 0 ? (
573
+ <Stack gap='xs'>
574
+ {nonFieldErrors.map((message) => (
575
+ <Text key={message}>{message}</Text>
576
+ ))}
577
+ </Stack>
578
+ ) : (
579
+ <Text>{t`Errors exist for one or more form fields`}</Text>
580
+ )}
581
+ </Alert>
582
+ )}
583
+ <Boundary label={`ApiForm-${id}-PreFormContent`}>
584
+ {props.preFormContent}
585
+ {props.preFormSuccess && (
586
+ <Alert color='green' radius='sm'>
587
+ {props.preFormSuccess}
588
+ </Alert>
589
+ )}
590
+ {props.preFormWarning && (
591
+ <Alert color='orange' radius='sm'>
592
+ {props.preFormWarning}
593
+ </Alert>
594
+ )}
595
+ </Boundary>
596
+ <Boundary label={`ApiForm-${id}-FormContent`}>
597
+ <FormProvider {...form}>
598
+ <Stack gap='xs'>
599
+ {Object.entries(fields).map(([fieldName, field]) => {
600
+ return (
601
+ <ApiFormField
602
+ key={fieldName}
603
+ fieldName={fieldName}
604
+ definition={field}
605
+ control={form.control}
606
+ url={url}
607
+ setFields={setFields}
608
+ onKeyDown={(value) => {
609
+ if (
610
+ value == 'Enter' &&
611
+ !isLoading &&
612
+ (!props.fetchInitialData || isDirty)
613
+ ) {
614
+ form.handleSubmit(submitForm, onFormError)();
615
+ }
616
+ }}
617
+ />
618
+ );
619
+ })}
620
+ </Stack>
621
+ </FormProvider>
622
+ </Boundary>
623
+ <Boundary label={`ApiForm-${id}-PostFormContent`}>
624
+ {props.postFormContent}
625
+ </Boundary>
626
+ </Stack>
627
+ </div>
628
+ </Paper>
629
+
630
+ {/* Footer with Action Buttons */}
631
+ <Divider />
632
+ <div>
633
+ <Group justify='right'>
634
+ {props.actions?.map((action, i) => (
635
+ <Button
636
+ key={`${i}-${action.text}`}
637
+ onClick={action.onClick}
638
+ variant={action.variant ?? 'outline'}
639
+ radius='sm'
640
+ color={action.color}
641
+ >
642
+ {action.text}
643
+ </Button>
644
+ ))}
645
+ <Button
646
+ onClick={form.handleSubmit(submitForm, onFormError)}
647
+ variant='filled'
648
+ radius='sm'
649
+ color={props.submitColor ?? 'green'}
650
+ disabled={isLoading || (props.fetchInitialData && !isDirty)}
651
+ >
652
+ {props.submitText ?? t`Submit`}
653
+ </Button>
654
+ </Group>
655
+ </div>
656
+ </Boundary>
657
+ </Stack>
658
+ );
659
+ }
660
+
661
+ export function CreateApiForm({
662
+ id,
663
+ props
664
+ }: Readonly<{
665
+ id?: string;
666
+ props: ApiFormProps;
667
+ }>) {
668
+ const createProps = useMemo<ApiFormProps>(
669
+ () => ({
670
+ ...props,
671
+ method: 'POST'
672
+ }),
673
+ [props]
674
+ );
675
+
676
+ return <OptionsApiForm props={createProps} id={id} />;
677
+ }
678
+
679
+ export function EditApiForm({
680
+ id,
681
+ props
682
+ }: Readonly<{
683
+ id?: string;
684
+ props: ApiFormProps;
685
+ }>) {
686
+ const editProps = useMemo<ApiFormProps>(
687
+ () => ({
688
+ ...props,
689
+ fetchInitialData: props.fetchInitialData ?? true,
690
+ submitText: props.submitText ?? t`Update`,
691
+ method: 'PUT'
692
+ }),
693
+ [props]
694
+ );
695
+
696
+ return <OptionsApiForm props={editProps} id={id} />;
697
+ }
698
+
699
+ export function DeleteApiForm({
700
+ id,
701
+ props
702
+ }: Readonly<{
703
+ id?: string;
704
+ props: ApiFormProps;
705
+ }>) {
706
+ const deleteProps = useMemo<ApiFormProps>(
707
+ () => ({
708
+ ...props,
709
+ method: 'DELETE',
710
+ submitText: t`Delete`,
711
+ submitColor: 'red',
712
+ fields: {}
713
+ }),
714
+ [props]
715
+ );
716
+
717
+ return <OptionsApiForm props={deleteProps} id={id} />;
718
+ }