1

I have the following component in react

import React, { Component } from 'react';
import axios from 'axios'
import Post from './Post'
import Navbar from '../Navbar'

class SinglePost extends Component {
    constructor(props) {
        super(props);
        this.state = { 
            post: {}
        }
    }
    componentDidMount() {
        const { post_id } = this.props.match.params
        axios.get(`http://127.0.0.1:8000/posts/${post_id}`)
        .then(res => {
            this.setState({ post: res.data })
            console.log(res.data)
            console.log(this.state)
        })
        .catch(err => console.log(err))
    }
    render() { 
        return (
            <div>
                <Navbar />
                <div style={{height: '95px'}}></div>
                <Post key={this.state.post.id} post={this.state.post} />
            </div>
        )
    }
}

export default SinglePost;

The correct response is recieved from the backend and set to state properly. However the data passed to the Post component shows up as undefined.

Here is teh post component:

import React from 'react'
import Comment from './Comment'
import CommentForm from './CommentForm'

const placeholder_url = 'https://image.shutterstock.com/image-vector/ui-image-placeholder-wireframes-apps-260nw-1037719204.jpg'
const placeholder = ''

function Post(props) {
    console.log(props) //Shows up as an empty object
    return (
        <div className="uk-background-muted uk-margin-top uk-container">
            <div>
                <div className="uk-background-default"> 
                    <img src={props.post.user.pp || placeholder} width="50" height="50" alt={props.post.user.username} />
                    <a href={`/users/${props.post.user.id}`}>{props.post.user.username}</a>
                </div>
                <img className="uk-align-center" src={props.post.image || placeholder_url} alt="placeholder" />
                <ul>
                    <li>Likes: {props.post.post_liked.length}</li>
                    <li>Dislikes: {props.post.post_disliked.length}</li>
                    <li>Comments: {props.post.comment_post.length}</li>
                </ul>
                <p className="uk-text-large">{props.post.text}</p>
                {props.post.comment_post.map(comment => (<Comment key={comment.id} comment={comment} />))}
                <div>
                <CommentForm key={props.post.id} post={props.post} />
                </div>
            </div>
        </div>
    )
  }

export default Post

This is the error I get: TypeError: Cannot read property 'pp' of undefined

svoruganti
  • 307
  • 1
  • 12
  • 4
    This is because on first render post is {} (an empty object) , in SinglePost before the return just do a checking if(this.state.post === {}) return null, to avoid rendering Post with an empty object. Or check it in the Post component right before the return : if(props.post === {}) return null – CevaComic Jun 09 '20 at 16:37
  • @CevaComic's answer won't work, because you can't compare empty objects in Javascript... `{} === {}` = `false`. See https://stackoverflow.com/questions/679915/how-do-i-test-for-an-empty-javascript-object – simmer Jun 09 '20 at 16:44
  • 1
    Sorry, my bad, I usually make the check like so if(!post.id) , I never used that method but i thought it works. Thanks @simmer – CevaComic Jun 09 '20 at 16:48
  • No worries. It's not intuitive! – simmer Jun 09 '20 at 16:50

2 Answers2

1

State-setting is asynchronous, so whenever you want to act upon data from state, you should include a manual check against undefined values.

Try something like this:

class SinglePost extends Component {
    constructor(props) {
        super(props);
        this.state = { 
            post: null // null initial value is easier to check than an empty object
        }
    }
    componentDidMount() {
        const { post_id } = this.props.match.params
        axios.get(`http://127.0.0.1:8000/posts/${post_id}`)
        .then(res => {
            this.setState({ post: res.data })
            console.log(res.data)
            console.log(this.state)
        })
        .catch(err => console.log(err))
    }
    render() { 
        return (
            <div>
                <Navbar />
                <div style={{height: '95px'}}></div>

                {this.state.post &&
                    <Post key={this.state.post.id} post={this.state.post} />
                }
            </div>
        )
    }
}
simmer
  • 2,275
  • 1
  • 16
  • 20
  • I tried this it still doesnt work, the condition results in true, so the data gets passed to teh post, but it still shwos up as undefinded. – svoruganti Jun 09 '20 at 16:48
  • Ok, I've updated my answer to use `null` as an initial value for `this.state.post`, which will be easier to check against. Try that. – simmer Jun 09 '20 at 16:53
1

The error happens in the first render when post = {} and the Post component is trying to get props.post.user.pp.

To solve the problem, in the SinglePost component, you could initialize post to be null instead of {} and in the Post component, handle the post === null case:

function Post(props) {
  if(!props.post) {
    return null;
    // return <span>Loading</span>;
  }
  return (
      <div className="uk-background-muted uk-margin-top uk-container">
        ...
Hangindev
  • 2,748
  • 5
  • 17