为什么我的React函数组件的方法不起作用

时间:2020-04-13 17:06:55

标签: javascript reactjs themoviedb-api

我正在用Moviedb API开发一个项目。我以电影组件的名称创建了电影列表。我在此组件中提出了api请求。从Movie组件到MovieInfo组件,我通过道具发送电影的发行日期和电影的类型。

但是我无法将子字符串和映射方法应用于带有props的MovieInfo组件中的这些属性。

class Movie extends Component {

state = {
    movie: [],
    loading: false,
    actors: [],
    directors: [],
    visible : 6 // This state is for how many actors rendered. 
}



componentDidMount() {
    this.setState({
        loading: true
    })

    let moviesEndPoint = `${BASE_URL}/movie/${this.props.match.params.movieId}?api_key=${API_KEY}&language=tr`
    let creditsEndPoint = `${BASE_URL}/movie/${this.props.match.params.movieId}/credits?api_key=${API_KEY}`;
    this.getMovieWithId(moviesEndPoint);
    this.getDirectorsAndActors(creditsEndPoint);
}

getMovieWithId = moviesEndPoint => {
    fetch(moviesEndPoint)
        .then(response => response.json())
        .then((movie) => {
            // console.log(movie);

            if (movie.overview !== "" && !movie.status_code) {
                this.setState({
                    movie,
                    loading: false
                })
            }
            else { // if have not turkish overview fetch this 
                let engEndPoint = `${BASE_URL}/movie/${this.props.match.params.movieId}?api_key=${API_KEY}`
                fetch(engEndPoint)
                    .then(response => response.json())
                    .then((movie) => {
                        this.setState({
                            movie
                        })
                    })
            }
        })
}

getDirectorsAndActors = creditsEndPoint => {
    fetch(creditsEndPoint)
        .then(response => response.json())
        .then((credits) => {
            // console.log(credits)
            const filterDirector = credits.crew.filter(person => person.job === "Director"); // filter directors from all employees
            // console.log(filterDirector)
            this.setState({
                actors: credits.cast,
                directors: filterDirector[0].name,
                loading: false
            })
        })
}

render() {
        const { movie, loading, actors, directors, visible } = this.state
        const { location } = this.props
    return (
        <>
            {
                loading ? <Spinner /> : null
            }
            {this.state.movie ?
                <MovieInfo
                    movieInfo={movie}
                    actors={actors}
                    directors={directors}
                    searchWord={location.searchWord}
                    visible = {visible}
                    loadMore = {this.loadMore}
                    loading = {loading}
                /> : null
            }

            {
                !actors && !loading ? <h1>Film Bulunamadı! </h1> : null
            }

        </>
    )
}

}

这是我的MovieInfo组件内部的无效代码,并且我的错误是这样的:

TypeError: Cannot read property 'substring' of undefined
TypeError: Cannot read property 'map' of undefined



const MovieInfo = ({ movieInfo, searchWord, directors, actors, visible, loadMore, loading }) => {

const editReleaseDate = (date) => {  //? Idk why doesn't work !
    // return date.substring(5).split("-").concat(date.substring(0,4)).join("/")
    return date

    // console.log(date)
    // return date;
}

return (

<Col sm={5} className="movieInfo p-4 animated fadeInRightBig">
                <p className = "movie-title" > {movieInfo.title} </p>
                <h5 className = "mb-4 text-warning">Yayınlanma Tarihi: <span className = "text-light">{editReleaseDate(movieInfo.release_date)}</span></h5>
                <h5 className = "text-warning">Açıklama</h5>
                <p>{movieInfo.overview} </p>
                <ProgressBar label={`IMDB: ${movieInfo.vote_average}`} animated now = {`${movieInfo.vote_average}`} min={0} max={10} />
                <h5 className = "text-warning mt-3">Türü: 

                {  //? Idk why doesn't work !
                    // movieInfo.genres.map((genre, i) => {
                    //     return <span key = {i} >{genre.name}</span>
                    // })
                }

                </h5>
                <h5 className ="mt-2 text-warning">Yönetmen:  <span className = "text-light">{directors} </span> </h5>
                <div> <i className="fas fa-film fa-5x"></i> </div>
            </Col>
)

1 个答案:

答案 0 :(得分:0)

您有2个api调用处于1 loading状态,因此一旦完成一个操作,它就告诉组件完成加载,即使第二个操作已经完成。我将其分为2种不同的加载状态,loadingMovieloadingActors

class Movie extends Component {

 state = {
    movie: [],
    loadingMovie: false,
    loadingActors: false,
    actors: [],
    directors: [],
    visible : 6 // This state is for how many actors rendered. 
 };

componentDidMount() {
    this.setState({
        ...this.state,
        loadingMovie: true,
        loadingActors: true,
    });

    let moviesEndPoint = `${BASE_URL}/movie/${this.props.match.params.movieId}?api_key=${API_KEY}&language=tr`;
    let creditsEndPoint = `${BASE_URL}/movie/${this.props.match.params.movieId}/credits?api_key=${API_KEY}`;
    this.getMovieWithId(moviesEndPoint);
    this.getDirectorsAndActors(creditsEndPoint);
}

getMovieWithId = moviesEndPoint => {
    fetch(moviesEndPoint)
        .then(response => response.json())
        .then((movie) => {
            // console.log(movie);

            if (movie.overview !== "" && !movie.status_code) {
                this.setState({
                    movie,
                    loadingMovie: false
                })
            }
            else { // if have not turkish overview fetch this 
                let engEndPoint = `${BASE_URL}/movie/${this.props.match.params.movieId}?api_key=${API_KEY}`
                fetch(engEndPoint)
                    .then(response => response.json())
                    .then((movie) => {
                        this.setState({
                            movie,
                            loadingMovie: false
                        });
                    })
            }
        })
}

getDirectorsAndActors = creditsEndPoint => {
    fetch(creditsEndPoint)
        .then(response => response.json())
        .then((credits) => {
            // console.log(credits)
            if (credits && credits.crew) {
               const filterDirector = credits.crew.filter(person => person.job === "Director"); // filter directors from all employees
               // console.log(filterDirector)
               this.setState({
                   actors: credits.cast,
                   directors: filterDirector[0].name,
                   loadingActors: false
               })
            } else {
               console.log('bad credits');
            }
        })
}

render() {
        const { movie, loadingActors, loadingMovie, actors, directors, visible } = this.state;
        const { location } = this.props;
    return (
        <>
            {loadingActors || loadingMovie ? <Spinner /> :
            (movie.length && actors.length) ?
                <MovieInfo
                    movieInfo={movie}
                    actors={actors}
                    directors={directors}
                    searchWord={location.searchWord}
                    visible = {visible}
                    loadMore = {this.loadMore}
                    loading = {(loadingActors || loadingMovie)}
                /> : null
            }
            {
                actors.length && !loadingActors ? <h1>Film Bulunamadı! </h1> : null
            }
        </>
    );
}
}