开发者问题收集

无法在 React 中将值设置为只读属性

2019-08-29
3413

我有两个不同的单选按钮,它们在 onChange 时触发一个函数。 我有两个属性:selectedLabel 和 selectedTestdata。 我想在触发函数时更改这些属性的值。

我收到一条错误消息,提示我无法更改只读属性。

这是我的代码:

import React, { useState, Component } from 'react';
import { connect } from 'react-redux';
import { Link } from 'react-router-dom';
import './styles.css'
import { bindActionCreators } from 'redux';
import PropTypes from 'prop-types';
import { testdataRequest } from '../../actions/testdata'
import { labelsRequest } from '../../actions/labels'
....

export class TemplateDetails extends Component {

    componentDidMount() {
        this.props.testdataRequestConnect(this.props.match.params.templateid, this.props.match.params.locale);
        this.props.labelsRequestConnect(this.props.match.params.templateid);
    }

    testDataReloader = (e) => {
        this.props.selectedTestdata = e.target.value;
        console.log(e.target.value);
    }

    labelsReloader = event => {
        console.log(JSON.parse(event.target.value));
    }
    

    renderTestdata() {
        const { testdata } = this.props;
        if (!testdata.length) {
            return (
                <span
                    data-at="no-customers"
                    className="pos-v-center pos-left pos-right pos-absolute t-center t-light-grey">
                    <span className="d-block t-1 t-bold">No testdata found</span>
                </span>
            );
        }

        return (
            <div
                style={{ flexGrow: '1', flexShrink: '1' }}
                className="t-left ox-hidden oy-hidden"
                data-at="customer-search-list">
                {testdata.map((data, index) => (
                    <div key={data}>
                        <div className="form-element type-radio">
                            <span className="box-toggler">
                                <input onChange={this.testDataReloader} value={data} type="radio" name="radio-testdata" className="form-input" id={index}/>
                                <span className="indicator"></span>
                            </span>
                            <label htmlFor={index} className="form-placeholder">{data}</label>
                            <span className="icon icon--sm pos-absolute pos-right pos-v-center">
                                <svg title="icon--arrow-small-right" role="img">
                                    <use xmlnsXlink="http://www.w3.org/1999/xlink" xlinkHref="#icon--arrow-small-right" />
                                </svg>
                            </span>
                        </div>
                    </div>
                ))}
            </div>
        );
    }

    renderLabels() {
        const { labels } = this.props;
        if (!labels.length) {
            return (
                <span
                    data-at="no-customers"
                    className="pos-v-center pos-left pos-right pos-absolute t-center t-light-grey">
                    <span className="d-block t-1 t-bold">No labels found</span>
                </span>
            );
        }

        return (
            <div
                style={{ flexGrow: '1', flexShrink: '1' }}
                className="t-left"
                data-at="customer-search-list">
                {labels.map((data, index) => (
                    <div key={data.uuid}>
                        <div className="form-element type-radio">
                            <span className="box-toggler">
                                <input onChange={this.labelsReloader} value={JSON.stringify(data)} type="radio" name="radio-labels" className="form-input" id={data.uuid}/>
                                <span className="indicator"></span>
                            </span>
                            <label htmlFor={data.uuid} className="form-placeholder">{data.values.creationTime}</label>
                            <span className="icon icon--sm pos-absolute pos-right pos-v-center">
                                <svg title="icon--arrow-small-right" role="img">
                                    <use xmlnsXlink="http://www.w3.org/1999/xlink" xlinkHref="#icon--arrow-small-right" />
                                </svg>
                            </span>
                        </div>
                    </div>
                ))}
            </div>
        );
    }

    render() {
        const { isFetchingTestdata, errorTestdata, isFetchingLabels, errorLabels } = this.props;
        return (
            <div className="pos-relative h-100 o-hidden">
                <div className="d-flex h-100 w-100 pos-relative">
                    <div className="col-4 p-0">
                        <div className="d-flex flex-column h-100 pl-4 pb-4">
                            <div
                                className="px-2 mt-4 w-100"
                                style={{ maxWidth: '500px' }}>
                                <Link to="/templates">
                                    <Button variant="link">
                                        <span className="t-1" style={{ lineHeight: '1' }}>&lsaquo;</span> Templates
                                    </Button>
                                </Link>
                            </div>

                            {/* Test data */}
                            <div className="d-flex flex-wrap align-content-start py-2 pos-relative bg-white shadow t-center mt-4 p-4 w-100 h-100" style={{ flexGrow: '1', overflowY: 'scroll' }}>
                                <div className="w-100" style={{ height: 'fit-content' }}>
                                    {errorTestdata && <Alert variant="error">{errorTestdata}</Alert>}
                                    <h1>Test data</h1>
                                    <p className="t-grey mb-4">
                                        Available testdata
                                        <span className="d-block">Select one to render the template with it</span>
                                    </p>
                                </div>
                                {isFetchingTestdata ? <span className="throbber" /> : this.renderTestdata()}
                            </div>

                            {/* Labels */}
                            <div className="d-flex flex-wrap py-2 pos-relative bg-white shadow t-center mt-4 p-4 w-100 h-100" style={{ flexGrow: '1', overflowY: 'scroll' }}>
                                <div className="w-100">
                                    <h1>Labels</h1>
                                    <p className="t-grey mb-4">
                                        {errorLabels && <Alert variant="error">{errorLabels}</Alert>}
                                        Available labels
                                        <span className="d-block">Select one to render the template with them</span>
                                    </p>
                                </div>
                                {isFetchingLabels ? <span className="throbber" /> : this.renderLabels()}
                            </div>

                        </div>
                    </div>
                    <div className="col-8 mr-4">
                        <div className="d-flex flex-column h-100 pl-4 pb-4 pr-4">
                            <div className="d-flex flex-column bg-white shadow t-left p-4 mt-4 w-100 h-100" style={{ minHeight: '20px', maxHeight: '70px' }}>
                                <div className="w-100 h-100 mb-2">
                                    <h1 className="mb-2">Subject </h1>
                                </div>
                            </div>
                            <div className="d-flex flex-column bg-white shadow t-center p-4 mt-4 w-100 h-100">
                                <div>
                                    <h1>Template</h1>
                                    <p className="t-grey mb-4">
                                        Live preview of the template
                                    </p>
                                </div>
                            </div>
                        </div>

                    </div>
                </div>
            </div>


        );
    }
}

TemplateDetails.propTypes = {
    isFetchingTestdata: PropTypes.bool,
    isFetchingLabels: PropTypes.bool,
    errorTestdata: PropTypes.string,
    errorLabels: PropTypes.string,
    selectedLabel: PropTypes.string,
    selectedTestdata: PropTypes.string,
    templatesRequestConnect: PropTypes.func,
    labelsRequestConnect: PropTypes.func,
    testdata: PropTypes.array,
    labels: PropTypes.array,
};

TemplateDetails.defaultProps = {
    isFetchingTestdata: false,
    isFetchingLabels: false,
    errorTestdata: '',
    errorLabel: '',
    selectedLabel: '',
    selectedTestdata: '',
    templatesRequestConnect: () => { },
    labelsRequestConnect: () => { },
    testdata: [],
    labels: [],
};

export default connect(
    (state) => ({
        isFetchingTestdata: state.testdataStore.isFetching,
        errorTestdata: state.testdataStore.error,
        testdata: state.testdataStore.testdata,
        isFetchingLabels: state.labelsStore.isFetching,
        errorLabels: state.labelsStore.error,
        labels: state.labelsStore.labels,
    }),
    (dispatch) => bindActionCreators({
        testdataRequestConnect: testdataRequest,
        labelsRequestConnect: labelsRequest,
    }, dispatch),
)(TemplateDetails);
2个回答

props 是只读的 ,而您尝试分配给它:

testDataReloader = e => {
  // this.props.selectedTestdata = e.target.value;
  console.log(e.target.value);
};

关于为什么您无法更改 props, 在 Stack Overflow 上有很多类似的讨论。

Dennis Vash
2019-08-29

您无法直接更改 props,因为 props 是只读的。要实现您的结果,您必须应用不同的方法。

您可以做的一件事是,您可以将 selectedTestdata 和另一件事从父组件传递为 props,这将负责更改您的父状态值。

例如。在父组件

<Child selectedTestdata={this.state.selectedTestdata} udpatedTestData={this.updateTestData}/>

和父组件中,您可以定义方法 updateTestData 来更改 selectedTestdata 的值。

在您的子组件中,您可以调用此 prop 方法

testDataReloader = (e) => {
    this.props.updateTestData(e.target.value);
    console.log(e.target.value);
}
Prabhat Kumar
2019-08-29