最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Focus input after redux dispatch finished - Stack Overflow

programmeradmin1浏览0评论

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
Add a ment  | 

4 Answers 4

Reset to default 4

Promisify 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.

发布评论

评论列表(0)

  1. 暂无评论