试图在其他类中重用组件并获得错误:“警告:setState(...):只能更新已安装或安装的>组件。”

时间:2018-03-02 22:12:02

标签: javascript reactjs

我正在整理一个小POC,其中一个正在执行搜索功能。我们的想法是“搜索”将负责以下事项:
  - 显示搜索输入表格(例如,文本,日期和位置参数)
  - 点击后端AWS Lambda搜索API   - 将结果对象返回到搜索对象
  - 能够在多个页面上重复使用

我有两个不同的页面,我想利用“搜索”功能,但以不同的方式呈现结果。

搜索组件/表单独立运行,但我无法弄清楚如何将其嵌入其他网页。每当我尝试在“搜索表单”中输入任何内容时,控制台都会抛出以下错误:

  

index.js:2178警告:setState(...):只能更新已安装或安装的>零件。这通常意味着您在已卸载的组件上调用了setState()。 >这是一个无操作。

     

请检查搜索组件的代码。

下面是“搜索”代码,以及显示结果的页面开头。我是前端开发人员的新手,所以我可能在这里做一些愚蠢的事情......寻找关于我做错了什么的输入!谢谢你的帮助。

import React, { Component } from "react";
import {FormGroup, FormControl, ControlLabel } from "react-bootstrap";
import LoaderButton from "../components/LoaderButton";
import "./Home.css";
import DatePicker from 'react-datepicker';
import moment from 'moment';
import PlacesAutocomplete from 'react-places-autocomplete';
import { searchAssets } from "../libs/awsLib";

import 'react-datepicker/dist/react-datepicker.css';

// CSS Modules, react-datepicker-cssmodules.css
import 'react-datepicker/dist/react-datepicker-cssmodules.css';


export default class Search extends Component {
    constructor(props) {
        super(props);

        this.state = {
            isLoading: false,
            freeText: "",
            startDate: null,
            endDate: null,
            radius: 5,
            address: '',
            searchResults: null
        };

        this.handleChange = this.handleChange.bind(this);
        this.handleSubmit = this.handleSubmit.bind(this);
        this.handleStartDateChange = this.handleStartDateChange.bind(this);
        this.handleEndDateChange = this.handleEndDateChange.bind(this);
        this.handleLocationChange = this.handleLocationChange.bind(this);
    }

    // Basic change handling
    handleChange = event => {
        this.setState({ [event.target.id]: event.target.value });
    }

    // Location change needed because event.id cannot be taken from places object
    handleLocationChange = address => {
        console.log("The address has been changed: " + address);
        this.setState({ address: address });
    }

    // I need two separate handlers for each piece of the date range
    handleStartDateChange = date => {
        this.setState({ startDate: date });
    }

    handleEndDateChange = date => {
        this.setState({ endDate: date });
    }

    // 
    handleSubmit = async event => {
        event.preventDefault();
        console.log(event);

        // Construct the query string for AWS search

        var queryStatement = {
            accountId: 'account1', // Dummy
            groupId: 'group1', // Dummy
            caseId: '999', // Dummy
            freeText: this.state.freeText,
            radius: this.state.radius
        };

        if (this.state.startDate != null) {
            queryStatement['startDate'] = this.state.startDate.format("MM/DD/YYYY");
        }

        if (this.state.endDate != null) {
            queryStatement['endDate'] = this.state.endDate.format("MM/DD/YYYY");
        }

        if (this.state.address !== '') {
            queryStatement['address'] = this.state.address
        }

        console.log(queryStatement);

        this.setState({ isLoading: true });

        // Submit to the search API and load the Payload as a JSON object
        try {

            var resultSet;
            resultSet = await searchAssets(queryStatement);
            if (resultSet['StatusCode'] !== 200) {
                console.log("Error in lambda function");
            }
            console.log(JSON.parse(resultSet['Payload']));

            this.setState({
                isLoading: false,
                searchResults: JSON.parse(resultSet['Payload'])
            });

        }
        catch (e) {
            console.log(e);
            this.setState({ isLoading: false });
        }
    }

    render() {

        const autoLocationProps = {
          value: this.state.address,
          onChange: this.handleLocationChange,
        }

        console.log(this.state);

        // Only fetch suggestions when the input text is longer than 3 characters.
        const shouldFetchSuggestions = ({ value }) => value.length > 3


        return (
            <div className="Search">
                <form onSubmit={this.handleSubmit}>
                    <FormGroup controlId="freeText" bsSize="large">
                        <ControlLabel>Enter text to search on</ControlLabel>
                        <FormControl
                            type="textarea"
                            onChange={this.handleChange}
                            value={this.state.freeText}
                            placeholder="Enter any values to search on"
                        />
                    </FormGroup>
                    <FormGroup controlId="address" bsSize="large">
                        <ControlLabel>Enter search location</ControlLabel>
                        <PlacesAutocomplete 
                            inputProps={autoLocationProps}
                            shouldFetchSuggestions={shouldFetchSuggestions}
                            placeholderText="Start typing an address"
                        />
                    </FormGroup>
                    <FormGroup controlId="radius" bsSize="large">
                        <ControlLabel>Enter search radius</ControlLabel>
                        <FormControl
                            onChange={this.handleChange}
                            type="text"
                            value={this.state.radius}
                        />
                    </FormGroup>
                    <FormGroup controlId="startDate" bsSize="large">
                        <ControlLabel>Enter start date</ControlLabel>
                        <DatePicker
                            onChange={this.handleStartDateChange}
                            selected={this.state.startDate}
                            placeholderText="Enter start date"
                            isClearable={true}
                            maxDate={this.state.endDate}
                        />
                    </FormGroup>
                        <FormGroup controlId="endDate" bsSize="large">
                        <ControlLabel>Enter end date</ControlLabel>
                        <DatePicker
                            onChange={this.handleEndDateChange}
                            selected={this.state.endDate}
                            placeholderText="Enter end date"
                            isClearable={true}
                            minDate={this.state.startDate}
                        />
                    </FormGroup>
                    <LoaderButton
                        block
                        bsSize="large"
                        type="submit"
                        isLoading={this.state.isLoading}
                        text="Search for files"
                        loadingText="Searching..."
                    />
                </form>
            </div>
        );
    }
}

以表格形式查看:

import React, { Component } from "react";
import {Table, thead, tbody, th, tr, td } from "react-bootstrap";
import Search from "./Search";
import "./Home.css";

export default class Viewfiles extends Component {
    constructor(props) {
        super(props);
    }  

    render() {   

        // Eventually we will render the results in the table...
        var resultsObject = new Search();

        return (
            <div className="Viewfiles">
                {resultsObject.render()}
                <hr/>
                <Table striped condensed responsive >
                    <thead>
                        <tr>
                        <th>Thumbnail</th>
                        <th>Checksum</th>
                        <th>Address</th>
                        <th>Checksum</th>
                        <th>Description</th>
                        </tr>
                    </thead>
                </Table>
            </div>
        );
    }
}

1 个答案:

答案 0 :(得分:2)

React组件应在JSX中呈现,如<Search />。如果你自己实例化它们,React将无法使用状态和所有内容正确地安装它们。