@novastera-oss/material-react-table 4.0.0-beta.11

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 (324) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +203 -0
  3. package/dist/index.cjs +5207 -0
  4. package/dist/index.cjs.map +1 -0
  5. package/dist/index.d.ts +1871 -0
  6. package/dist/index.js +5101 -0
  7. package/dist/index.js.map +1 -0
  8. package/locales/ar/index.d.ts +3 -0
  9. package/locales/ar/index.esm.d.ts +3 -0
  10. package/locales/ar/index.esm.js +96 -0
  11. package/locales/ar/index.js +98 -0
  12. package/locales/ar/package.json +6 -0
  13. package/locales/az/index.d.ts +3 -0
  14. package/locales/az/index.esm.d.ts +3 -0
  15. package/locales/az/index.esm.js +96 -0
  16. package/locales/az/index.js +98 -0
  17. package/locales/az/package.json +6 -0
  18. package/locales/bg/index.d.ts +3 -0
  19. package/locales/bg/index.esm.d.ts +3 -0
  20. package/locales/bg/index.esm.js +96 -0
  21. package/locales/bg/index.js +98 -0
  22. package/locales/bg/package.json +6 -0
  23. package/locales/cs/index.d.ts +3 -0
  24. package/locales/cs/index.esm.d.ts +3 -0
  25. package/locales/cs/index.esm.js +96 -0
  26. package/locales/cs/index.js +98 -0
  27. package/locales/cs/package.json +6 -0
  28. package/locales/da/index.d.ts +3 -0
  29. package/locales/da/index.esm.d.ts +3 -0
  30. package/locales/da/index.esm.js +96 -0
  31. package/locales/da/index.js +98 -0
  32. package/locales/da/package.json +6 -0
  33. package/locales/de/index.d.ts +3 -0
  34. package/locales/de/index.esm.d.ts +3 -0
  35. package/locales/de/index.esm.js +96 -0
  36. package/locales/de/index.js +98 -0
  37. package/locales/de/package.json +6 -0
  38. package/locales/el/index.d.ts +3 -0
  39. package/locales/el/index.esm.d.ts +3 -0
  40. package/locales/el/index.esm.js +96 -0
  41. package/locales/el/index.js +98 -0
  42. package/locales/el/package.json +6 -0
  43. package/locales/en/index.d.ts +3 -0
  44. package/locales/en/index.esm.d.ts +3 -0
  45. package/locales/en/index.esm.js +96 -0
  46. package/locales/en/index.js +98 -0
  47. package/locales/en/package.json +6 -0
  48. package/locales/es/index.d.ts +3 -0
  49. package/locales/es/index.esm.d.ts +3 -0
  50. package/locales/es/index.esm.js +96 -0
  51. package/locales/es/index.js +98 -0
  52. package/locales/es/package.json +6 -0
  53. package/locales/et/index.d.ts +3 -0
  54. package/locales/et/index.esm.d.ts +3 -0
  55. package/locales/et/index.esm.js +96 -0
  56. package/locales/et/index.js +98 -0
  57. package/locales/et/package.json +6 -0
  58. package/locales/fa/index.d.ts +3 -0
  59. package/locales/fa/index.esm.d.ts +3 -0
  60. package/locales/fa/index.esm.js +96 -0
  61. package/locales/fa/index.js +98 -0
  62. package/locales/fa/package.json +6 -0
  63. package/locales/fi/index.d.ts +3 -0
  64. package/locales/fi/index.esm.d.ts +3 -0
  65. package/locales/fi/index.esm.js +96 -0
  66. package/locales/fi/index.js +98 -0
  67. package/locales/fi/package.json +6 -0
  68. package/locales/fr/index.d.ts +3 -0
  69. package/locales/fr/index.esm.d.ts +3 -0
  70. package/locales/fr/index.esm.js +96 -0
  71. package/locales/fr/index.js +98 -0
  72. package/locales/fr/package.json +6 -0
  73. package/locales/he/index.d.ts +3 -0
  74. package/locales/he/index.esm.d.ts +3 -0
  75. package/locales/he/index.esm.js +96 -0
  76. package/locales/he/index.js +98 -0
  77. package/locales/he/package.json +6 -0
  78. package/locales/hr/index.d.ts +3 -0
  79. package/locales/hr/index.esm.d.ts +3 -0
  80. package/locales/hr/index.esm.js +96 -0
  81. package/locales/hr/index.js +98 -0
  82. package/locales/hr/package.json +6 -0
  83. package/locales/hu/index.d.ts +3 -0
  84. package/locales/hu/index.esm.d.ts +3 -0
  85. package/locales/hu/index.esm.js +96 -0
  86. package/locales/hu/index.js +98 -0
  87. package/locales/hu/package.json +6 -0
  88. package/locales/hy/index.d.ts +3 -0
  89. package/locales/hy/index.esm.d.ts +3 -0
  90. package/locales/hy/index.esm.js +96 -0
  91. package/locales/hy/index.js +98 -0
  92. package/locales/hy/package.json +6 -0
  93. package/locales/id/index.d.ts +3 -0
  94. package/locales/id/index.esm.d.ts +3 -0
  95. package/locales/id/index.esm.js +96 -0
  96. package/locales/id/index.js +98 -0
  97. package/locales/id/package.json +6 -0
  98. package/locales/it/index.d.ts +3 -0
  99. package/locales/it/index.esm.d.ts +3 -0
  100. package/locales/it/index.esm.js +96 -0
  101. package/locales/it/index.js +98 -0
  102. package/locales/it/package.json +6 -0
  103. package/locales/ja/index.d.ts +3 -0
  104. package/locales/ja/index.esm.d.ts +3 -0
  105. package/locales/ja/index.esm.js +96 -0
  106. package/locales/ja/index.js +98 -0
  107. package/locales/ja/package.json +6 -0
  108. package/locales/ko/index.d.ts +3 -0
  109. package/locales/ko/index.esm.d.ts +3 -0
  110. package/locales/ko/index.esm.js +96 -0
  111. package/locales/ko/index.js +98 -0
  112. package/locales/ko/package.json +6 -0
  113. package/locales/nl/index.d.ts +3 -0
  114. package/locales/nl/index.esm.d.ts +3 -0
  115. package/locales/nl/index.esm.js +96 -0
  116. package/locales/nl/index.js +98 -0
  117. package/locales/nl/package.json +6 -0
  118. package/locales/no/index.d.ts +3 -0
  119. package/locales/no/index.esm.d.ts +3 -0
  120. package/locales/no/index.esm.js +96 -0
  121. package/locales/no/index.js +98 -0
  122. package/locales/no/package.json +6 -0
  123. package/locales/np/index.d.ts +3 -0
  124. package/locales/np/index.esm.d.ts +3 -0
  125. package/locales/np/index.esm.js +96 -0
  126. package/locales/np/index.js +98 -0
  127. package/locales/np/package.json +6 -0
  128. package/locales/pl/index.d.ts +3 -0
  129. package/locales/pl/index.esm.d.ts +3 -0
  130. package/locales/pl/index.esm.js +96 -0
  131. package/locales/pl/index.js +98 -0
  132. package/locales/pl/package.json +6 -0
  133. package/locales/pt/index.d.ts +3 -0
  134. package/locales/pt/index.esm.d.ts +3 -0
  135. package/locales/pt/index.esm.js +96 -0
  136. package/locales/pt/index.js +98 -0
  137. package/locales/pt/package.json +6 -0
  138. package/locales/pt-BR/index.d.ts +3 -0
  139. package/locales/pt-BR/index.esm.d.ts +3 -0
  140. package/locales/pt-BR/index.esm.js +96 -0
  141. package/locales/pt-BR/index.js +98 -0
  142. package/locales/pt-BR/package.json +6 -0
  143. package/locales/ro/index.d.ts +3 -0
  144. package/locales/ro/index.esm.d.ts +3 -0
  145. package/locales/ro/index.esm.js +96 -0
  146. package/locales/ro/index.js +98 -0
  147. package/locales/ro/package.json +6 -0
  148. package/locales/ru/index.d.ts +3 -0
  149. package/locales/ru/index.esm.d.ts +3 -0
  150. package/locales/ru/index.esm.js +96 -0
  151. package/locales/ru/index.js +98 -0
  152. package/locales/ru/package.json +6 -0
  153. package/locales/sk/index.d.ts +3 -0
  154. package/locales/sk/index.esm.d.ts +3 -0
  155. package/locales/sk/index.esm.js +96 -0
  156. package/locales/sk/index.js +98 -0
  157. package/locales/sk/package.json +6 -0
  158. package/locales/sr-Cyrl-RS/index.d.ts +3 -0
  159. package/locales/sr-Cyrl-RS/index.esm.d.ts +3 -0
  160. package/locales/sr-Cyrl-RS/index.esm.js +96 -0
  161. package/locales/sr-Cyrl-RS/index.js +98 -0
  162. package/locales/sr-Cyrl-RS/package.json +6 -0
  163. package/locales/sr-Latn-RS/index.d.ts +3 -0
  164. package/locales/sr-Latn-RS/index.esm.d.ts +3 -0
  165. package/locales/sr-Latn-RS/index.esm.js +96 -0
  166. package/locales/sr-Latn-RS/index.js +98 -0
  167. package/locales/sr-Latn-RS/package.json +6 -0
  168. package/locales/sv/index.d.ts +3 -0
  169. package/locales/sv/index.esm.d.ts +3 -0
  170. package/locales/sv/index.esm.js +96 -0
  171. package/locales/sv/index.js +98 -0
  172. package/locales/sv/package.json +6 -0
  173. package/locales/tr/index.d.ts +3 -0
  174. package/locales/tr/index.esm.d.ts +3 -0
  175. package/locales/tr/index.esm.js +96 -0
  176. package/locales/tr/index.js +98 -0
  177. package/locales/tr/package.json +6 -0
  178. package/locales/uk/index.d.ts +3 -0
  179. package/locales/uk/index.esm.d.ts +3 -0
  180. package/locales/uk/index.esm.js +96 -0
  181. package/locales/uk/index.js +98 -0
  182. package/locales/uk/package.json +6 -0
  183. package/locales/vi/index.d.ts +3 -0
  184. package/locales/vi/index.esm.d.ts +3 -0
  185. package/locales/vi/index.esm.js +96 -0
  186. package/locales/vi/index.js +98 -0
  187. package/locales/vi/package.json +6 -0
  188. package/locales/zh-Hans/index.d.ts +3 -0
  189. package/locales/zh-Hans/index.esm.d.ts +3 -0
  190. package/locales/zh-Hans/index.esm.js +96 -0
  191. package/locales/zh-Hans/index.js +98 -0
  192. package/locales/zh-Hans/package.json +6 -0
  193. package/locales/zh-Hant/index.d.ts +3 -0
  194. package/locales/zh-Hant/index.esm.d.ts +3 -0
  195. package/locales/zh-Hant/index.esm.js +96 -0
  196. package/locales/zh-Hant/index.js +98 -0
  197. package/locales/zh-Hant/package.json +6 -0
  198. package/package.json +103 -0
  199. package/src/components/MaterialReactTable.tsx +36 -0
  200. package/src/components/body/MRT_TableBody.tsx +233 -0
  201. package/src/components/body/MRT_TableBodyCell.tsx +356 -0
  202. package/src/components/body/MRT_TableBodyCellValue.tsx +130 -0
  203. package/src/components/body/MRT_TableBodyRow.tsx +306 -0
  204. package/src/components/body/MRT_TableBodyRowGrabHandle.tsx +61 -0
  205. package/src/components/body/MRT_TableBodyRowPinButton.tsx +58 -0
  206. package/src/components/body/MRT_TableDetailPanel.tsx +118 -0
  207. package/src/components/buttons/MRT_ColumnPinningButtons.tsx +73 -0
  208. package/src/components/buttons/MRT_CopyButton.tsx +91 -0
  209. package/src/components/buttons/MRT_EditActionButtons.tsx +135 -0
  210. package/src/components/buttons/MRT_ExpandAllButton.tsx +82 -0
  211. package/src/components/buttons/MRT_ExpandButton.tsx +106 -0
  212. package/src/components/buttons/MRT_GrabHandleButton.tsx +66 -0
  213. package/src/components/buttons/MRT_RowPinButton.tsx +84 -0
  214. package/src/components/buttons/MRT_ShowHideColumnsButton.tsx +50 -0
  215. package/src/components/buttons/MRT_ToggleDensePaddingButton.tsx +52 -0
  216. package/src/components/buttons/MRT_ToggleFiltersButton.tsx +40 -0
  217. package/src/components/buttons/MRT_ToggleFullScreenButton.tsx +51 -0
  218. package/src/components/buttons/MRT_ToggleGlobalFilterButton.tsx +46 -0
  219. package/src/components/buttons/MRT_ToggleRowActionMenuButton.tsx +126 -0
  220. package/src/components/footer/MRT_TableFooter.tsx +92 -0
  221. package/src/components/footer/MRT_TableFooterCell.tsx +111 -0
  222. package/src/components/footer/MRT_TableFooterRow.tsx +99 -0
  223. package/src/components/head/MRT_TableHead.tsx +95 -0
  224. package/src/components/head/MRT_TableHeadCell.tsx +357 -0
  225. package/src/components/head/MRT_TableHeadCellColumnActionsButton.tsx +100 -0
  226. package/src/components/head/MRT_TableHeadCellFilterContainer.tsx +52 -0
  227. package/src/components/head/MRT_TableHeadCellFilterLabel.tsx +180 -0
  228. package/src/components/head/MRT_TableHeadCellGrabHandle.tsx +91 -0
  229. package/src/components/head/MRT_TableHeadCellResizeHandle.tsx +96 -0
  230. package/src/components/head/MRT_TableHeadCellSortLabel.tsx +102 -0
  231. package/src/components/head/MRT_TableHeadRow.tsx +94 -0
  232. package/src/components/inputs/MRT_EditCellTextField.tsx +195 -0
  233. package/src/components/inputs/MRT_FilterCheckbox.tsx +107 -0
  234. package/src/components/inputs/MRT_FilterRangeFields.tsx +42 -0
  235. package/src/components/inputs/MRT_FilterRangeSlider.tsx +145 -0
  236. package/src/components/inputs/MRT_FilterTextField.tsx +629 -0
  237. package/src/components/inputs/MRT_GlobalFilterTextField.tsx +165 -0
  238. package/src/components/inputs/MRT_SelectCheckbox.tsx +142 -0
  239. package/src/components/menus/MRT_ActionMenuItem.tsx +62 -0
  240. package/src/components/menus/MRT_CellActionMenu.tsx +109 -0
  241. package/src/components/menus/MRT_ColumnActionMenu.tsx +354 -0
  242. package/src/components/menus/MRT_FilterOptionMenu.tsx +294 -0
  243. package/src/components/menus/MRT_RowActionMenu.tsx +90 -0
  244. package/src/components/menus/MRT_ShowHideColumnsMenu.tsx +181 -0
  245. package/src/components/menus/MRT_ShowHideColumnsMenuItems.tsx +204 -0
  246. package/src/components/modals/MRT_EditRowModal.tsx +114 -0
  247. package/src/components/table/MRT_Table.tsx +86 -0
  248. package/src/components/table/MRT_TableContainer.tsx +115 -0
  249. package/src/components/table/MRT_TableLoadingOverlay.tsx +58 -0
  250. package/src/components/table/MRT_TablePaper.tsx +91 -0
  251. package/src/components/toolbar/MRT_BottomToolbar.tsx +112 -0
  252. package/src/components/toolbar/MRT_LinearProgressBar.tsx +53 -0
  253. package/src/components/toolbar/MRT_TablePagination.tsx +250 -0
  254. package/src/components/toolbar/MRT_ToolbarAlertBanner.tsx +182 -0
  255. package/src/components/toolbar/MRT_ToolbarDropZone.tsx +82 -0
  256. package/src/components/toolbar/MRT_ToolbarInternalButtons.tsx +73 -0
  257. package/src/components/toolbar/MRT_TopToolbar.tsx +134 -0
  258. package/src/fns/aggregationFns.ts +3 -0
  259. package/src/fns/filterFns.ts +196 -0
  260. package/src/fns/sortingFns.ts +33 -0
  261. package/src/hooks/display-columns/getMRT_RowActionsColumnDef.tsx +28 -0
  262. package/src/hooks/display-columns/getMRT_RowDragColumnDef.tsx +29 -0
  263. package/src/hooks/display-columns/getMRT_RowExpandColumnDef.tsx +91 -0
  264. package/src/hooks/display-columns/getMRT_RowNumbersColumnDef.tsx +30 -0
  265. package/src/hooks/display-columns/getMRT_RowPinningColumnDef.tsx +24 -0
  266. package/src/hooks/display-columns/getMRT_RowSelectColumnDef.tsx +34 -0
  267. package/src/hooks/display-columns/getMRT_RowSpacerColumnDef.tsx +33 -0
  268. package/src/hooks/useMRT_ColumnVirtualizer.ts +124 -0
  269. package/src/hooks/useMRT_Effects.ts +106 -0
  270. package/src/hooks/useMRT_RowVirtualizer.ts +88 -0
  271. package/src/hooks/useMRT_Rows.ts +44 -0
  272. package/src/hooks/useMRT_TableInstance.ts +317 -0
  273. package/src/hooks/useMRT_TableOptions.ts +270 -0
  274. package/src/hooks/useMaterialReactTable.ts +12 -0
  275. package/src/icons.ts +73 -0
  276. package/src/index.ts +91 -0
  277. package/src/locales/ar.ts +97 -0
  278. package/src/locales/az.ts +97 -0
  279. package/src/locales/bg.ts +97 -0
  280. package/src/locales/cs.ts +98 -0
  281. package/src/locales/da.ts +97 -0
  282. package/src/locales/de.ts +97 -0
  283. package/src/locales/el.ts +97 -0
  284. package/src/locales/en.ts +97 -0
  285. package/src/locales/es.ts +97 -0
  286. package/src/locales/et.ts +98 -0
  287. package/src/locales/fa.ts +97 -0
  288. package/src/locales/fi.ts +98 -0
  289. package/src/locales/fr.ts +97 -0
  290. package/src/locales/he.ts +97 -0
  291. package/src/locales/hr.ts +97 -0
  292. package/src/locales/hu.ts +97 -0
  293. package/src/locales/hy.ts +97 -0
  294. package/src/locales/id.ts +98 -0
  295. package/src/locales/it.ts +97 -0
  296. package/src/locales/ja.ts +97 -0
  297. package/src/locales/ko.ts +97 -0
  298. package/src/locales/mk.ts +96 -0
  299. package/src/locales/nl.ts +97 -0
  300. package/src/locales/no.ts +99 -0
  301. package/src/locales/np.ts +98 -0
  302. package/src/locales/pl.ts +97 -0
  303. package/src/locales/pt-BR.ts +97 -0
  304. package/src/locales/pt.ts +97 -0
  305. package/src/locales/ro.ts +97 -0
  306. package/src/locales/ru.ts +97 -0
  307. package/src/locales/sk.ts +98 -0
  308. package/src/locales/sr-Cyrl-RS.ts +97 -0
  309. package/src/locales/sr-Latn-RS.ts +97 -0
  310. package/src/locales/sv.ts +97 -0
  311. package/src/locales/tr.ts +97 -0
  312. package/src/locales/uk.ts +97 -0
  313. package/src/locales/vi.ts +97 -0
  314. package/src/locales/zh-Hans.ts +96 -0
  315. package/src/locales/zh-Hant.ts +96 -0
  316. package/src/types.ts +1295 -0
  317. package/src/utils/cell.utils.ts +234 -0
  318. package/src/utils/column.utils.ts +209 -0
  319. package/src/utils/displayColumn.utils.ts +152 -0
  320. package/src/utils/row.utils.ts +260 -0
  321. package/src/utils/style.utils.ts +221 -0
  322. package/src/utils/tanstack.helpers.ts +64 -0
  323. package/src/utils/utils.ts +56 -0
  324. package/src/utils/virtualization.utils.ts +24 -0
@@ -0,0 +1,354 @@
1
+ import { type MouseEvent, useState } from 'react';
2
+ import Menu, { type MenuProps } from '@mui/material/Menu';
3
+ import { MRT_ActionMenuItem } from './MRT_ActionMenuItem';
4
+ import { MRT_FilterOptionMenu } from './MRT_FilterOptionMenu';
5
+ import {
6
+ type MRT_Header,
7
+ type MRT_RowData,
8
+ type MRT_TableInstance,
9
+ } from '../../types';
10
+
11
+ export interface MRT_ColumnActionMenuProps<TData extends MRT_RowData>
12
+ extends Partial<MenuProps> {
13
+ anchorEl: HTMLElement | null;
14
+ header: MRT_Header<TData>;
15
+ setAnchorEl: (anchorEl: HTMLElement | null) => void;
16
+ table: MRT_TableInstance<TData>;
17
+ }
18
+
19
+ export const MRT_ColumnActionMenu = <TData extends MRT_RowData>({
20
+ anchorEl,
21
+ header,
22
+ setAnchorEl,
23
+ table,
24
+ ...rest
25
+ }: MRT_ColumnActionMenuProps<TData>) => {
26
+ const {
27
+ getAllLeafColumns,
28
+ getState,
29
+ options: {
30
+ columnFilterDisplayMode,
31
+ columnFilterModeOptions,
32
+ enableColumnFilterModes,
33
+ enableColumnFilters,
34
+ enableColumnPinning,
35
+ enableColumnResizing,
36
+ enableGrouping,
37
+ enableHiding,
38
+ enableSorting,
39
+ enableSortingRemoval,
40
+ icons: {
41
+ ClearAllIcon,
42
+ DynamicFeedIcon,
43
+ FilterListIcon,
44
+ FilterListOffIcon,
45
+ PushPinIcon,
46
+ RestartAltIcon,
47
+ SortIcon,
48
+ ViewColumnIcon,
49
+ VisibilityOffIcon,
50
+ },
51
+ localization,
52
+ mrtTheme: { menuBackgroundColor },
53
+ renderColumnActionsMenuItems,
54
+ },
55
+ refs: { filterInputRefs },
56
+ setColumnFilterFns,
57
+ setColumnOrder,
58
+ setColumnSizingInfo,
59
+ setShowColumnFilters,
60
+ } = table;
61
+ const { column } = header;
62
+ const { columnDef } = column;
63
+ const { columnSizing, columnVisibility, density, showColumnFilters } =
64
+ getState();
65
+ const columnFilterValue = column.getFilterValue();
66
+
67
+ const [filterMenuAnchorEl, setFilterMenuAnchorEl] =
68
+ useState<HTMLElement | null>(null);
69
+
70
+ const handleClearSort = () => {
71
+ column.clearSorting();
72
+ setAnchorEl(null);
73
+ };
74
+
75
+ const handleSortAsc = () => {
76
+ column.toggleSorting(false);
77
+ setAnchorEl(null);
78
+ };
79
+
80
+ const handleSortDesc = () => {
81
+ column.toggleSorting(true);
82
+ setAnchorEl(null);
83
+ };
84
+
85
+ const handleResetColumnSize = () => {
86
+ setColumnSizingInfo((old) => ({ ...old, isResizingColumn: false }));
87
+ column.resetSize();
88
+ setAnchorEl(null);
89
+ };
90
+
91
+ const handleHideColumn = () => {
92
+ column.toggleVisibility(false);
93
+ setAnchorEl(null);
94
+ };
95
+
96
+ const handlePinColumn = (pinDirection: 'left' | 'right' | false) => {
97
+ column.pin(pinDirection);
98
+ setAnchorEl(null);
99
+ };
100
+
101
+ const handleGroupByColumn = () => {
102
+ column.toggleGrouping();
103
+ setColumnOrder((old: any) => ['mrt-row-expand', ...old]);
104
+ setAnchorEl(null);
105
+ };
106
+
107
+ const handleClearFilter = () => {
108
+ column.setFilterValue(undefined);
109
+ setAnchorEl(null);
110
+ if (['empty', 'notEmpty'].includes(columnDef._filterFn)) {
111
+ setColumnFilterFns((prev) => ({
112
+ ...prev,
113
+ [header.id]: allowedColumnFilterOptions?.[0] ?? 'fuzzy',
114
+ }));
115
+ }
116
+ };
117
+
118
+ const handleFilterByColumn = () => {
119
+ setShowColumnFilters(true);
120
+ queueMicrotask(() => filterInputRefs.current?.[`${column.id}-0`]?.focus());
121
+ setAnchorEl(null);
122
+ };
123
+
124
+ const handleShowAllColumns = () => {
125
+ getAllLeafColumns()
126
+ .filter((col) => col.columnDef.enableHiding !== false)
127
+ .forEach((col) => col.toggleVisibility(true));
128
+ setAnchorEl(null);
129
+ };
130
+
131
+ const handleOpenFilterModeMenu = (event: MouseEvent<HTMLElement>) => {
132
+ event.stopPropagation();
133
+ setFilterMenuAnchorEl(event.currentTarget);
134
+ };
135
+
136
+ const isSelectFilter = !!columnDef.filterSelectOptions;
137
+
138
+ const allowedColumnFilterOptions =
139
+ columnDef?.columnFilterModeOptions ?? columnFilterModeOptions;
140
+
141
+ const showFilterModeSubMenu =
142
+ enableColumnFilterModes &&
143
+ columnDef.enableColumnFilterModes !== false &&
144
+ !isSelectFilter &&
145
+ (allowedColumnFilterOptions === undefined ||
146
+ !!allowedColumnFilterOptions?.length);
147
+
148
+ const internalColumnMenuItems = [
149
+ ...(enableSorting && column.getCanSort()
150
+ ? [
151
+ enableSortingRemoval !== false && (
152
+ <MRT_ActionMenuItem
153
+ disabled={column.getIsSorted() === false}
154
+ icon={<ClearAllIcon />}
155
+ key={0}
156
+ label={localization.clearSort}
157
+ onClick={handleClearSort}
158
+ table={table}
159
+ />
160
+ ),
161
+ <MRT_ActionMenuItem
162
+ disabled={column.getIsSorted() === 'asc'}
163
+ icon={
164
+ <SortIcon style={{ transform: 'rotate(180deg) scaleX(-1)' }} />
165
+ }
166
+ key={1}
167
+ label={localization.sortByColumnAsc?.replace(
168
+ '{column}',
169
+ String(columnDef.header),
170
+ )}
171
+ onClick={handleSortAsc}
172
+ table={table}
173
+ />,
174
+ <MRT_ActionMenuItem
175
+ disabled={column.getIsSorted() === 'desc'}
176
+ divider={enableColumnFilters || enableGrouping || enableHiding}
177
+ icon={<SortIcon />}
178
+ key={2}
179
+ label={localization.sortByColumnDesc?.replace(
180
+ '{column}',
181
+ String(columnDef.header),
182
+ )}
183
+ onClick={handleSortDesc}
184
+ table={table}
185
+ />,
186
+ ]
187
+ : []),
188
+ ...(enableColumnFilters && column.getCanFilter()
189
+ ? [
190
+ <MRT_ActionMenuItem
191
+ disabled={
192
+ !columnFilterValue ||
193
+ (Array.isArray(columnFilterValue) &&
194
+ !columnFilterValue.filter((value) => value).length)
195
+ }
196
+ icon={<FilterListOffIcon />}
197
+ key={3}
198
+ label={localization.clearFilter}
199
+ onClick={handleClearFilter}
200
+ table={table}
201
+ />,
202
+ columnFilterDisplayMode === 'subheader' && (
203
+ <MRT_ActionMenuItem
204
+ disabled={showColumnFilters && !enableColumnFilterModes}
205
+ divider={enableGrouping || enableHiding}
206
+ icon={<FilterListIcon />}
207
+ key={4}
208
+ label={localization.filterByColumn?.replace(
209
+ '{column}',
210
+ String(columnDef.header),
211
+ )}
212
+ onClick={
213
+ showColumnFilters
214
+ ? handleOpenFilterModeMenu
215
+ : handleFilterByColumn
216
+ }
217
+ onOpenSubMenu={
218
+ showFilterModeSubMenu ? handleOpenFilterModeMenu : undefined
219
+ }
220
+ table={table}
221
+ />
222
+ ),
223
+ showFilterModeSubMenu && (
224
+ <MRT_FilterOptionMenu
225
+ anchorEl={filterMenuAnchorEl}
226
+ header={header}
227
+ key={5}
228
+ onSelect={handleFilterByColumn}
229
+ setAnchorEl={setFilterMenuAnchorEl}
230
+ table={table}
231
+ />
232
+ ),
233
+ ].filter(Boolean)
234
+ : []),
235
+ ...(enableGrouping && column.getCanGroup()
236
+ ? [
237
+ <MRT_ActionMenuItem
238
+ divider={enableColumnPinning}
239
+ icon={<DynamicFeedIcon />}
240
+ key={6}
241
+ label={localization[
242
+ column.getIsGrouped() ? 'ungroupByColumn' : 'groupByColumn'
243
+ ]?.replace('{column}', String(columnDef.header))}
244
+ onClick={handleGroupByColumn}
245
+ table={table}
246
+ />,
247
+ ]
248
+ : []),
249
+ ...(enableColumnPinning && column.getCanPin()
250
+ ? [
251
+ <MRT_ActionMenuItem
252
+ disabled={column.getIsPinned() === 'left' || !column.getCanPin()}
253
+ icon={<PushPinIcon style={{ transform: 'rotate(90deg)' }} />}
254
+ key={7}
255
+ label={localization.pinToLeft}
256
+ onClick={() => handlePinColumn('left')}
257
+ table={table}
258
+ />,
259
+ <MRT_ActionMenuItem
260
+ disabled={column.getIsPinned() === 'right' || !column.getCanPin()}
261
+ icon={<PushPinIcon style={{ transform: 'rotate(-90deg)' }} />}
262
+ key={8}
263
+ label={localization.pinToRight}
264
+ onClick={() => handlePinColumn('right')}
265
+ table={table}
266
+ />,
267
+ <MRT_ActionMenuItem
268
+ disabled={!column.getIsPinned()}
269
+ divider={enableHiding}
270
+ icon={<PushPinIcon />}
271
+ key={9}
272
+ label={localization.unpin}
273
+ onClick={() => handlePinColumn(false)}
274
+ table={table}
275
+ />,
276
+ ]
277
+ : []),
278
+ ...(enableColumnResizing && column.getCanResize()
279
+ ? [
280
+ <MRT_ActionMenuItem
281
+ disabled={columnSizing[column.id] === undefined}
282
+ icon={<RestartAltIcon />}
283
+ key={10}
284
+ label={localization.resetColumnSize}
285
+ onClick={handleResetColumnSize}
286
+ table={table}
287
+ />,
288
+ ]
289
+ : []),
290
+ ...(enableHiding
291
+ ? [
292
+ <MRT_ActionMenuItem
293
+ disabled={!column.getCanHide()}
294
+ icon={<VisibilityOffIcon />}
295
+ key={11}
296
+ label={localization.hideColumn?.replace(
297
+ '{column}',
298
+ String(columnDef.header),
299
+ )}
300
+ onClick={handleHideColumn}
301
+ table={table}
302
+ />,
303
+ <MRT_ActionMenuItem
304
+ disabled={
305
+ !Object.values(columnVisibility).filter((visible) => !visible)
306
+ .length
307
+ }
308
+ icon={<ViewColumnIcon />}
309
+ key={12}
310
+ label={localization.showAllColumns?.replace(
311
+ '{column}',
312
+ String(columnDef.header),
313
+ )}
314
+ onClick={handleShowAllColumns}
315
+ table={table}
316
+ />,
317
+ ]
318
+ : []),
319
+ ].filter(Boolean);
320
+
321
+ return (
322
+ <Menu
323
+ anchorEl={anchorEl}
324
+ disableScrollLock
325
+ onClose={() => setAnchorEl(null)}
326
+ open={!!anchorEl}
327
+ slotProps={{
328
+ list: {
329
+ dense: density === 'compact',
330
+ sx: {
331
+ backgroundColor: menuBackgroundColor,
332
+ },
333
+ ...rest.slotProps?.list,
334
+ },
335
+ ...rest.slotProps,
336
+ }}
337
+ {...rest}
338
+ >
339
+ {columnDef.renderColumnActionsMenuItems?.({
340
+ closeMenu: () => setAnchorEl(null),
341
+ column,
342
+ internalColumnMenuItems,
343
+ table,
344
+ }) ??
345
+ renderColumnActionsMenuItems?.({
346
+ closeMenu: () => setAnchorEl(null),
347
+ column,
348
+ internalColumnMenuItems,
349
+ table,
350
+ }) ??
351
+ internalColumnMenuItems}
352
+ </Menu>
353
+ );
354
+ };
@@ -0,0 +1,294 @@
1
+ import { useMemo } from 'react';
2
+ import Menu, { type MenuProps } from '@mui/material/Menu';
3
+ import { MRT_ActionMenuItem } from './MRT_ActionMenuItem';
4
+ import {
5
+ type MRT_FilterOption,
6
+ type MRT_Header,
7
+ type MRT_InternalFilterOption,
8
+ type MRT_Localization,
9
+ type MRT_RowData,
10
+ type MRT_TableInstance,
11
+ } from '../../types';
12
+
13
+ export const mrtFilterOptions = (
14
+ localization: MRT_Localization,
15
+ ): MRT_InternalFilterOption[] => [
16
+ {
17
+ divider: false,
18
+ label: localization.filterFuzzy,
19
+ option: 'fuzzy',
20
+ symbol: '≈',
21
+ },
22
+ {
23
+ divider: false,
24
+ label: localization.filterContains,
25
+ option: 'contains',
26
+ symbol: '*',
27
+ },
28
+ {
29
+ divider: false,
30
+ label: localization.filterStartsWith,
31
+ option: 'startsWith',
32
+ symbol: 'a',
33
+ },
34
+ {
35
+ divider: true,
36
+ label: localization.filterEndsWith,
37
+ option: 'endsWith',
38
+ symbol: 'z',
39
+ },
40
+ {
41
+ divider: false,
42
+ label: localization.filterEquals,
43
+ option: 'equals',
44
+ symbol: '=',
45
+ },
46
+ {
47
+ divider: true,
48
+ label: localization.filterNotEquals,
49
+ option: 'notEquals',
50
+ symbol: '≠',
51
+ },
52
+ {
53
+ divider: false,
54
+ label: localization.filterBetween,
55
+ option: 'between',
56
+ symbol: '⇿',
57
+ },
58
+ {
59
+ divider: true,
60
+ label: localization.filterBetweenInclusive,
61
+ option: 'betweenInclusive',
62
+ symbol: '⬌',
63
+ },
64
+ {
65
+ divider: false,
66
+ label: localization.filterGreaterThan,
67
+ option: 'greaterThan',
68
+ symbol: '>',
69
+ },
70
+ {
71
+ divider: false,
72
+ label: localization.filterGreaterThanOrEqualTo,
73
+ option: 'greaterThanOrEqualTo',
74
+ symbol: '≥',
75
+ },
76
+ {
77
+ divider: false,
78
+ label: localization.filterLessThan,
79
+ option: 'lessThan',
80
+ symbol: '<',
81
+ },
82
+ {
83
+ divider: true,
84
+ label: localization.filterLessThanOrEqualTo,
85
+ option: 'lessThanOrEqualTo',
86
+ symbol: '≤',
87
+ },
88
+ {
89
+ divider: false,
90
+ label: localization.filterEmpty,
91
+ option: 'empty',
92
+ symbol: '∅',
93
+ },
94
+ {
95
+ divider: false,
96
+ label: localization.filterNotEmpty,
97
+ option: 'notEmpty',
98
+ symbol: '!∅',
99
+ },
100
+ ];
101
+
102
+ const rangeModes = ['between', 'betweenInclusive', 'inNumberRange'];
103
+ const emptyModes = ['empty', 'notEmpty'];
104
+ const arrModes = ['arrIncludesSome', 'arrIncludesAll', 'arrIncludes'];
105
+ const rangeVariants = ['range-slider', 'date-range', 'datetime-range', 'range'];
106
+
107
+ export interface MRT_FilterOptionMenuProps<TData extends MRT_RowData>
108
+ extends Partial<MenuProps> {
109
+ anchorEl: HTMLElement | null;
110
+ header?: MRT_Header<TData>;
111
+ onSelect?: () => void;
112
+ setAnchorEl: (anchorEl: HTMLElement | null) => void;
113
+ setFilterValue?: (filterValue: any) => void;
114
+ table: MRT_TableInstance<TData>;
115
+ }
116
+
117
+ export const MRT_FilterOptionMenu = <TData extends MRT_RowData>({
118
+ anchorEl,
119
+ header,
120
+ onSelect,
121
+ setAnchorEl,
122
+ setFilterValue,
123
+ table,
124
+ ...rest
125
+ }: MRT_FilterOptionMenuProps<TData>) => {
126
+ const {
127
+ getState,
128
+ options: {
129
+ columnFilterModeOptions,
130
+ globalFilterModeOptions,
131
+ localization,
132
+ mrtTheme: { menuBackgroundColor },
133
+ renderColumnFilterModeMenuItems,
134
+ renderGlobalFilterModeMenuItems,
135
+ },
136
+ setColumnFilterFns,
137
+ setGlobalFilterFn,
138
+ } = table;
139
+ const { density, globalFilterFn } = getState();
140
+ const { column } = header ?? {};
141
+ const { columnDef } = column ?? {};
142
+ const currentFilterValue = column?.getFilterValue();
143
+
144
+ let allowedColumnFilterOptions =
145
+ columnDef?.columnFilterModeOptions ?? columnFilterModeOptions;
146
+
147
+ if (rangeVariants.includes(columnDef?.filterVariant as string)) {
148
+ allowedColumnFilterOptions = [
149
+ ...rangeModes,
150
+ ...(allowedColumnFilterOptions ?? []),
151
+ ].filter((option) => rangeModes.includes(option));
152
+ }
153
+
154
+ const internalFilterOptions = useMemo(
155
+ () =>
156
+ mrtFilterOptions(localization).filter((filterOption) =>
157
+ columnDef
158
+ ? allowedColumnFilterOptions === undefined ||
159
+ allowedColumnFilterOptions?.includes(filterOption.option)
160
+ : (!globalFilterModeOptions ||
161
+ globalFilterModeOptions.includes(filterOption.option)) &&
162
+ ['contains', 'fuzzy', 'startsWith'].includes(filterOption.option),
163
+ ),
164
+ [],
165
+ );
166
+
167
+ const handleSelectFilterMode = (option: MRT_FilterOption) => {
168
+ const prevFilterMode = columnDef?._filterFn ?? '';
169
+ if (!header || !column) {
170
+ // global filter mode
171
+ setGlobalFilterFn(option);
172
+ } else if (option !== prevFilterMode) {
173
+ // column filter mode
174
+ setColumnFilterFns((prev: { [key: string]: any }) => ({
175
+ ...prev,
176
+ [header.id]: option,
177
+ }));
178
+
179
+ // reset filter value and/or perform new filter render
180
+ if (emptyModes.includes(option)) {
181
+ // will now be empty/notEmpty filter mode
182
+ if (
183
+ currentFilterValue !== ' ' &&
184
+ !emptyModes.includes(prevFilterMode)
185
+ ) {
186
+ column.setFilterValue(' ');
187
+ } else if (currentFilterValue) {
188
+ column.setFilterValue(currentFilterValue); // perform new filter render
189
+ }
190
+ } else if (
191
+ columnDef?.filterVariant === 'multi-select' ||
192
+ arrModes.includes(option as string)
193
+ ) {
194
+ // will now be array filter mode
195
+ if (
196
+ currentFilterValue instanceof String ||
197
+ (currentFilterValue as Array<any>)?.length
198
+ ) {
199
+ column.setFilterValue([]);
200
+ setFilterValue?.([]);
201
+ } else if (currentFilterValue) {
202
+ column.setFilterValue(currentFilterValue); // perform new filter render
203
+ }
204
+ } else if (
205
+ columnDef?.filterVariant?.includes('range') ||
206
+ rangeModes.includes(option as MRT_FilterOption)
207
+ ) {
208
+ // will now be range filter mode
209
+ if (
210
+ !Array.isArray(currentFilterValue) ||
211
+ (!(currentFilterValue as Array<any>)?.every((v) => v === '') &&
212
+ !rangeModes.includes(prevFilterMode))
213
+ ) {
214
+ column.setFilterValue(['', '']);
215
+ setFilterValue?.('');
216
+ } else {
217
+ column.setFilterValue(currentFilterValue); // perform new filter render
218
+ }
219
+ } else {
220
+ // will now be single value filter mode
221
+ if (Array.isArray(currentFilterValue)) {
222
+ column.setFilterValue('');
223
+ setFilterValue?.('');
224
+ } else if (
225
+ currentFilterValue === ' ' &&
226
+ emptyModes.includes(prevFilterMode)
227
+ ) {
228
+ column.setFilterValue(undefined);
229
+ } else {
230
+ column.setFilterValue(currentFilterValue); // perform new filter render
231
+ }
232
+ }
233
+ }
234
+ setAnchorEl(null);
235
+ onSelect?.();
236
+ };
237
+
238
+ const filterOption =
239
+ !!header && columnDef ? columnDef._filterFn : globalFilterFn;
240
+
241
+ return (
242
+ <Menu
243
+ anchorEl={anchorEl}
244
+ anchorOrigin={{ horizontal: 'right', vertical: 'center' }}
245
+ disableScrollLock
246
+ onClose={() => setAnchorEl(null)}
247
+ open={!!anchorEl}
248
+ slotProps={{
249
+ list: {
250
+ dense: density === 'compact',
251
+ sx: {
252
+ backgroundColor: menuBackgroundColor,
253
+ },
254
+ ...rest.slotProps?.list,
255
+ },
256
+ ...rest.slotProps,
257
+ }}
258
+ {...rest}
259
+ >
260
+ {(header && column && columnDef
261
+ ? (columnDef.renderColumnFilterModeMenuItems?.({
262
+ column: column as any,
263
+ internalFilterOptions,
264
+ onSelectFilterMode: handleSelectFilterMode,
265
+ table,
266
+ }) ??
267
+ renderColumnFilterModeMenuItems?.({
268
+ column: column as any,
269
+ internalFilterOptions,
270
+ onSelectFilterMode: handleSelectFilterMode,
271
+ table,
272
+ }))
273
+ : renderGlobalFilterModeMenuItems?.({
274
+ internalFilterOptions,
275
+ onSelectFilterMode: handleSelectFilterMode,
276
+ table,
277
+ })) ??
278
+ internalFilterOptions.map(
279
+ ({ divider, label, option, symbol }, index) => (
280
+ <MRT_ActionMenuItem
281
+ divider={divider}
282
+ icon={symbol}
283
+ key={index}
284
+ label={label}
285
+ onClick={() => handleSelectFilterMode(option as MRT_FilterOption)}
286
+ selected={option === filterOption}
287
+ table={table}
288
+ value={option}
289
+ />
290
+ ),
291
+ )}
292
+ </Menu>
293
+ );
294
+ };