2017-08-27 158 views
0

我有兩個組件:CardContainerSingleVideoContainer。 CardContainer將根據數據庫中的數據包含多個SingleVideoContainer。作爲道具傳遞給另一個組件的道具

import React, { Component } from 'react'; 
import Modal from 'boron/WaveModal'; 

//Default firebase App 
import * as firebase from 'firebase'; 
import { firebaseApp } from '../firebase/firebase'; 
import SingleCardContainer from '../cards/SingleCardContainer'; 

var dataRef = firebaseApp.database(); 
var dataArray = []; 

class CardContainer extends Component { 
    constructor(props) { 
     super(props); 
     this.state = { 
      userid: "", 
      videoLink: "", 
      likes: "", 
      challenges: "", 
      videoCat: "", 
      videoDesc: "", 
      videoTitle: "", 
      profilePic: "", 
      disikes: "" 
     } 

     this.showModal = this.showModal.bind(this); 
     this.hideModal = this.hideModal.bind(this); 
     this.componentDidMount = this.componentDidMount.bind(this); 
    } 

    showModal() { 
     this.refs.modal.show(); 
    } 
    hideModal() { 
     this.refs.modal.hide(); 
    } 

    componentDidMount() { 
    } 


    render() { 
     function initApp() { 
      var videosRef = dataRef.ref('posts/'); 
      videosRef.on('value', function (snapshot) { 
       snapshot.forEach(function (data) { 
       var userInfo = {}; 
       var userArray = []; 
       //Set up the next user array group 
       //9 items for 1 user. 
       userInfo.userid = data.val().userid; 
       userInfo.likes = data.val().likes; 
       userInfo.dislikes = data.val().dislikes; 
       userInfo.challenges = data.val().challenges; 
       userInfo.profilePic = data.val().profilePic; 
       userInfo.videoCategory = data.val().videoCategory; 
       userInfo.videoDesc = data.val().videoDesc; 
       userInfo.videoTitle = data.val().videoTitle; 
       userInfo.videoURL = data.val().videoURL; 

       //Then save the object in the array 
       userArray.push(userInfo); 
       //change the index to next user. 
        }) 
      }); 
     } 


     /** 
     * This loads when the page loads (right before renders) 
     */ 
     window.addEventListener('load', function() { 
      initApp() 
     }); 

     return (
      <div id="bodyType"> 
      { 
       userArray.map(
        data => <SingleCardContainer 
          userid={data.userid} 
          title={data.title} 
          likes={data.likes} 
          dislikes={data.challenges} 
          videoURL={data.videoURL} 
       />) 

      } 
     </div> 
     ) 
    } 
} 
export default CardContainer; 

我想呈現SingleCardContainer通過傳遞來自dataArray的信息作爲道具。每個用戶有9個項目需要作爲道具傳入。 我該怎麼做?我可以渲染1但不是多個。

這裏是SingleCardContainer:

import React, { Component } from 'react'; 
import { 
    Player, ControlBar, 
    ForwardControl, CurrentTimeDisplay, 
    TimeDivider, VolumeMenuButton, BigPlayButton 
} from 'video-react'; 

import ModalContainer from '../cards/ModalContainer'; 
class SingleCardContainer extends Component { 
    constructor(props) { 
     super(props); 

     this.likeButton = this.likeButton.bind(this); 
     this.challengeButton = this.challengeButton.bind(this); 
     this.dislikeButton = this.dislikeButton.bind(this); 
    } 

    likeButton() { 
    } 
    challengeButton() { 
    } 
    dislikeButton() { 
    } 
    //We will have to pass down the states from CardContainer as props to this so that they get updated in real-time *fingers-crossed* 

    render() { 
     return (
      <div className="container"> 
       <div className="card" id="generalCard"> 
        <div className="card-text"> 
         <div id="singleVideoContainer"> 
          <h3>{this.props.title}</h3><p> {this.props.userid}</p> 
          <Player poster="" src={this.props.videoURL}></Player> 
          <div id="videoInfoSection"> 
           <div className="row" id="buttonContainerRow"> 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
             <a className="supportButtons" onClick={() => this.likeButton()} role="button" href="#"><i className="fa fa-thumbs-up"></i></a> 
             <p>{this.props.likes}</p> 
            </div> 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
             <a className="supportButtons" onClick={() => this.challengeButton()} role="button" href="#"><i className="fa fa-shield"></i></a> 
             <p>{this.props.challenges}</p> 
            </div> 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
             <a className="supportButtons" onClick={() => this.dislikeButton()} role="button" href="#"><i className="fa fa-thumbs-down"></i></a> 
             <p>{this.props.dislikes}</p> 
            </div> 
           </div> 
           <div id="commentSection"> 
            <p>{this.props.videoDesc}</p> 
           </div> 
          </div> 
         </div> 
        </div> 
       </div> 
      </div> 
     ); 
    } 
} 

export default SingleCardContainer; 
+0

似乎有代碼缺失。 dataArray'聲明在哪裏? React組件的定義在哪裏? – styfle

+0

@styfle嘿,我已經添加了我的CardContainer組件中的所有代碼。 – Bazooka

+0

嗯,我想我看到你的問題....你不應該在渲染函數中添加事件偵聽器。嘗試在類的外部移動'window.addEventListener'。另外,SingleCardContainer的定義在哪裏? – styfle

回答

0

你應該把任何數據調用到componentDidMount從未render。另請注意,您可以簡單地使用map將快照數組轉換爲值。然後致電setState,最終會用新數據呼叫render

import React from 'react'; 
import * as firebase from 'firebase'; 
import { firebaseApp } from '../firebase/firebase'; 

class CardContainer extends React.Component { 
    constructor(props) { 
     super(props); 
     this.state = { data: [] }; 
    } 

    componentDidMount() { 
     var dataRef = firebaseApp.database(); 
     var videosRef = dataRef.ref('posts/'); 
     videosRef.on('value', (snapshot) => { 
      var data = snapshot.map(o => o.val()); 
      this.setState({ data: data }); 
     }); 
    } 

    render() { 
     const { data } = this.state; 
     return (<div> 
      {data.map(o => <SingleCardContainer {...o} />)} 
     </div>); 
    } 
} 

class SingleCardContainer extends React.Component { 
    constructor(props) { 
     super(props); 
    } 

    render() { 
     const {title, userid, videoURL, videoDesc, likes, dislikes, challenges} = this.props; 
     return (
      <div className="container"> 
       <div className="card" id="generalCard"> 
        <div className="card-text"> 
         <div id="singleVideoContainer"> 
          <h3>{title}</h3> 
          <p>{userid}</p> 
          <p>{videoURL}</p> 
          <div id="videoInfoSection"> 
           <div className="row" id="buttonContainerRow"> 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
             <a className="supportButtons" onClick={console.log} role="button" href="#"><i className="fa fa-thumbs-up"></i></a> 
             <p>{likes}</p> 
            </div> 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
             <a className="supportButtons" onClick={console.log} role="button" href="#"><i className="fa fa-shield"></i></a> 
             <p>{challenges}</p> 
            </div> 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
             <a className="supportButtons" onClick={console.log} role="button" href="#"><i className="fa fa-thumbs-down"></i></a> 
             <p>{dislikes}</p> 
            </div> 
           </div> 
           <div id="commentSection"> 
            <p>{videoDesc}</p> 
           </div> 
          </div> 
         </div> 
        </div> 
       </div> 
      </div> 
     ); 
    } 
} 

下面我有一個工作的代碼片段,我剝出火力地堡,用硬編碼的數據替換它來證明,這將與之反應工作。

class CardContainer extends React.Component { 
 
    constructor(props) { 
 
     super(props); 
 
     this.state = { data: [] }; 
 
    } 
 

 
    componentDidMount() { 
 
     const data = [ 
 
      {title: 'Title1', userid:'u1', videoURL:'video1.webm', videoDesc:'desc1', likes: 'like1', dislikes: 'dislike1', challenges: 'challenge1'}, 
 
      {title: 'Title2', userid:'u2', videoURL:'video2.webm', videoDesc:'desc2', likes: 'like2', dislikes: 'dislike2', challenges: 'challenge2'}, 
 
      {title: 'Title3', userid:'u3', videoURL:'video3.webm', videoDesc:'desc3', likes: 'like3', dislikes: 'dislike3', challenges: 'challenge3'}, 
 
      ]; 
 
     this.setState({ data: data }); 
 
    } 
 

 

 
    render() { 
 
     const { data } = this.state; 
 
     return (<div> 
 
      {data.map(o => <SingleCardContainer {...o} />)} 
 
     </div>); 
 
    } 
 
} 
 

 
class SingleCardContainer extends React.Component { 
 
    constructor(props) { 
 
     super(props); 
 
    } 
 

 
    render() { 
 
     const {title, userid, videoURL, videoDesc, likes, dislikes, challenges} = this.props; 
 
     return (
 
      <div className="container"> 
 
       <div className="card" id="generalCard"> 
 
        <div className="card-text"> 
 
         <div id="singleVideoContainer"> 
 
          <h3>{title}</h3> 
 
          <p>{userid}</p> 
 
          <p>{videoURL}</p> 
 
          <div id="videoInfoSection"> 
 
           <div className="row" id="buttonContainerRow"> 
 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
 
             <a className="supportButtons" onClick={console.log} role="button" href="#"><i className="fa fa-thumbs-up"></i></a> 
 
             <p>{likes}</p> 
 
            </div> 
 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
 
             <a className="supportButtons" onClick={console.log} role="button" href="#"><i className="fa fa-shield"></i></a> 
 
             <p>{challenges}</p> 
 
            </div> 
 
            <div className="col-md-4 col-xs-6 col-sm-4"> 
 
             <a className="supportButtons" onClick={console.log} role="button" href="#"><i className="fa fa-thumbs-down"></i></a> 
 
             <p>{dislikes}</p> 
 
            </div> 
 
           </div> 
 
           <div id="commentSection"> 
 
            <p>{videoDesc}</p> 
 
           </div> 
 
          </div> 
 
         </div> 
 
        </div> 
 
       </div> 
 
      </div> 
 
     ); 
 
    } 
 
} 
 

 
ReactDOM.render(<CardContainer />, document.querySelector('div'));
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/15.6.1/react.min.js"></script> 
 
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/15.6.1/react-dom.min.js"></script> 
 

 
<div><!--REACT ROOT--></div>

不過,我不熟悉火力地堡,所以你碰上會由於數據從火力地堡來,我不擁有控制權,可能需要不同的任何錯誤題。

+0

嗨Styfle,你的回答完美無缺!之所以花費這麼長時間來接受它是因爲我無法弄清楚如何在firebase函數中使用「this.setState」(它不會直接讓你),所以我不得不將它分配給首先是本地變量,我昨晚想到了。謝謝! – Bazooka

+0

正確,'this'指的是本地函數範圍[見MDN](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/this),因此您可以將其分配給首先變量或將'function'更改爲[Arrow函數](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/this#Arrow_functions),就像我用'videosRef.on '在我的答案中。 – styfle

相關問題