ReactJS - ссылка не работает с connect и redux-формой
У меня проблема при использовании ref
через соединение и редукс-форму вместе.
Структура компонента такова, что у ChildComponent есть формы, и я использую
class EditEvent extends Component {
constructor (props) {
super(props);
this.handleStartDateChange = this.handleStartDateChange.bind(this);
this.handleEndDataChange = this.handleEndDataChange.bind(this);
this.state = {loading: false};
}
componentDidMount(){
this.props.fetchEvent(this.props.session,true);
this.props.fetchScheduleDates(this.props.session);
}
submitSettings(){
this.props.form.submit();
}
handleStartDateChange(date) {
this.props.updateUnsavedChanges();
this.props.dateRangeUpdate({startDate:date,endDate:date,allDate:{}});
}
handleEndDataChange(date) {
this.props.updateUnsavedChanges();
this.props.dateRangeUpdate({startDate:this.props.scheduledates.startDate,endDate:date,allDate:{}});
}
renderBlockList(startDate,endDate,allDate){
/*if(isEmpty(allDate)){
return 'Please select date duration first.';
}*/
var arr = [];
var day = {};
var n = endDate.diff(startDate,'days')+1;
for (var i=1; i <= n; i++ ) {
if(i !== 1){
day = moment(day).add(1,'days');
} else {
day = startDate;
}
var activeDate = _.find(allDate, {scheduledate:moment(day).format('YYYY-MM-DD')});
arr.push(<DateBlock key={i} day={day} activeDate={activeDate}/>);
}
return arr;
}
handleEventName(){
this.props.updateUnsavedChanges();
}
handleEventDesc(){
this.props.updateUnsavedChanges();
}
onFormSubmit(values){
debugger;
if (this.props.selectedFeatures.length === 0) {
this.props.showError('Select at least one feature!');
} else if (values.eventname === '') {
this.props.showError('Please enter a name');
} else {
this.props.submitSettingsHandler(values);
}
}
render(){
var eventStartDate = moment();
var eventEndDate = moment().add(6,'days');
var isDisabled = false;
var allDate = [];
if(this.props.scheduledates.startDate !== undefined){
eventStartDate = this.props.scheduledates.startDate;
eventEndDate = this.props.scheduledates.endDate;
allDate = this.props.allDate;
}
if(this.props.scheduledates.isEditable !== undefined){
isDisabled = true;
}
var {handleSubmit, pristine, submitting, invalid} = this.props;
return(
<form onSubmit={handleSubmit(this.onFormSubmit.bind(this))} encType="multipart/form-data">
<div className="general-tab-warp">
<Field
name="eventname"
label="Event name"
placeholder="E.g google I/O 2017"
type="text"
component={this.renderFieldText}
bsClass="row"
formdiv="false"
labelposition={LABEL_POSITION_TOP}
onChange={this.handleEventName.bind(this)}
/>
<Field
name="eventdesc"
label="Event Description"
placeholder="Enter event description"
type="text"
component={this.renderFieldTextarea}
bsClass="row"
formdiv="false"
labelposition={LABEL_POSITION_TOP}
onChange={this.handleEventDesc.bind(this)}
/>
<div className="form-group row">
<label className="control-label-top col-sm-12">Duration*</label>
<div className="col-sm-12">
<Row>
<Col md={6} style={{width:'48%'}}>
<DatePicker
selected = {eventStartDate}
className = "form-control"
dateFormat = "DD/MM/YYYY"
onChange = {this.handleStartDateChange}
name = "startdate"
selectsStart
startDate = {eventStartDate}
endDate = {eventEndDate}
minDate = {moment().add(1,'days')}
disabled={isDisabled}
/>
</Col>
<Col md={1} style={{padding:'0px',marginTop:'6px', width:'4%'}}>➔</Col>
<Col md={6} style={{width:'48%'}}>
<DatePicker
selected = {eventEndDate}
className = "form-control"
dateFormat = "DD/MM/YYYY"
onChange = {this.handleEndDataChange}
name = "enddate"
minDate = {eventStartDate}
maxDate = {moment(eventStartDate).add(6, "days")}
selectsEnd
startDate = {eventStartDate}
endDate = {eventEndDate}
disabled={isDisabled}
/>
</Col>
</Row>
<div className="duration-info">
* You can't update the date after publishing the app.
</div>
</div>
</div>
<div className="form-group row">
<label className="control-label-top col-sm-12">Days</label>
<div className="col-sm-12">
<ul className="event-days">
{this.renderBlockList(eventStartDate,eventEndDate,allDate)}
</ul>
</div>
</div>
<div className="form-group row">
<label className="control-label-top col-sm-12">Event Privacy</label>
<div className="col-md-12">
<FormGroup>
<Radio name="event_privacy" inline>
Public
</Radio>{' '}
<Radio name="event_privacy" inline>
Private
</Radio>
</FormGroup>
</div>
</div>
</form>
);
}
}
function mapStateToProps(state){
return{
features: state.features,
session: state.session,
initialValues: state.eventData[0],
scheduledates:state.scheduledates,
allDate:state.updateScheduleDates,
selectedFeatures: state.selectedFeatures
};
}
function validate(values) {
var errors = {};
if (values['eventname'] === '') {
errors['eventname'] = 'Event name is required';
}
return errors;
}
export default connect(mapStateToProps,{fetchEvent,updateUnsavedChanges,fetchScheduleDates,dateRangeUpdate,patchEventDetails,showError,showSuccess},null,{withRef:true})(reduxForm({
validate,
form:'EditEventForm',
enableReinitialize: true,
keepDirtyOnReinitialize: true
})(EditEvent));
Компонент имеет функцию
submitSettings(){
this.props.form.submit();
}
Я хочу вызвать submitSettings из родительского компонента, в котором я использую ref, см. Ниже
constructor(props){
super(props);
this.editEventRef = React.createRef();
}
onSaveClick(){
this.editEventRef.current.submitSettings();
}
render(){
return(
<EditEvent ref={this.editEventRef}/>
)
}
Когда вызывается onSaveClick, я всегда получаю следующую ошибку
Uncaught TypeError: this.editEventRef.current.submitSettings не является функцией
Ошибка идет, и функция работает нормально, как только я изменяю следующую строку и удаляю использование reduxForm()
в connect()
export default
connect(mapStateToProps,
{fetchEventDetails}
,null,{withRef:true})(EditEvent);
Может кто-нибудь, пожалуйста, направьте меня, чтобы решить эту проблему? Я что-то пропустил?
Спасибо за помощь.
Обновление Это npm-версии пакетов, которые я использую, если это поможет.
"реагировать": "^16.7.0" "реагировать-дом": "^16.7.0", "реагировать-редукс": "^5.1.1", "редукс-форма": "^6.7.0",
1 ответ
Итак, потратив так много времени, я нашел решение, которое верно для "реакции": "^16.7.0" и "избыточная форма": "^8.1.0", который является самым последним прямо сейчас.
Когда вы используете connect() reduxForm()
как я делаю в вопросе, нужно использовать
this.editEventRef.current.wrappedInstance.submitSettings();
Поэтому, когда в дочернем компоненте есть reduxForm, вам нужно использовать .current.wrappedInstance для доступа к обратному вызову, иначе используйте только .current.
Вам нужно использовать {forwardRef:true}
как опция в connect()
Я не нашел это нигде документально, но выше работает.
Надеюсь, это поможет кому-то, ищущему подобный ответ.
Приветствия.