@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,750 @@
1
+ import { t } from '@lingui/core/macro';
2
+ import { Box, Stack } from '@mantine/core';
3
+ import { useQuery } from '@tanstack/react-query';
4
+ import { useContextMenu } from 'mantine-contextmenu';
5
+ import {
6
+ DataTable,
7
+ type DataTableCellClickHandler,
8
+ type DataTableRowExpansionProps,
9
+ type DataTableSortStatus
10
+ } from 'mantine-datatable';
11
+ import type React from 'react';
12
+ import { useCallback, useEffect, useMemo, useState } from 'react';
13
+ import { useNavigate } from 'react-router-dom';
14
+
15
+ import type { ModelType } from '@lib/enums/ModelType';
16
+ import type { TableFilter } from '@lib/types/Filters';
17
+ import type { ApiFormFieldSet } from '@lib/types/Forms';
18
+ import type { TableState } from '@lib/types/Tables';
19
+ import { hideNotification, showNotification } from '@mantine/notifications';
20
+ import { Boundary } from '../components/Boundary';
21
+ import { useApi } from '../contexts/ApiContext';
22
+ import { resolveItem } from '../functions/conversion';
23
+ import { cancelEvent } from '../functions/events';
24
+ import { extractAvailableFields, mapFields } from '../functions/forms';
25
+ import { navigateToLink } from '../functions/navigation';
26
+ import { getDetailUrl } from '../functions/urls';
27
+ import { useLocalState } from '../states/LocalState';
28
+ import type { TableColumn } from './Column';
29
+ import InvenTreeTableHeader from './InvenTreeTableHeader';
30
+ import { type RowAction, RowActions } from './RowActions';
31
+
32
+ const defaultPageSize: number = 25;
33
+ const PAGE_SIZES = [10, 15, 20, 25, 50, 100, 500];
34
+
35
+ /**
36
+ * Set of optional properties which can be passed to an InvenTreeTable component
37
+ *
38
+ * @param params : any - Base query parameters
39
+ * @param tableState : TableState - State manager for the table
40
+ * @param defaultSortColumn : string - Default column to sort by
41
+ * @param noRecordsText : string - Text to display when no records are found
42
+ * @param enableBulkDelete : boolean - Enable bulk deletion of records
43
+ * @param enableDownload : boolean - Enable download actions
44
+ * @param enableFilters : boolean - Enable filter actions
45
+ * @param enableSelection : boolean - Enable row selection
46
+ * @param enableSearch : boolean - Enable search actions
47
+ * @param enableLabels : boolean - Enable printing of labels against selected items
48
+ * @param enableReports : boolean - Enable printing of reports against selected items
49
+ * @param enablePagination : boolean - Enable pagination
50
+ * @param enableRefresh : boolean - Enable refresh actions
51
+ * @param enableColumnSwitching : boolean - Enable column switching
52
+ * @param enableColumnCaching : boolean - Enable caching of column names via API
53
+ * @param barcodeActions : any[] - List of barcode actions
54
+ * @param tableFilters : TableFilter[] - List of custom filters
55
+ * @param tableActions : any[] - List of custom action groups
56
+ * @param dataFormatter : (data: any) => any - Callback function to reformat data returned by server (if not in default format)
57
+ * @param rowActions : (record: any) => RowAction[] - Callback function to generate row actions
58
+ * @param onRowClick : (record: any, index: number, event: any) => void - Callback function when a row is clicked
59
+ * @param onCellClick : (event: any, record: any, index: number, column: any, columnIndex: number) => void - Callback function when a cell is clicked
60
+ * @param modelType: ModelType - The model type for the table
61
+ * @param minHeight: number - Minimum height of the table (default 300px)
62
+ * @param noHeader: boolean - Hide the table header
63
+ */
64
+ export type InvenTreeTableProps<T = any> = {
65
+ params?: any;
66
+ defaultSortColumn?: string;
67
+ noRecordsText?: string;
68
+ enableBulkDelete?: boolean;
69
+ enableDownload?: boolean;
70
+ enableFilters?: boolean;
71
+ enableSelection?: boolean;
72
+ enableSearch?: boolean;
73
+ enablePagination?: boolean;
74
+ enableRefresh?: boolean;
75
+ enableColumnSwitching?: boolean;
76
+ enableColumnCaching?: boolean;
77
+ enableLabels?: boolean;
78
+ enableReports?: boolean;
79
+ afterBulkDelete?: () => void;
80
+ barcodeActions?: React.ReactNode[];
81
+ tableFilters?: TableFilter[];
82
+ tableActions?: React.ReactNode[];
83
+ rowExpansion?: DataTableRowExpansionProps<T>;
84
+ dataFormatter?: (data: any) => any;
85
+ rowActions?: (record: T) => RowAction[];
86
+ onRowClick?: (record: T, index: number, event: any) => void;
87
+ onCellClick?: DataTableCellClickHandler<T>;
88
+ modelType?: ModelType;
89
+ rowStyle?: (record: T, index: number) => any;
90
+ modelField?: string;
91
+ onCellContextMenu?: (record: T, event: any) => void;
92
+ minHeight?: number;
93
+ noHeader?: boolean;
94
+ };
95
+
96
+ /**
97
+ * Default table properties (used if not specified)
98
+ */
99
+ const defaultInvenTreeTableProps: InvenTreeTableProps = {
100
+ params: {},
101
+ noRecordsText: t`No records found`,
102
+ enableDownload: false,
103
+ enableLabels: false,
104
+ enableReports: false,
105
+ enableFilters: true,
106
+ enablePagination: true,
107
+ enableRefresh: true,
108
+ enableSearch: true,
109
+ enableSelection: false,
110
+ defaultSortColumn: '',
111
+ barcodeActions: [],
112
+ tableFilters: [],
113
+ tableActions: []
114
+ };
115
+
116
+ /**
117
+ * Table Component which extends DataTable with custom InvenTree functionality
118
+ */
119
+ export function InvenTreeTable<T extends Record<string, any>>({
120
+ url,
121
+ tableState,
122
+ tableData,
123
+ columns,
124
+ props
125
+ }: Readonly<{
126
+ url?: string;
127
+ tableState: TableState;
128
+ tableData?: any[];
129
+ columns: TableColumn<T>[];
130
+ props: InvenTreeTableProps<T>;
131
+ }>) {
132
+ const {
133
+ getTableColumnNames,
134
+ setTableColumnNames,
135
+ getTableSorting,
136
+ setTableSorting,
137
+ userTheme
138
+ } = useLocalState();
139
+
140
+ const [fieldNames, setFieldNames] = useState<Record<string, string>>({});
141
+
142
+ const api = useApi();
143
+ const navigate = useNavigate();
144
+ const { showContextMenu } = useContextMenu();
145
+
146
+ // Construct table filters - note that we can introspect filter labels from column names
147
+ const filters: TableFilter[] = useMemo(() => {
148
+ return (
149
+ props.tableFilters
150
+ ?.filter((f: any) => f.active != false)
151
+ ?.map((filter) => {
152
+ return {
153
+ ...filter,
154
+ label: filter.label ?? fieldNames[filter.name] ?? `${filter.name}`
155
+ };
156
+ }) ?? []
157
+ );
158
+ }, [props.tableFilters, fieldNames]);
159
+
160
+ // Request OPTIONS data from the API, before we load the table
161
+ const tableOptionQuery = useQuery({
162
+ enabled: !!url && !tableData,
163
+ queryKey: ['options', url, tableState.tableKey, props.enableColumnCaching],
164
+ retry: 3,
165
+ refetchOnMount: true,
166
+ gcTime: 5000,
167
+ queryFn: async () => {
168
+ if (!url) {
169
+ return null;
170
+ }
171
+
172
+ if (props.enableColumnCaching == false) {
173
+ return null;
174
+ }
175
+
176
+ // If we already have field names, no need to fetch them again
177
+ if (fieldNames && Object.keys(fieldNames).length > 0) {
178
+ return null;
179
+ }
180
+
181
+ return api
182
+ .options(url, {
183
+ params: tableProps.params
184
+ })
185
+ .then((response) => {
186
+ if (response.status == 200) {
187
+ // Extract field information from the API
188
+
189
+ const names: Record<string, string> = {};
190
+
191
+ const fields: ApiFormFieldSet =
192
+ extractAvailableFields(response, 'GET', true) || {};
193
+
194
+ // Extract flattened map of fields
195
+ mapFields(fields, (path, field) => {
196
+ if (field.label) {
197
+ names[path] = field.label;
198
+ }
199
+ });
200
+
201
+ const cacheKey = tableState.tableKey.replaceAll('-', '');
202
+
203
+ setFieldNames(names);
204
+ setTableColumnNames(cacheKey)(names);
205
+ }
206
+
207
+ return null;
208
+ })
209
+ .catch(() => {
210
+ hideNotification('table-options-error');
211
+ showNotification({
212
+ id: 'table-options-error',
213
+ title: t`API Error`,
214
+ message: t`Failed to load table options`,
215
+ color: 'red'
216
+ });
217
+
218
+ return null;
219
+ });
220
+ }
221
+ });
222
+
223
+ // Rebuild set of translated column names
224
+ useEffect(() => {
225
+ if (props.enableColumnCaching == false) {
226
+ return;
227
+ }
228
+
229
+ const cacheKey = tableState.tableKey.replaceAll('-', '');
230
+
231
+ // First check the local cache
232
+ const cachedNames = getTableColumnNames(cacheKey);
233
+
234
+ if (cachedNames != null) {
235
+ // Cached names are available - use them!
236
+ setFieldNames(cachedNames);
237
+ return;
238
+ }
239
+
240
+ tableOptionQuery.refetch();
241
+ }, [url, props.params, props.enableColumnCaching]);
242
+
243
+ // Build table properties based on provided props (and default props)
244
+ const tableProps: InvenTreeTableProps<T> = useMemo(() => {
245
+ return {
246
+ ...defaultInvenTreeTableProps,
247
+ ...props
248
+ };
249
+ }, [props]);
250
+
251
+ const enableSelection: boolean = useMemo(() => {
252
+ return tableProps.enableSelection || tableProps.enableBulkDelete || false;
253
+ }, [tableProps]);
254
+
255
+ // Check if any columns are switchable (can be hidden)
256
+ const hasSwitchableColumns: boolean = useMemo(() => {
257
+ if (props.enableColumnSwitching == false) {
258
+ return false;
259
+ } else {
260
+ return columns.some((col: TableColumn) => {
261
+ if (col.hidden == true) {
262
+ // Not a switchable column - is hidden
263
+ return false;
264
+ } else if (col.switchable == false) {
265
+ return false;
266
+ } else {
267
+ return true;
268
+ }
269
+ });
270
+ }
271
+ }, [columns, props.enableColumnSwitching]);
272
+
273
+ const onSelectedRecordsChange = useCallback(
274
+ (records: any[]) => {
275
+ tableState.setSelectedRecords(records);
276
+ },
277
+ [tableState.setSelectedRecords]
278
+ );
279
+
280
+ // Update column visibility when hiddenColumns change
281
+ const dataColumns: any = useMemo(() => {
282
+ const cols: TableColumn[] = columns
283
+ .filter((col) => col?.hidden != true)
284
+ .map((col) => {
285
+ let hidden: boolean = col.hidden ?? false;
286
+
287
+ if (col.switchable ?? true) {
288
+ hidden = tableState.hiddenColumns.includes(col.accessor);
289
+ }
290
+
291
+ return {
292
+ ...col,
293
+ hidden: hidden,
294
+ noWrap: true,
295
+ title: col.title ?? fieldNames[col.accessor] ?? `${col.accessor}`
296
+ };
297
+ });
298
+
299
+ // If row actions are available, add a column for them
300
+ if (tableProps.rowActions) {
301
+ cols.push({
302
+ accessor: '--actions--',
303
+ title: ' ',
304
+ hidden: false,
305
+ switchable: false,
306
+ width: 50,
307
+ render: (record: any, index?: number | undefined) => (
308
+ <RowActions
309
+ actions={tableProps.rowActions?.(record) ?? []}
310
+ disabled={tableState.selectedRecords.length > 0}
311
+ index={index}
312
+ />
313
+ )
314
+ });
315
+ }
316
+
317
+ return cols;
318
+ }, [
319
+ columns,
320
+ fieldNames,
321
+ tableProps.rowActions,
322
+ tableState.hiddenColumns,
323
+ tableState.selectedRecords
324
+ ]);
325
+
326
+ // Callback when column visibility is toggled
327
+ function toggleColumn(columnName: string) {
328
+ const newColumns = [...dataColumns];
329
+
330
+ const colIdx = newColumns.findIndex((col) => col.accessor == columnName);
331
+
332
+ if (colIdx >= 0 && colIdx < newColumns.length) {
333
+ newColumns[colIdx].hidden = !newColumns[colIdx].hidden;
334
+ }
335
+
336
+ tableState.setHiddenColumns(
337
+ newColumns.filter((col) => col.hidden).map((col) => col.accessor)
338
+ );
339
+ }
340
+
341
+ // Reset the pagination state when the search term changes
342
+ useEffect(() => {
343
+ tableState.setPage(1);
344
+ }, [tableState.searchTerm]);
345
+
346
+ // Data Sorting
347
+ const [sortStatus, setSortStatus] = useState<DataTableSortStatus<T>>({
348
+ columnAccessor: tableProps.defaultSortColumn ?? '',
349
+ direction: 'asc'
350
+ });
351
+
352
+ /*
353
+ * Construct query filters for the current table
354
+ */
355
+ const getTableFilters = useCallback(
356
+ (paginate = false) => {
357
+ const queryParams = {
358
+ ...tableProps.params
359
+ };
360
+
361
+ // Add custom filters
362
+ if (tableState.filterSet.activeFilters) {
363
+ tableState.filterSet.activeFilters.forEach((flt) => {
364
+ queryParams[flt.name] = flt.value;
365
+ });
366
+ }
367
+
368
+ // Allow override of filters based on URL query parameters
369
+ if (tableState.queryFilters) {
370
+ for (const [key, value] of tableState.queryFilters) {
371
+ queryParams[key] = value;
372
+ }
373
+ }
374
+
375
+ // Add custom search term
376
+ if (tableState.searchTerm) {
377
+ queryParams.search = tableState.searchTerm;
378
+ }
379
+
380
+ // Pagination
381
+ if (tableProps.enablePagination && paginate) {
382
+ const pageSize = tableState.pageSize ?? defaultPageSize;
383
+ if (pageSize != tableState.pageSize) tableState.setPageSize(pageSize);
384
+ queryParams.limit = pageSize;
385
+ queryParams.offset = (tableState.page - 1) * pageSize;
386
+ }
387
+
388
+ // Ordering
389
+ const ordering = getOrderingTerm();
390
+
391
+ if (ordering) {
392
+ if (sortStatus.direction == 'asc') {
393
+ queryParams.ordering = ordering;
394
+ } else {
395
+ queryParams.ordering = `-${ordering}`;
396
+ }
397
+ }
398
+
399
+ return queryParams;
400
+ },
401
+ [
402
+ tableProps.params,
403
+ tableProps.enablePagination,
404
+ tableState.filterSet.activeFilters,
405
+ tableState.queryFilters,
406
+ tableState.searchTerm,
407
+ tableState.pageSize,
408
+ tableState.setPageSize,
409
+ sortStatus,
410
+ getOrderingTerm
411
+ ]
412
+ );
413
+
414
+ useEffect(() => {
415
+ const tableKey: string = tableState.tableKey.split('-')[0];
416
+ const sorting: DataTableSortStatus = getTableSorting(tableKey);
417
+
418
+ if (sorting) {
419
+ setSortStatus(sorting);
420
+ }
421
+ }, []);
422
+
423
+ // Return the ordering parameter
424
+ function getOrderingTerm() {
425
+ const key = sortStatus.columnAccessor;
426
+
427
+ // Sorting column not specified
428
+ if (key == '') {
429
+ return '';
430
+ }
431
+
432
+ // Find matching column:
433
+ // If column provides custom ordering term, use that
434
+ const column = dataColumns.find((col: any) => col.accessor == key);
435
+ return column?.ordering || key;
436
+ }
437
+
438
+ // Missing records text (based on server response)
439
+ const [missingRecordsText, setMissingRecordsText] = useState<string>(
440
+ tableProps.noRecordsText ?? t`No records found`
441
+ );
442
+
443
+ const handleSortStatusChange = (status: DataTableSortStatus<T>) => {
444
+ tableState.setPage(1);
445
+ setSortStatus(status);
446
+
447
+ const tableKey = tableState.tableKey.split('-')[0];
448
+ setTableSorting(tableKey)(status);
449
+ };
450
+
451
+ // Function to perform API query to fetch required data
452
+ const fetchTableData = async () => {
453
+ const queryParams = getTableFilters(true);
454
+
455
+ if (!url) {
456
+ return [];
457
+ }
458
+
459
+ return api
460
+ .get(url, {
461
+ params: queryParams,
462
+ timeout: 5 * 1000
463
+ })
464
+ .then((response) => {
465
+ switch (response.status) {
466
+ case 200:
467
+ setMissingRecordsText(
468
+ tableProps.noRecordsText ?? t`No records found`
469
+ );
470
+
471
+ let results = response.data?.results ?? response.data ?? [];
472
+
473
+ if (props.dataFormatter) {
474
+ // Custom data formatter provided
475
+ results = props.dataFormatter(results);
476
+ }
477
+
478
+ if (!Array.isArray(results)) {
479
+ setMissingRecordsText(t`Server returned incorrect data type`);
480
+ results = [];
481
+ }
482
+
483
+ tableState.setRecordCount(response.data?.count ?? results.length);
484
+
485
+ return results;
486
+ case 400:
487
+ setMissingRecordsText(t`Bad request`);
488
+ break;
489
+ case 401:
490
+ setMissingRecordsText(t`Unauthorized`);
491
+ break;
492
+ case 403:
493
+ setMissingRecordsText(t`Forbidden`);
494
+ break;
495
+ case 404:
496
+ setMissingRecordsText(t`Not found`);
497
+ break;
498
+ default:
499
+ setMissingRecordsText(
500
+ `${t`Unknown error`}: ${response.statusText}`
501
+ );
502
+ break;
503
+ }
504
+
505
+ return [];
506
+ })
507
+ .catch((error) => {
508
+ setMissingRecordsText(`${t`Error`}: ${error.message}`);
509
+ return [];
510
+ });
511
+ };
512
+
513
+ const {
514
+ data: apiData,
515
+ isFetching,
516
+ isLoading,
517
+ refetch
518
+ } = useQuery({
519
+ queryKey: [
520
+ 'tabledata',
521
+ url,
522
+ tableState.page,
523
+ props.params,
524
+ sortStatus.columnAccessor,
525
+ sortStatus.direction,
526
+ tableState.tableKey,
527
+ tableState.filterSet.activeFilters,
528
+ tableState.searchTerm
529
+ ],
530
+ enabled: !!url && !tableData,
531
+ queryFn: fetchTableData,
532
+ refetchOnMount: true
533
+ });
534
+
535
+ // Refetch data when the query parameters change
536
+ useEffect(() => {
537
+ refetch();
538
+ }, [tableState.queryFilters]);
539
+
540
+ useEffect(() => {
541
+ tableState.setIsLoading(
542
+ isFetching ||
543
+ isLoading ||
544
+ tableOptionQuery.isFetching ||
545
+ tableOptionQuery.isLoading
546
+ );
547
+ }, [isFetching, isLoading, tableOptionQuery]);
548
+
549
+ // Update tableState.records when new data received
550
+ useEffect(() => {
551
+ const data = tableData ?? apiData ?? [];
552
+
553
+ tableState.setRecords(data);
554
+
555
+ // set pagesize to length if pagination is disabled
556
+ if (!tableProps.enablePagination) {
557
+ tableState.setPageSize(data?.length ?? defaultPageSize);
558
+ }
559
+ }, [tableData, apiData]);
560
+
561
+ // Callback when a cell is clicked
562
+ const handleCellClick = useCallback(
563
+ ({
564
+ event,
565
+ record,
566
+ index,
567
+ column,
568
+ columnIndex
569
+ }: {
570
+ event: React.MouseEvent;
571
+ record: any;
572
+ index: number;
573
+ column: any;
574
+ columnIndex: number;
575
+ }) => {
576
+ // Ignore any click on the 'actions' column
577
+ if (column.accessor == '--actions--') {
578
+ return;
579
+ }
580
+
581
+ if (props.onCellClick) {
582
+ props.onCellClick({ event, record, index, column, columnIndex });
583
+ } else if (props.onRowClick) {
584
+ props.onRowClick(record, index, event);
585
+ } else if (tableProps.modelType) {
586
+ const accessor = tableProps.modelField ?? 'pk';
587
+ const pk = resolveItem(record, accessor);
588
+
589
+ if (pk) {
590
+ cancelEvent(event);
591
+ // If a model type is provided, navigate to the detail view for that model
592
+ const url = getDetailUrl(tableProps.modelType, pk);
593
+ navigateToLink(url, navigate, event);
594
+ }
595
+ }
596
+ },
597
+ [props.onRowClick, props.onCellClick]
598
+ );
599
+
600
+ // Callback when a cell is right-clicked
601
+ const handleCellContextMenu = ({
602
+ record,
603
+ column,
604
+ event
605
+ }: {
606
+ record: any;
607
+ column: any;
608
+ event: any;
609
+ }) => {
610
+ if (column?.noContext === true) {
611
+ return;
612
+ }
613
+ if (props.onCellContextMenu) {
614
+ return props.onCellContextMenu(record, event);
615
+ } else if (props.rowActions) {
616
+ const empty = () => {};
617
+ const items = props.rowActions(record).map((action) => ({
618
+ key: action.title ?? '',
619
+ title: action.title ?? '',
620
+ color: action.color,
621
+ icon: action.icon,
622
+ onClick: action.onClick ?? empty,
623
+ hidden: action.hidden,
624
+ disabled: action.disabled
625
+ }));
626
+ return showContextMenu(items)(event);
627
+ } else {
628
+ return showContextMenu([])(event);
629
+ }
630
+ };
631
+
632
+ // pagination refresh table if pageSize changes
633
+ function updatePageSize(newData: number) {
634
+ tableState.setPageSize(newData);
635
+ tableState.setPage(1);
636
+ tableState.refreshTable();
637
+ }
638
+
639
+ /**
640
+ * Memoize row expansion options:
641
+ * - If rowExpansion is not provided, return undefined
642
+ * - Otherwise, return the rowExpansion object
643
+ * - Utilize the useTable hook to track expanded rows
644
+ */
645
+ const rowExpansion: DataTableRowExpansionProps<T> | undefined =
646
+ useMemo(() => {
647
+ if (!props.rowExpansion) {
648
+ return undefined;
649
+ }
650
+
651
+ return {
652
+ ...props.rowExpansion,
653
+ expanded: {
654
+ recordIds: tableState.expandedRecords,
655
+ onRecordIdsChange: (ids: any[]) => {
656
+ tableState.setExpandedRecords(ids);
657
+ }
658
+ }
659
+ };
660
+ }, [
661
+ tableState.expandedRecords,
662
+ tableState.setExpandedRecords,
663
+ props.rowExpansion
664
+ ]);
665
+
666
+ const optionalParams = useMemo(() => {
667
+ let _params: Record<string, any> = {};
668
+
669
+ if (tableProps.enablePagination) {
670
+ _params = {
671
+ ..._params,
672
+ totalRecords: tableState.recordCount,
673
+ recordsPerPage: tableState.pageSize,
674
+ page: tableState.page,
675
+ onPageChange: tableState.setPage,
676
+ recordsPerPageOptions: PAGE_SIZES,
677
+ onRecordsPerPageChange: updatePageSize
678
+ };
679
+ }
680
+
681
+ return _params;
682
+ }, [
683
+ tableProps.enablePagination,
684
+ tableState.recordCount,
685
+ tableState.pageSize,
686
+ tableState.page,
687
+ tableState.setPage,
688
+ updatePageSize
689
+ ]);
690
+
691
+ return (
692
+ <>
693
+ <Stack gap='xs'>
694
+ {!tableProps.noHeader && (
695
+ <Boundary label={`InvenTreeTableHeader-${tableState.tableKey}`}>
696
+ <InvenTreeTableHeader
697
+ tableUrl={url}
698
+ tableState={tableState}
699
+ tableProps={tableProps}
700
+ hasSwitchableColumns={hasSwitchableColumns}
701
+ columns={dataColumns}
702
+ filters={filters}
703
+ toggleColumn={toggleColumn}
704
+ />
705
+ </Boundary>
706
+ )}
707
+ <Boundary label={`InvenTreeTable-${tableState.tableKey}`}>
708
+ <Box pos='relative'>
709
+ <DataTable
710
+ withTableBorder={!tableProps.noHeader}
711
+ withColumnBorders
712
+ striped
713
+ highlightOnHover
714
+ loaderType={userTheme.loader}
715
+ pinLastColumn={tableProps.rowActions != undefined}
716
+ idAccessor={tableState.idAccessor ?? 'pk'}
717
+ minHeight={tableProps.minHeight ?? 300}
718
+ sortStatus={sortStatus}
719
+ onSortStatusChange={handleSortStatusChange}
720
+ selectedRecords={
721
+ enableSelection ? tableState.selectedRecords : undefined
722
+ }
723
+ onSelectedRecordsChange={
724
+ enableSelection ? onSelectedRecordsChange : undefined
725
+ }
726
+ rowExpansion={rowExpansion}
727
+ rowStyle={tableProps.rowStyle}
728
+ fetching={isFetching}
729
+ noRecordsText={missingRecordsText}
730
+ records={tableState.records}
731
+ columns={dataColumns}
732
+ onCellClick={handleCellClick}
733
+ noHeader={tableProps.noHeader ?? false}
734
+ defaultColumnProps={{
735
+ noWrap: true,
736
+ textAlign: 'left',
737
+ cellsStyle: () => (theme) => ({
738
+ // TODO @SchrodingersGat : Need a better way of handling "wide" cells,
739
+ overflow: 'hidden'
740
+ })
741
+ }}
742
+ onCellContextMenu={handleCellContextMenu}
743
+ {...optionalParams}
744
+ />
745
+ </Box>
746
+ </Boundary>
747
+ </Stack>
748
+ </>
749
+ );
750
+ }