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

Update combo box components to React 16.3 lifecycle #890

Merged
Merged
138 changes: 75 additions & 63 deletions src/components/combo_box/combo_box.js
Original file line number Diff line number Diff line change
Expand Up @@ -56,19 +56,16 @@ export class EuiComboBox extends Component {

const initialSearchValue = '';
const { options, selectedOptions } = props;
const matchingOptions = this.getMatchingOptions(options, selectedOptions, initialSearchValue);

this.state = {
matchingOptions: getMatchingOptions(options, selectedOptions, initialSearchValue, props.async),
listBounds: undefined,
searchValue: initialSearchValue,
isListOpen: false,
listPosition: 'bottom',
activeOptionIndex: undefined,
};

// Cached derived state.
this.matchingOptions = matchingOptions;
this.listBounds = undefined;

// Refs.
this.comboBox = undefined;
this.autoSizeInput = undefined;
Expand All @@ -77,12 +74,6 @@ export class EuiComboBox extends Component {
this.options = [];
}

getMatchingOptions = (options, selectedOptions, searchValue) => {
// Assume the consumer has already filtered the options against the search value.
const isPreFiltered = this.props.async;
return getMatchingOptions(options, selectedOptions, searchValue, isPreFiltered);
};

openList = () => {
this.setState({
isListOpen: true,
Expand All @@ -96,7 +87,7 @@ export class EuiComboBox extends Component {
});
};

updateListPosition = (listBounds = this.listBounds) => {
updateListPosition = (listBounds = this.state.listBounds) => {
if (!this._isMounted) {
return;
}
Expand All @@ -111,9 +102,7 @@ export class EuiComboBox extends Component {

const comboBoxBounds = this.comboBox.getBoundingClientRect();

// Cache for future calls. Assign values directly instead of destructuring because listBounds is
// a DOMRect, not a JS object.
this.listBounds = {
listBounds = {
bottom: listBounds.bottom,
height: listBounds.height,
left: comboBoxBounds.left,
Expand All @@ -124,13 +113,16 @@ export class EuiComboBox extends Component {
y: listBounds.y,
};

const { position, left, top } = calculatePopoverPosition(comboBoxBounds, this.listBounds, 'bottom', 0, ['bottom', 'top']);
const { position, left, top } = calculatePopoverPosition(comboBoxBounds, listBounds, 'bottom', 0, ['bottom', 'top']);

this.optionsList.style.top = `${top + window.scrollY}px`;
this.optionsList.style.left = `${left}px`;
this.optionsList.style.width = `${comboBoxBounds.width}px`;

// Cache for future calls. Assign values directly instead of destructuring because listBounds is
// a DOMRect, not a JS object.
this.setState({
listBounds,
width: comboBoxBounds.width,
listPosition: position,
});
Expand Down Expand Up @@ -181,42 +173,42 @@ export class EuiComboBox extends Component {

incrementActiveOptionIndex = throttle(amount => {
// If there are no options available, reset the focus.
if (!this.matchingOptions.length) {
if (!this.state.matchingOptions.length) {
this.clearActiveOption();
return;
}

let nextActiveOptionIndex;
this.setState(({ activeOptionIndex, matchingOptions }) => {
let nextActiveOptionIndex;

if (!this.hasActiveOption()) {
// If this is the beginning of the user's keyboard navigation of the menu, then we'll focus
// either the first or last item.
nextActiveOptionIndex = amount < 0 ? this.matchingOptions.length - 1 : 0;
} else {
nextActiveOptionIndex = this.state.activeOptionIndex + amount;
if (!this.hasActiveOption()) {
// If this is the beginning of the user's keyboard navigation of the menu, then we'll focus
// either the first or last item.
nextActiveOptionIndex = amount < 0 ? matchingOptions.length - 1 : 0;
} else {
nextActiveOptionIndex = activeOptionIndex + amount;

if (nextActiveOptionIndex < 0) {
nextActiveOptionIndex = this.matchingOptions.length - 1;
} else if (nextActiveOptionIndex === this.matchingOptions.length) {
nextActiveOptionIndex = 0;
if (nextActiveOptionIndex < 0) {
nextActiveOptionIndex = matchingOptions.length - 1;
} else if (nextActiveOptionIndex === matchingOptions.length) {
nextActiveOptionIndex = 0;
}
}
}

// Group titles are included in option list but are not selectable
// Skip group title options
const direction = amount > 0 ? 1 : -1;
while (this.matchingOptions[nextActiveOptionIndex].isGroupLabelOption) {
nextActiveOptionIndex = nextActiveOptionIndex + direction;
// Group titles are included in option list but are not selectable
// Skip group title options
const direction = amount > 0 ? 1 : -1;
while (matchingOptions[nextActiveOptionIndex].isGroupLabelOption) {
nextActiveOptionIndex = nextActiveOptionIndex + direction;

if (nextActiveOptionIndex < 0) {
nextActiveOptionIndex = this.matchingOptions.length - 1;
} else if (nextActiveOptionIndex === this.matchingOptions.length) {
nextActiveOptionIndex = 0;
if (nextActiveOptionIndex < 0) {
nextActiveOptionIndex = matchingOptions.length - 1;
} else if (nextActiveOptionIndex === matchingOptions.length) {
nextActiveOptionIndex = 0;
}
}
}

this.setState({
activeOptionIndex: nextActiveOptionIndex,
return { activeOptionIndex: nextActiveOptionIndex };
});
}, 200);

Expand Down Expand Up @@ -294,10 +286,10 @@ export class EuiComboBox extends Component {

doesSearchMatchOnlyOption = () => {
const { searchValue } = this.state;
if (this.matchingOptions.length !== 1) {
if (this.state.matchingOptions.length !== 1) {
return false;
}
return this.matchingOptions[0].label.toLowerCase() === searchValue.toLowerCase();
return this.state.matchingOptions[0].label.toLowerCase() === searchValue.toLowerCase();
};

areAllOptionsSelected = () => {
Expand Down Expand Up @@ -491,35 +483,55 @@ export class EuiComboBox extends Component {
}, 100);
}

// TODO: React 16.3 - ideally refactor from class members into state
// and use getDerivedStateFromProps; otherwise componentDidUpdate
componentWillUpdate(nextProps, nextState) {
static getDerivedStateFromProps(nextProps, prevState) {
const { options, selectedOptions } = nextProps;
const { searchValue } = nextState;

if (
options !== this.props.options
|| selectedOptions !== this.props.selectedOptions
|| searchValue !== this.props.searchValue
) {
// Clear refs to options if the ones we can display changes.
this.options = [];
}
const { searchValue } = prevState;

// Calculate and cache the options which match the searchValue, because we use this information
// in multiple places and it would be expensive to calculate repeatedly.
const matchingOptions = this.getMatchingOptions(options, selectedOptions, nextState.searchValue);
this.matchingOptions = matchingOptions;
const matchingOptions = getMatchingOptions(options, selectedOptions, searchValue, nextProps.async);

return { matchingOptions };
}

updateMatchingOptionsIfDifferent(newMatchingOptions) {
const { matchingOptions } = this.state;

let areOptionsDifferent = false;

if (matchingOptions.length !== newMatchingOptions.length) {
areOptionsDifferent = true;
} else {
for (let i = 0; i < matchingOptions.length; i++) {
if (matchingOptions[i].label !== newMatchingOptions[i].label) {
areOptionsDifferent = true;
break;
}
}
}

if (areOptionsDifferent) {
this.options = [];
this.setState({ matchingOptions: newMatchingOptions });

if (!matchingOptions.length) {
// Prevent endless setState -> componentWillUpdate -> setState loop.
if (nextState.hasActiveOption) {
this.clearActiveOption();
if (!newMatchingOptions.length) {
// Prevent endless setState -> componentWillUpdate -> setState loop.
if (this.state.hasActiveOption) {
this.clearActiveOption();
}
}
}
}

componentDidUpdate() {
const { options, selectedOptions } = this.props;
const { searchValue } = this.state;

// React 16.3 has a bug (fixed in 16.4) where getDerivedStateFromProps
// isn't called after a state change, and we track `searchValue` in state
// instead we need to react to a change in searchValue here
this.updateMatchingOptionsIfDifferent(getMatchingOptions(options, selectedOptions, searchValue, this.props.async));

this.focusActiveOption();
}

Expand Down Expand Up @@ -573,7 +585,7 @@ export class EuiComboBox extends Component {
selectedOptions={selectedOptions}
onCreateOption={onCreateOption}
searchValue={searchValue}
matchingOptions={this.matchingOptions}
matchingOptions={this.state.matchingOptions}
listRef={this.optionsListRef}
optionRef={this.optionRef}
onOptionClick={this.onOptionClick}
Expand Down
9 changes: 4 additions & 5 deletions src/components/combo_box/combo_box_input/combo_box_input.js
Original file line number Diff line number Diff line change
Expand Up @@ -61,9 +61,8 @@ export class EuiComboBoxInput extends Component {
});
};

// TODO: React 16.3 - componentDidUpdate
componentWillUpdate(nextProps) {
const { searchValue } = nextProps;
componentDidUpdate(prevProps) {
const { searchValue } = prevProps;

// We need to update the position of everything if the user enters enough input to change
// the size of the input.
Expand Down Expand Up @@ -150,9 +149,9 @@ export class EuiComboBoxInput extends Component {

const clickProps = {};

if (!isDisabled) {
if (!isDisabled && onClear && hasSelectedOptions) {
Copy link
Contributor

Choose a reason for hiding this comment

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

This is a lot cleaner - nice

clickProps.clear = {
onClick: hasSelectedOptions ? onClear : undefined,
onClick: onClear,
};
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -65,9 +65,8 @@ export class EuiComboBoxOptionsList extends Component {
window.addEventListener('resize', this.updatePosition);
}

// TODO: React 16.3 - componentDidUpdate
componentWillUpdate(nextProps) {
const { options, selectedOptions, searchValue } = nextProps;
componentDidUpdate(prevProps) {
const { options, selectedOptions, searchValue } = prevProps;

// We don't compare matchingOptions because that will result in a loop.
if (
Expand Down