I am dynamically creating list of inputs with React and Redux. After clicking a button an input is added to the end of list. I need to focus last added input. I tried this code but it focuses penultimate input
const mapDispatchToProps = (dispatch, ownProps) => ({
onOptionsChange: (newOptions) => {
dispatch(formActions.updateOptions(newOptions));
}
});
...
this.props.onOptionsChange({ ...this.props, inputsList}); // change list of inputs
ReactDOM.findDOMNode(this.inputs[this.props.choices.length - 1]).focus();
In logs I can see that focus() is executed before props from state are updated. How can I wait for dispatch to finish?
I am dynamically creating list of inputs with React and Redux. After clicking a button an input is added to the end of list. I need to focus last added input. I tried this code but it focuses penultimate input
const mapDispatchToProps = (dispatch, ownProps) => ({
onOptionsChange: (newOptions) => {
dispatch(formActions.updateOptions(newOptions));
}
});
...
this.props.onOptionsChange({ ...this.props, inputsList}); // change list of inputs
ReactDOM.findDOMNode(this.inputs[this.props.choices.length - 1]).focus();
In logs I can see that focus() is executed before props from state are updated. How can I wait for dispatch to finish?
Share Improve this question asked Dec 28, 2016 at 16:52 igoigo 6,8686 gold badges45 silver badges51 bronze badges 2-
can you add the
autoFocus
prop to your input? – BravoZulu Commented Dec 28, 2016 at 17:02 - autoFocus look like an interesting choice – FlatLander Commented Jan 9, 2017 at 1:58
4 Answers
Reset to default 4Promisify your dispatch
Lets write a middleware that returns a promise after pleting the dispatch.
const thenMiddleware = store => next => action => {
return new Promise((resolve, reject) => {
try {
resolve(next(action));
} catch(e) {
reject(e);
}
})
};
Prepare your store first, by installing the middleware
import { createStore, applyMiddleware } from 'redux';
import rootReducer from './reducers/index';
/**
* Our star of the show
*/
const thenMiddleware = store => next => action => {
return new Promise((resolve, reject) => {
try {
resolve(next(action));
} catch(e) {
reject(e);
}
})
};
const store = createStore(
rootReducer,
applyMiddleware(thenMiddleware)
);
Now we have to return the dispatch inside our mapDispatchToProps method
const mapDispatchToProps = (dispatch, ownProps) => ({
onOptionsChange: (newOptions) => {
return dispatch(formActions.updateOptions(newOptions));
}
});
Finally hook your event on plete
Now that you know which input you created last time, you can attach an on plete function to this dispatch.
this.props.onOptionsChange(newOptions).then(() => {
// do your input focusing here
});
I would implement ponentDidUpdate
and check the length of your "input-array" or whatever data-structure you are using:
ponentDidUpdate(prevProps, prevState) {
if (prevProps.choices.length < this.props.choices.length) {
ReactDOM.findDOMNode(this.inputs[this.props.choices.length - 1]).focus();
}
}
Create new wrapping ponent for input
element. That ponent will autofocus, when rendered.
class Input extends React.Component {
keepRef = (ref) => {
this.ref = ref;
}
ponentDidMount() {
this.ref.focus();
}
render () {
return (
<input ref={this.keepRef} {...this.props} />
)
}
}
If you don't want Input
ponent to handle focus, move logic to it's parent.
I faced a similiar problem, wherein, I had a list of items and I need to focus on the last element whenever a new item is added. The below code is using flux architecture but i guess it should solve your problem.
addNewItem: function() {
var self = this;
var list = this.state.list || {};
var listOrder = this.state.listOrder || [];
var newKey = "some_random_key";
list[newKey] = {
"product": ""
};
listOrder.push(newKey);
this.state.list = list;
this.state.listOrder = listOrder;
this.setState({list: this.state.list, listOrder: this.state.listOrder, showSaveButton: true});
setTimeout(function(){
$(".buy-form-item-list-row .buy-form-item-list-item-name").last().focus();
}, 0);
},
So in the above code.. listOrder contains the keys of list of items. Eg: list is an object which contain the item. Eg:
listOrder=[ran_1, ran_2];
list={ ran_1: { product: "Eggs - 5" }, ran_2: { product: "Bread - 1" } }
render method, iterate over the listOrder and creates a Input tag with className="buy-form-item-list-item-name" for each item.
The trick to focus on the last item is the setTimeout with an interval of 0. When you call setState it does not trigger render function at that moment itself, instead it waits for the function to execute pletely.
Hope this helps.