5

尝试使用 React 进行中继,今天遇到了这个问题。这是我到目前为止所做的。

根查询:

query {
  tasks {
    id
    taskName
    taskStatus
    userId
  }
}

React 组件层次结构

App   
 ↳--TaskList (props: tasks)
    ↳--TaskListItem (props: task)

现在由于托管原则,我知道我必须在每个组件中编写片段来描述它们的数据需求。

任务列表项.js

const TaskListItemContainer = createFragmentContainer(
    TaskListItem,
    graphql`
        fragment TaskListItem_task on task {
            id
            taskName
            taskDone
            authorId
        }
    `
);

任务列表.js

const TaskListContainer = createFragmentContainer(
    TaskList,
    graphql`
        fragment TaskList_tasks on task {
            tasks {
                ...TaskListItem_task
            }
        }
    `
);

应用程序.js

<QueryRenderer
   environment={relayEnvironment}
   query={graphql`
       query AppQuery {
         ...TaskList_tasks
       }
     `
   }

当我运行中继编译器时,出现以下错误。

Fragment "TaskList_tasks" cannot be spread here as objects of type "Query" can never be of type "task".

App.js (3:15)
2:             query AppQuery {
3:               ...TaskList_tasks
                 ^
4:             }

由于这个问题,无法弄清楚如何组织结构。我是否应该修改模式只是为了方便客户端的片段结构和重用?

4

1 回答 1

14

一个基本的 Fragment 包含五样东西:

  • fragment关键字_
  • 片段名称
  • on关键字_
  • 片段适用的类型
  • 包裹在一组花括号中的选择集

选择集是您指定的类型的一个或多个字段,当您使用 Fragment 时要请求这些字段。将 Fragment 视为单个选择集的替代品。如果我有这样的查询:

query {
  foo
  bar
}

然后{ foo bar }是我请求的选择集,在这种情况下是Query类型(或您的模式中调用的任何查询根操作类型)。所以如果我想使用一个片段,我会写:

query {
  ...QueryFields
}

fragment QueryFields on Query {
  foo
  bar
}

在您的代码中,您尝试编写如下查询:

query {
  ...TaskList_tasks
}

TaskList_tasks但是,正如错误所示,与片段关联的类型是task. 但是您不是task在这里替换类型的选择集,而是替换类型的选择集Query。所以你的请求无效。

TLDR;您需要将片段上的类型更改为Query

fragment TaskList_tasks on Query {
  tasks {
    ...TaskListItem_task
  }
}
于 2019-04-22T17:29:12.390 回答