订阅完成后,为什么我的数据会消失?

时间:2017-04-11 02:00:54

标签: reactjs graphql apollo graphql-subscriptions

为什么订阅完成后我的数据未定义? 我试图找出我的props.data.allLocations的undefined来自哪里。任何帮助将不胜感激。

//高阶组件​​

export const LocationList = graphql(

      gql`
        query ($site: ID!) {
          allLocations(
            filter: {
            site: {
              id:$site
            } 
          }
          ) {
            id
            name
          }
        }
          `,


     {

       options: (ownProps)=>({
         variables: {
           site: ownProps.site
         },
       }),
      //props were injected from the JSX element
      props: props => { 

          return {
            data: props.data,
            subscribeToData: params => {
              return props.data.subscribeToMore({
                document:

                  gql`
                    subscription newLocations {
                      Location {
                        mutation
                        node {
                          id
                          name
                        }
                      }
                    }
                   `,
                variables: {
                  //Empty for now
                  //site: props.site,
                },
                updateQuery: (previousState, {subscriptionData}) => {

                  if (!subscriptionData.data) {
                                return previousState;
                            }

                var newArray =[subscriptionData.data.Location.node].concat(previousState.allLocations)
                var newState = { 
                  ...previousState,
                  allLocations: [
                    {
                      ...subscriptionData.data.Location.node
                    },
                      ...previousState.allLocations
                  ],

                };

                return newState
                }
              })
            },

          }
      },
      onError: (err) => console.error(err)
           } )(EntityList)

//列出组件

class EntityList extends Component {


    componentWillReceiveProps(newProps) {
    if (!newProps.data.loading) {
      console.log(newProps)
      if (this.subscription && this.props.hasOwnProperty('subscribeToData')) {
        if (newProps.data.allLocations !== this.props.data.allLocations) {
          console.log("Resubscribe")
          // if the todos have changed, we need to unsubscribe before resubscribing
          this.subscription()
        } else {
          console.log('else')
          // we already have an active subscription with the right params
          return
        }
      }
      this.subscription = this.props.subscribeToData(newProps)

    }}


  render () {

    if (this.props.data.loading) {
      return (<div>Loading</div>)
    }


    var Entities = [];
    for(var key in this.props.data) {

        if(this.props.data[key] instanceof Array){
          Entities = Entities.concat(this.props.data[key])

          //console.log(this.props.data[key])
        }
      }


    return (
      <div className='w-100 flex justify-center bg-transparent db'  >
        <div className='w-100 db' >
          {Entities.map((entity) =>

           ( 
              <EntityListView 
                user={this.props.user} 
                key={entity.id} 
                entityId={entity.id} 
                name={entity.name} 
                profilePic={(entity.profilePic)?entity.profilePic.uuid : this.props.defaultPic.uuid} 
                clickFunc={this.props.clickFunc}
              /> 
            ))} 

        </div>
      </div>
    )
  }
}

1 个答案:

答案 0 :(得分:0)

感谢所有关注此事的人。 我按照@nburk的建议拆分了我的查询。 我能够通过创建自己的高阶组件来保持模块化。 如果有人有兴趣:

const Subscriber = (document, config) => {  

  return (WrappedComponent) => { 

    return class extends Component {

      constructor(props) {
        super(props)
        this.state={}
        this.subscription = null
      }
        componentWillReceiveProps(nextProps) {
          console.log(nextProps)
          if (!this.subscription && !nextProps.data.loading) {
            console.log("Sucess")
            let { subscribeToMore } = this.props.data
            this.subscription = [subscribeToMore(
            {
              document: document,
              updateQuery: (previousResult, { subscriptionData }) => {
                console.log(subscriptionData)
                var newResult = [subscriptionData.data[config.modelName].node].concat(previousResult[config.propName])
                console.log(newResult)
                return { 
                  //"allItems"
                  [config.propName]: newResult,
                }

              },
              onError: (err) => console.error(err),
            }
              )]
                this.setState({});
              }
            }


      render () {

        if (this.props.data.loading) {
          return (<div>Loading</div>)
        }
           return (
                  <WrappedComponent {...this.props} />
                 )

      }
    }
  }
}

export default Subscriber

/*prop config = {
  propName: "string", //allItems // from query 
  //so I do not have to parse the gql query (would have to recieve from parent)
  modelName: "string" //Item 
  // from subscribe //so I do not have to parse the gql subscribe
}
*/

我现在将我的组件包装成这样:

export const LocationList = graphql(gql`..., 
  {options: (ownProps)=>({
    variables: {
      site: ownProps.site
    }})})(
  Subscriber(gql`..., 
    propName:"allLocations",
    modelName:"Location",
  )(EntityList))

由于