Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Exchange defaultProps for default parameters in function components #1226

Open
wants to merge 14 commits into
base: main
Choose a base branch
from
Open
Original file line number Diff line number Diff line change
Expand Up @@ -16,20 +16,22 @@ export interface SelectTreeProps<T> extends CheckboxTreeProps<T> {
}

function SelectTree<T>(props: SelectTreeProps<T>) {
const [buttonDisplayContent, setButtonDisplayContent] = useState<ReactNode>(
props.currentList && props.currentList.length
? props.currentList.join(', ')
: props.buttonDisplayContent
);
const {
selectedList,
onSelectionChange,
shouldCloseOnSelection,
wrapPopover,
currentList,
selectedList = [],
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think we need this. Isn't selectedList an option prop for CheckboxTree?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is but we use selectedList below to auto close the popover.

setKey(selectedList.join(', '));

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(line 45 ish)

onSelectionChange,
hasPopoverButton = true,
instantUpdate = true,
wrapPopover,
} = props;

const [buttonDisplayContent, setButtonDisplayContent] = useState<ReactNode>(
currentList && currentList.length
? currentList.join(', ')
: props.buttonDisplayContent
);

// This local state is updated whenever a checkbox is clicked in the species tree.
// When `instantUpdate` is false, pass the final value to `onSelectionChange` when the popover closes.
// When it is true we call `onSelectionChange` whenever `localSelectedList` changes
Expand All @@ -47,8 +49,9 @@ function SelectTree<T>(props: SelectTreeProps<T>) {
// live updates to caller when needed
useEffect(() => {
if (!instantUpdate) return;
if (!onSelectionChange) return;
onSelectionChange(localSelectedList);
}, [onSelectionChange, localSelectedList]);
}, [onSelectionChange, localSelectedList, instantUpdate]);

function truncatedButtonContent(selectedList: string[]) {
return (
Expand All @@ -72,48 +75,15 @@ function SelectTree<T>(props: SelectTreeProps<T>) {
? truncatedButtonContent(localSelectedList)
: props.buttonDisplayContent
);
if (!instantUpdate) onSelectionChange(localSelectedList);
if (!instantUpdate && onSelectionChange)
onSelectionChange(localSelectedList);
};

const checkboxTree = (
<CheckboxTree
tree={props.tree}
getNodeId={props.getNodeId}
getNodeChildren={props.getNodeChildren}
onExpansionChange={props.onExpansionChange}
shouldExpandDescendantsWithOneChild={
props.shouldExpandDescendantsWithOneChild
}
shouldExpandOnClick={props.shouldExpandOnClick}
showRoot={props.showRoot}
renderNode={props.renderNode}
expandedList={props.expandedList}
isSelectable={props.isSelectable}
{...props}
selectedList={localSelectedList}
filteredList={props.filteredList}
customCheckboxes={props.customCheckboxes}
isMultiPick={props.isMultiPick}
name={props.name}
onSelectionChange={setLocalSelectedList}
currentList={props.currentList}
defaultList={props.defaultList}
isSearchable={props.isSearchable}
autoFocusSearchBox={props.autoFocusSearchBox}
showSearchBox={props.showSearchBox}
searchBoxPlaceholder={props.searchBoxPlaceholder}
searchIconName={props.searchIconName}
searchBoxHelp={props.searchBoxHelp}
searchTerm={props.searchTerm}
onSearchTermChange={props.onSearchTermChange}
searchPredicate={props.searchPredicate}
renderNoResults={props.renderNoResults}
linksPosition={props.linksPosition}
additionalActions={props.additionalActions}
additionalFilters={props.additionalFilters}
isAdditionalFilterApplied={props.isAdditionalFilterApplied}
wrapTreeSection={props.wrapTreeSection}
styleOverrides={props.styleOverrides}
customTreeNodeCssSelectors={props.customTreeNodeCssSelectors}
/>
);

Expand Down
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It occurred to me that you could do something like this in the component:

function CheckboxTree<T>(partialProps: Props<T>) {
  const props = { ...defaultProps, ...partialProps };

  // ...

}

If a prop is defined in partialProps, it's value will be used; otherwise, the value in defaultProps will be used.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This would be so much nicer. I tried this for a bit but could not shake a missing in types validation default props error. Even after ensuring props was of the correct type.
Happy to try something else if you have an idea of how to get around this!

Original file line number Diff line number Diff line change
Expand Up @@ -162,7 +162,7 @@ export type CheckboxTreeProps<T> = {
isSelectable?: boolean;

/** List of selected nodes as represented by their ids, defaults to [ ] */
selectedList: string[];
selectedList?: string[];

/**
* List of filtered nodes as represented by their ids used to determine isLeafVisible node status.
Expand All @@ -182,7 +182,7 @@ export type CheckboxTreeProps<T> = {

/** Takes array of ids, thus encapsulates:
selectAll, clearAll, selectDefault, selectCurrent (i.e. reset) */
onSelectionChange: ChangeHandler;
onSelectionChange?: ChangeHandler;

/** List of “current” ids, if omitted (undefined or null), then don’t display link */
currentList?: string[];
Expand All @@ -193,7 +193,7 @@ export type CheckboxTreeProps<T> = {
//%%%%%%%%%%% Properties associated with search %%%%%%%%%%%

/** Indicates whether this is a searchable CBT. If so, then show boxes and respect the optional parameters below, also turn off expansion; default to false */
isSearchable: boolean;
isSearchable?: boolean;

/** Indicates if the search box should have autoFocus set to true */
autoFocusSearchBox?: boolean;
Expand All @@ -202,7 +202,7 @@ export type CheckboxTreeProps<T> = {
showSearchBox?: boolean;

/** PlaceHolder text; shown in grey if searchTerm is empty */
searchBoxPlaceholder: string;
searchBoxPlaceholder?: string;

/** Name of icon to show in search box */
searchIconName?: 'search' | 'filter';
Expand All @@ -214,13 +214,13 @@ export type CheckboxTreeProps<T> = {
searchBoxHelp?: string;

/** Current search term; if non-empty, expandability is disabled */
searchTerm: string;
searchTerm?: string;

/** Takes single arg: the new search text. Called when user types into the search box */
onSearchTermChange: (term: string) => void;
onSearchTermChange?: (term: string) => void;

/** Takes (node, searchTerms) and returns boolean. searchTerms is a list of query terms, parsed from the original input string. This function returns a boolean indicating if a node matches search criteria and should be shown */
searchPredicate: (node: T, terms: string[]) => boolean;
searchPredicate?: (node: T, terms: string[]) => boolean;

renderNoResults?: (searchTerm: string, tree: T) => React.ReactNode;

Expand All @@ -246,6 +246,31 @@ export type CheckboxTreeProps<T> = {
customTreeNodeCssSelectors?: object;
};

// Default values. Used across multiple functions in this file.
// Define defaultCheckboxTreeProps as a partial of CheckboxTreeProps

const defaultCheckboxTreeProps = {
showRoot: false,
expandedList: null,
isSelectable: false,
selectedList: [],
customCheckboxes: {},
isMultiPick: true,
onSelectionChange: () => {
/* */
},
isSearchable: false,
showSearchBox: true,
searchBoxPlaceholder: 'Search...',
searchBoxHelp: '',
searchTerm: '',
onSearchTermChange: () => {
/* */
},
searchPredicate: () => true,
linksPosition: LinksPosition.Both,
};

type TreeLinkHandler = MouseEventHandler<HTMLButtonElement>;

type TreeLinksProps = {
Expand Down Expand Up @@ -463,7 +488,7 @@ function applyPropsToStatefulTree<T>(
isSearchable: CheckboxTreeProps<T>['isSearchable'],
isMultiPick: CheckboxTreeProps<T>['isMultiPick'],
searchTerm: CheckboxTreeProps<T>['searchTerm'],
selectedList: CheckboxTreeProps<T>['selectedList'],
selectedList: CheckboxTreeProps<T>['selectedList'] = defaultCheckboxTreeProps.selectedList,
propsExpandedList: CheckboxTreeProps<T>['expandedList'],
isAdditionalFilterApplied: CheckboxTreeProps<T>['isAdditionalFilterApplied'],
isLeafVisible: (id: string) => boolean,
Expand Down Expand Up @@ -589,8 +614,8 @@ function applyPropsToStatefulTree<T>(
*/
function isActiveSearch<T>(
isAdditionalFilterApplied: CheckboxTreeProps<T>['isAdditionalFilterApplied'],
isSearchable: CheckboxTreeProps<T>['isSearchable'],
searchTerm: CheckboxTreeProps<T>['searchTerm']
isSearchable: CheckboxTreeProps<T>['isSearchable'] = defaultCheckboxTreeProps.isSearchable,
searchTerm: CheckboxTreeProps<T>['searchTerm'] = defaultCheckboxTreeProps.searchTerm
) {
return isSearchable && isFiltered(searchTerm, isAdditionalFilterApplied);
}
Expand Down Expand Up @@ -625,12 +650,12 @@ function isFiltered(searchTerm: string, isAdditionalFilterApplied?: boolean) {
*/
function createIsLeafVisible<T>(
tree: CheckboxTreeProps<T>['tree'],
searchTerm: CheckboxTreeProps<T>['searchTerm'],
searchPredicate: CheckboxTreeProps<T>['searchPredicate'],
searchTerm: CheckboxTreeProps<T>['searchTerm'] = defaultCheckboxTreeProps.searchTerm,
searchPredicate: CheckboxTreeProps<T>['searchPredicate'] = defaultCheckboxTreeProps.searchPredicate,
getNodeId: CheckboxTreeProps<T>['getNodeId'],
getNodeChildren: CheckboxTreeProps<T>['getNodeChildren'],
isAdditionalFilterApplied: CheckboxTreeProps<T>['isAdditionalFilterApplied'],
isSearchable: CheckboxTreeProps<T>['isSearchable'],
isSearchable: CheckboxTreeProps<T>['isSearchable'] = defaultCheckboxTreeProps.isSearchable,
filteredList: CheckboxTreeProps<T>['filteredList']
) {
// if not searching, if no additional filters are applied, and if filteredList is undefined, then all nodes are visible
Expand Down Expand Up @@ -695,32 +720,32 @@ function CheckboxTree<T>(props: CheckboxTreeProps<T>) {
tree,
getNodeId,
getNodeChildren,
searchTerm,
selectedList,
searchTerm = defaultCheckboxTreeProps.searchTerm,
selectedList = defaultCheckboxTreeProps.selectedList,
currentList,
defaultList,
isSearchable,
isSearchable = defaultCheckboxTreeProps.isSearchable,
isAdditionalFilterApplied,
name,
shouldExpandDescendantsWithOneChild,
onExpansionChange,
isSelectable,
isMultiPick,
onSelectionChange,
showRoot,
onSelectionChange = defaultCheckboxTreeProps.onSelectionChange,
isSelectable = defaultCheckboxTreeProps.isSelectable,
isMultiPick = defaultCheckboxTreeProps.isMultiPick,
showRoot = defaultCheckboxTreeProps.showRoot,
additionalActions,
linksPosition = LinksPosition.Both,
showSearchBox,
linksPosition = defaultCheckboxTreeProps.linksPosition,
showSearchBox = defaultCheckboxTreeProps.showSearchBox,
autoFocusSearchBox,
onSearchTermChange,
searchBoxPlaceholder,
onSearchTermChange = defaultCheckboxTreeProps.onSearchTermChange,
searchBoxPlaceholder = defaultCheckboxTreeProps.searchBoxPlaceholder,
searchIconName,
searchIconPosition,
searchBoxHelp,
searchBoxHelp = defaultCheckboxTreeProps.searchBoxHelp,
additionalFilters,
wrapTreeSection,
shouldExpandOnClick = true,
customCheckboxes,
customCheckboxes = defaultCheckboxTreeProps.customCheckboxes,
renderNoResults,
styleOverrides = {},
customTreeNodeCssSelectors = {},
Expand Down Expand Up @@ -764,7 +789,7 @@ function CheckboxTree<T>(props: CheckboxTreeProps<T>) {
* Creates a function that will handle selection-related tree link clicks
*/
function createSelector(listFetcher: ListFetcher) {
return createLinkHandler(listFetcher, props.onSelectionChange);
return createLinkHandler(listFetcher, onSelectionChange);
}

// define event handlers related to expansion
Expand Down Expand Up @@ -1071,29 +1096,6 @@ function defaultRenderNoResults() {
);
}

const defaultProps = {
showRoot: false,
expandedList: null,
isSelectable: false,
selectedList: [],
customCheckboxes: {},
isMultiPick: true,
onSelectionChange: () => {
/* */
},
isSearchable: false,
showSearchBox: true,
searchBoxPlaceholder: 'Search...',
searchBoxHelp: '',
searchTerm: '',
onSearchTermChange: () => {
/* */
},
searchPredicate: () => true,
linksPosition: LinksPosition.Both,
};

CheckboxTree.defaultProps = defaultProps;
CheckboxTree.LinkPlacement = LinksPosition;
export default CheckboxTree;

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,14 @@ const cx = makeClassNameHelper('field-detail');
* Main interactive filtering interface for a particular field.
*/
function FieldFilter(props) {
const { displayName = 'Items' } = props;

let className = cx('', props.hideFieldPanel && 'fullWidth');

return (
<div className={className}>
{!props.activeField ? (
<EmptyField displayName={props.displayName} />
<EmptyField displayName={displayName} />
) : (
<React.Fragment>
<h3>
Expand Down Expand Up @@ -48,9 +50,9 @@ function FieldFilter(props) {
props.activeFieldState.summary == null &&
props.activeFieldState.leafSummaries == null) ||
props.dataCount == null ? null : isMulti(props.activeField) ? (
<MultiFieldFilter {...props} />
<MultiFieldFilter {...props} displayName={displayName} />
) : (
<SingleFieldFilter {...props} />
<SingleFieldFilter {...props} displayName={displayName} />
)}
</React.Fragment>
)}
Expand Down Expand Up @@ -95,8 +97,4 @@ FieldFilter.propTypes = {
selectByDefault: PropTypes.bool.isRequired,
};

FieldFilter.defaultProps = {
displayName: 'Items',
};

export default FieldFilter;
Original file line number Diff line number Diff line change
Expand Up @@ -10,15 +10,26 @@ import FieldFilter from '../../Components/AttributeFilter/FieldFilter';
* Filtering UI for server-side filtering.
*/
function ServerSideAttributeFilter(props) {
var { displayName, fieldTree, hideFilterPanel, hideFieldPanel } = props;
const {
fieldTree,
displayName = 'Items',
hideFilterPanel = false,
hideFieldPanel = false,
hideGlobalCounts = false,
selectByDefault = false,
histogramScaleYAxisDefault = true,
histogramTruncateYAxisDefault = false,
} = props;

if (fieldTree == null) {
return <h3>Data is not available for {displayName}.</h3>;
}

return (
<div style={{ overflowX: 'auto', marginRight: '1em' }}>
{hideFilterPanel || <FilterList {...props} />}
{hideFilterPanel || (
<FilterList {...props} hideGlobalCounts={hideGlobalCounts} />
)}

{/* Main selection UI */}
<div className="filters ui-helper-clearfix">
Expand Down Expand Up @@ -79,16 +90,6 @@ ServerSideAttributeFilter.propTypes = {
onRangeScaleChange: PropTypes.func.isRequired,
};

ServerSideAttributeFilter.defaultProps = {
displayName: 'Items',
hideFilterPanel: false,
hideFieldPanel: false,
hideGlobalCounts: false,
selectByDefault: false,
histogramScaleYAxisDefault: true,
histogramTruncateYAxisDefault: false,
};

export default wrappable(ServerSideAttributeFilter);

export function withOptions(options) {
Expand Down
Loading
Loading