@@ -16,8 +16,9 @@ import shortid from 'shortid';
16
16
17
17
import classNames from 'classnames' ;
18
18
import assign from 'lodash.assign' ;
19
+ import isEqual from 'lodash.isequal' ;
20
+ import memoize from 'memoize-one' ;
19
21
import reject from 'lodash.reject' ;
20
-
21
22
// This component's `checkProps` which issues warnings to developers about properties when in development mode (similar to React's built in development tools)
22
23
import ColumnResizer from 'column-resizer' ;
23
24
import checkProps from './check-props' ;
@@ -74,6 +75,90 @@ const defaultProps = {
74
75
} ,
75
76
} ;
76
77
78
+ const getAssistiveText = memoize (
79
+ (
80
+ assistiveText ,
81
+ actionsHeaderText ,
82
+ columnSortText ,
83
+ columnSortedAscendingText ,
84
+ columnSortedDescendingText ,
85
+ selectAllRowsText ,
86
+ selectRowText
87
+ ) => {
88
+ const result = {
89
+ ...defaultProps . assistiveText ,
90
+ ...assistiveText ,
91
+ } ;
92
+ if ( actionsHeaderText ) {
93
+ result . actionsHeader = actionsHeaderText ;
94
+ }
95
+ if ( selectAllRowsText ) {
96
+ result . selectAllRows = selectAllRowsText ;
97
+ }
98
+ if ( columnSortedAscendingText ) {
99
+ result . columnSortedAscending = columnSortedAscendingText ;
100
+ }
101
+ if ( columnSortedDescendingText ) {
102
+ result . columnSortedDescending = columnSortedDescendingText ;
103
+ }
104
+ if ( columnSortText ) {
105
+ result . columnSort = columnSortText ;
106
+ }
107
+ if ( selectRowText ) {
108
+ result . selectRow = selectRowText ;
109
+ }
110
+ return result ;
111
+ } ,
112
+ isEqual
113
+ ) ;
114
+
115
+ const getColumnsAndRowActions = memoize (
116
+ ( children , id , fixedHeader , fixedLayout , search ) => {
117
+ const columns = [ ] ;
118
+ let RowActions = null ;
119
+
120
+ React . Children . forEach ( children , ( child ) => {
121
+ if ( child && child . type . displayName === DataTableColumn . displayName ) {
122
+ const { children : columnChildren , ...columnProps } = child . props ;
123
+ const props = { fixedLayout, search, id, ...columnProps } ;
124
+
125
+ let Cell ;
126
+ if (
127
+ columnChildren &&
128
+ columnChildren . type . displayName === DATA_TABLE_CELL
129
+ ) {
130
+ Cell = columnChildren . type ;
131
+ assign ( props , columnChildren . props ) ;
132
+ } else {
133
+ Cell = DataTableCell ;
134
+ }
135
+
136
+ // eslint-disable-next-line fp/no-mutating-methods
137
+ columns . push ( {
138
+ Cell,
139
+ props,
140
+ } ) ;
141
+ } else if (
142
+ child &&
143
+ child . type . displayName === DataTableRowActions . displayName
144
+ ) {
145
+ const { dropdown } = child . props ;
146
+ const dropdownPropOverrides = { } ;
147
+ if ( fixedHeader ) {
148
+ dropdownPropOverrides . menuPosition = 'overflowBoundaryElement' ;
149
+ }
150
+ RowActions = React . cloneElement ( child , {
151
+ dropdown : dropdown
152
+ ? React . cloneElement ( dropdown , dropdownPropOverrides )
153
+ : null ,
154
+ } ) ;
155
+ }
156
+ } ) ;
157
+ return { columns, RowActions } ;
158
+ } ,
159
+ isEqual
160
+ ) ;
161
+
77
162
/**
78
163
* DataTables support the display of structured data in rows and columns with an HTML table. To sort, filter or paginate the table, simply update the data passed in the items to the table and it will re-render itself appropriately. The table will throw a sort event as needed, and helper components for paging and filtering are coming soon.
79
164
*
@@ -366,9 +451,7 @@ class DataTable extends React.Component {
366
451
// Simulating a scroll here will ensure that enough rows are loaded to enable scrolling
367
452
this . loadMoreIfNeeded ( ) ;
368
453
}
369
- if ( this . props . items !== prevProps . items ) {
370
- this . interactiveElements = { } ;
371
- }
454
+
372
455
if (
373
456
this . state . allowKeyboardNavigation &&
374
457
! prevState . allowKeyboardNavigation
@@ -440,6 +523,23 @@ class DataTable extends React.Component {
440
523
return null ;
441
524
}
442
525
526
+ getTableContext = memoize ( ( state , isKeyboardNavigation ) => ( {
527
+ activeCell : state . activeCell ,
528
+ activeElement : state . activeElement ,
529
+ mode : state . mode ,
530
+ tableHasFocus : state . tableHasFocus ,
531
+ changeActiveCell : this . changeActiveCell ,
532
+ changeActiveElement : this . changeActiveElement ,
533
+ handleKeyDown : this . handleKeyDown ,
534
+ registerInteractiveElement : this . registerInteractiveElement ,
535
+ allowKeyboardNavigation : state . allowKeyboardNavigation ,
536
+ setAllowKeyboardNavigation : ( allowKeyboardNavigation ) => {
537
+ if ( isKeyboardNavigation ) {
538
+ this . setState ( { allowKeyboardNavigation } ) ;
539
+ }
540
+ } ,
541
+ } ) ) ;
542
+
443
543
handleToggleAll = ( e , { checked } ) => {
444
544
// REMOVE AT NEXT BREAKING CHANGE
445
545
// `onChange` is deprecated and replaced with `onRowChange`
@@ -587,6 +687,13 @@ class DataTable extends React.Component {
587
687
}
588
688
} ;
589
689
690
+ // eslint-disable-next-line camelcase
691
+ UNSAFE_componentWillUpdate ( nextProps ) {
692
+ if ( this . props . items !== nextProps . items ) {
693
+ this . interactiveElements = { } ;
694
+ }
695
+ }
696
+
590
697
isResizable ( ) {
591
698
return this . props . fixedLayout && this . props . resizable ;
592
699
}
@@ -989,71 +1096,24 @@ class DataTable extends React.Component {
989
1096
const allSelected = canSelectRows && numNonHeaderRows === numSelected ;
990
1097
const indeterminateSelected =
991
1098
canSelectRows && numNonHeaderRows !== numSelected && numSelected !== 0 ;
992
- const columns = [ ] ;
993
- let RowActions = null ;
994
1099
995
- React . Children . forEach ( this . props . children , ( child ) => {
996
- if ( child && child . type . displayName === DataTableColumn . displayName ) {
997
- const { children, ...columnProps } = child . props ;
998
-
999
- const props = assign ( { } , this . props ) ;
1000
- // eslint-disable-next-line fp/no-delete
1001
- delete props . children ;
1002
- assign ( props , columnProps ) ;
1003
-
1004
- let Cell ;
1005
- if ( children && children . type . displayName === DATA_TABLE_CELL ) {
1006
- Cell = children . type ;
1007
- assign ( props , children . props ) ;
1008
- } else {
1009
- Cell = DataTableCell ;
1010
- }
1011
-
1012
- // eslint-disable-next-line fp/no-mutating-methods
1013
- columns . push ( {
1014
- Cell,
1015
- props,
1016
- dataTableProps : this . props ,
1017
- } ) ;
1018
- } else if (
1019
- child &&
1020
- child . type . displayName === DataTableRowActions . displayName
1021
- ) {
1022
- const { dropdown } = child . props ;
1023
- const dropdownPropOverrides = { } ;
1024
- if ( this . getFixedHeader ( ) ) {
1025
- dropdownPropOverrides . menuPosition = 'overflowBoundaryElement' ;
1026
- }
1027
- RowActions = React . cloneElement ( child , {
1028
- dropdown : dropdown
1029
- ? React . cloneElement ( dropdown , dropdownPropOverrides )
1030
- : null ,
1031
- } ) ;
1032
- }
1033
- } ) ;
1100
+ const { columns, RowActions } = getColumnsAndRowActions (
1101
+ this . props . children ,
1102
+ this . props . id ,
1103
+ this . getFixedHeader ( ) ,
1104
+ this . props . fixedLayout ,
1105
+ this . props . search
1106
+ ) ;
1034
1107
1035
- const assistiveText = {
1036
- ...defaultProps . assistiveText ,
1037
- ...this . props . assistiveText ,
1038
- } ;
1039
- if ( this . props . assistiveTextForActionsHeader ) {
1040
- assistiveText . actionsHeader = this . props . assistiveTextForActionsHeader ;
1041
- }
1042
- if ( this . props . assistiveTextForSelectAllRows ) {
1043
- assistiveText . selectAllRows = this . props . assistiveTextForSelectAllRows ;
1044
- }
1045
- if ( this . props . assistiveTextForColumnSortedAscending ) {
1046
- assistiveText . columnSortedAscending = this . props . assistiveTextForColumnSortedAscending ;
1047
- }
1048
- if ( this . props . assistiveTextForColumnSortedDescending ) {
1049
- assistiveText . columnSortedDescending = this . props . assistiveTextForColumnSortedDescending ;
1050
- }
1051
- if ( this . props . assistiveTextForColumnSort ) {
1052
- assistiveText . columnSort = this . props . assistiveTextForColumnSort ;
1053
- }
1054
- if ( this . props . assistiveTextForSelectRow ) {
1055
- assistiveText . selectRow = this . props . assistiveTextForSelectRow ;
1056
- }
1108
+ const assistiveText = getAssistiveText (
1109
+ this . props . assistiveText ,
1110
+ this . props . assistiveTextForActionsHeader ,
1111
+ this . props . assistiveTextForSelectAllRows ,
1112
+ this . props . assistiveTextForColumnSortedAscending ,
1113
+ this . props . assistiveTextForColumnSortedDescending ,
1114
+ this . props . assistiveTextForColumnSort ,
1115
+ this . props . assistiveTextForSelectRow
1116
+ ) ;
1057
1117
1058
1118
if ( this . props . selectRows && this . props . selectRows !== 'radio' ) {
1059
1119
ariaProps [ 'aria-multiselectable' ] = 'true' ;
@@ -1066,26 +1126,11 @@ class DataTable extends React.Component {
1066
1126
select : canSelectRows ? this . headerRefs . select : [ ] ,
1067
1127
} ;
1068
1128
1069
- const tableContext = {
1070
- activeCell : this . state . activeCell ,
1071
- activeElement : this . state . activeElement ,
1072
- mode : this . state . mode ,
1073
- tableHasFocus : this . state . tableHasFocus ,
1074
- changeActiveCell : this . changeActiveCell ,
1075
- changeActiveElement : this . changeActiveElement ,
1076
- handleKeyDown : this . handleKeyDown ,
1077
- registerInteractiveElement : this . registerInteractiveElement ,
1078
- allowKeyboardNavigation : this . state . allowKeyboardNavigation ,
1079
- setAllowKeyboardNavigation : ( allowKeyboardNavigation ) => {
1080
- if ( this . getKeyboardNavigation ( ) ) {
1081
- this . setState ( { allowKeyboardNavigation } ) ;
1082
- }
1083
- } ,
1084
- } ;
1085
-
1086
1129
let component = (
1087
1130
< React . Fragment >
1088
- < TableContext . Provider value = { tableContext } >
1131
+ < TableContext . Provider
1132
+ value = { this . getTableContext ( this . state , this . getKeyboardNavigation ( ) ) }
1133
+ >
1089
1134
< table
1090
1135
{ ...ariaProps }
1091
1136
className = { classNames (
0 commit comments