Content Relationship
This article explains what the content relationship field is and how to configure it.

A content relationship field in the Page Builder.
Content writers can link pages through content relationship fields. Developers can then access content from those pages.
Content relationship fields are often used for:
- Connecting a blog post to its author.
- Displaying a testimonial on a landing page.
- Linking a product to its category.
Content from related pages can be accessed like any other field. Use isFilled.contentRelationship() to ensure the relationship has a value before accessing nested content.
import { isFilled } from "@prismicio/client";
import { PrismicText } from "@prismicio/react";
{
  isFilled.contentRelationship(post.data.author) && (
    <p>
      Written by <PrismicText field={post.data.author.data?.name} />
    </p>
  );
}Add a content relationship to a content model
- Open Slice Machine- In your Prismic project, start Slice Machine to begin editing content models. - npx start-slicemachine --open
- Add a content relationship field- In Slice Machine, navigate to the slice, page type, or custom type you want to modify. Add a content relationship field. - The label determines the label shown to content writers in the Page Builder. Use an easily understood name. - The API ID determines the property name in the Content API. Use a short, snake-cased name. 
- Select the allowed page type- You can restrict the field to a specific page type. For example, you can configure the field to only allow Author pages in an Author field. - In the field’s settings, click the Add type button to select a page type. - You’ll be able to select fields from that page type to include in the API response in the next step. 
- Select the fields to fetch- If you restrict the field to a specific page type, you can select which fields to include in the API response. - Under the Allowed type section, use the field picker to choose which fields should be included. Up to 2 levels of nested fields can be selected (e.g. - blog->- author->- profession).- The selected fields are included in Slice Machine’s generated TypeScript types. 
Display content relationships
Content from a related page can be displayed like any other field. The name of a blog post’s author, for example, can be rendered using the <PrismicText> component.
import { isFilled } from "@prismicio/client";
import { PrismicText } from "@prismicio/react";
{
  isFilled.contentRelationship(post.data.author) && (
    <p>
      Written by <PrismicText field={post.data.author.data?.name} />,
      {isFilled.contentRelationship(post.data.author.data?.profession) && (
        <PrismicText field={post.data.author.data.profession?.data.name} />
      )}
    </p>
  );
}Learn about all the available fields
Examples
Blog post authors
On a blog, you could have posts that reference their authors along with the author’s profession.
- Add a content relationship field- On a Post type, add a content relationship field called - author. Only allow linking Author pages on the content relationship.
- Select fields to include- Select the following fields to include in your API response: - name
- profession.name
 
- Display the author information- Use the author data in your blog post to display the author’s name and profession. - import { isFilled } from "@prismicio/client"; import { PrismicText } from "@prismicio/react"; { isFilled.contentRelationship(post.data.author) && ( <p> Written by <PrismicText field={post.data.author.data?.name} />, {isFilled.contentRelationship(post.data.author.data?.profession) && ( <PrismicText field={post.data.author.data.profession?.data.name} /> )} </p> ); }
Landing page testimonials
On a landing page, you could feature testimonials from your customers or clients.
- Add a content relationship field- On a Landing Page type, add a group field called - testimonialscontaining a content relationship field. Only allow linking Testimonial pages on the content relationship.
- Display the testimonials- Use the testimonial data in your landing page to display quotes with avatars. - import { isFilled } from "@prismicio/client"; import { PrismicNextImage } from "@prismicio/next"; import { PrismicRichText } from "@prismicio/react"; <ul> {page.data.testimonials.map( (item) => isFilled.contentRelationship(item.testimonial) && ( <li key={item.testimonial.id}> <PrismicNextImage field={item.testimonial.data?.avatar} /> <PrismicRichText field={item.testimonial.data?.quote} /> </li> ), )} </ul>;
Taxonomies
On a blog, you could organize posts by categories.
- Create a Category custom type- This type will represent blog post categories. It should be repeatable and have a Name field. 
- Add a content relationship field- On a Post type, add a content relationship field called - category. Only allow linking Category pages on the content relationship.
- Display the category- Use the category data in your blog post to display the post’s category. - import { isFilled } from "@prismicio/client"; { isFilled.contentRelationship(post.data.category) && ( <span>{post.data.category.data?.name}</span> ); }
- Create a category page- In your API query, use the - filtersoption to query all blog posts for a category.- This example fetches all blog posts from a category with the page ID - YksUgRIAACEA-UZD:- import { filter } from "@prismicio/client"; const posts = await client.getAllByType("post", { filters: [filter.at("my.post.category", "YksUgRIAACEA-UZD")], });
Nested menus
On a website, you could build a nested navigation menu.
- Create a Child Menu custom type- This type will represent a list of links. It should be repeatable and have a group field called - linkscontaining a link field and an icon field.
- Create a Menu custom type- This type will represent top-level lists of links. It should be a singleton and have a group field called - child_menuscontaining a content relationship field. The content relationship should only allow linking to Child Menu pages.
- Select fields to include- Select the following fields to include in your API response: - links.link
- links.icon
 
- Display the menu- Use the menu data to render your navigation with proper field checking. - import { isFilled } from "@prismicio/client"; import { PrismicNextImage, PrismicNextLink } from "@prismicio/next"; <ul> {menu.data.child_menus.map( (item) => isFilled.contentRelationship(item.child_menu) && ( <li key={item.child_menu.id}> <ul> {item.child_menu.data?.links.map( (linkItem) => isFilled.link(linkItem.link) && ( <li key={linkItem.link.url}> <PrismicNextImage field={linkItem.icon} /> <PrismicNextLink field={linkItem.link} /> </li> ), )} </ul> </li> ), )} </ul>;
Check if a content relationship has a value
Use isFilled.contentRelationship() from @prismicio/client to check if a content relationship field has a value.
import { isFilled } from "@prismicio/client";
if (isFilled.contentRelationship(slice.primary.my_content_relationship_field)) {
  // Field has a value
}Learn more about isFilled
API response
Here is what a content relationship looks like from the Content API:
{
  "author": {
    "id": "XxnD3REAACYAk_CJ",
    "type": "author",
    "tags": [],
    "slug": "ada-lovelace",
    "lang": "en-us",
    "uid": "ada-lovelace",
    "data": {
      "name": [
        {
          "type": "paragraph",
          "text": "Ada Lovelace",
          "spans": []
        }
      ],
      "profession": {
        "id": "ZpqX7SFJJKEBl_VK",
        "type": "profession",
        "tags": [],
        "slug": "mathematician",
        "lang": "en-us",
        "uid": "mathematician",
        "data": {
          "name": [
            {
              "type": "paragraph",
              "text": "Mathematician",
              "spans": []
            }
          ]
        },
        "link_type": "Document",
        "isBroken": false
      }
    },
    "link_type": "Document",
    "isBroken": false
  }
}The data property’s contents is determined by the selected fields.
When a selected field is deleted, it’s automatically removed from the API response.
GraphQuery (Legacy)
GraphQuery is a legacy API option that allows selective fetching (specific fields only) and deep fetching (fields from linked pages). This feature is maintained for existing users but is not recommended for new projects.
Basic Usage
GraphQuery uses GraphQL-like syntax. The top-level property represents a page type and its children represent its fields.
Pass the GraphQuery to the Prismic client’s graphQuery option.
const blogPost = await client.getByUID("blog_post", "my-first-post", {
  graphQuery: `
    {
      blog_post {
        title
        description
      }
    }
  `,
});This example fetches the title and description fields from blog posts.
Arrays
Fields from an array of objects can be selected. Repeatable groups, slice zones, and queries for multiple pages return arrays.
const blogPost = await client.getByUID("blog_post", "my-first-post", {
  graphQuery: `
    {
      blog_post {
        title
        gallery {
          photo
          caption
        }
      }
    }
  `,
});This example fetches the gallery group array with its photo and caption fields from blog posts.
Unions
Use unions for content that can return multiple kinds of content, like slices, slice variations, or linked pages.
Each kind of content should be listed with desired the fields. Prepend the API ID with ...on:
// Linked pages
const blogPost = await client.getByUID("blog_post", "my-first-post", {
  graphQuery: `
    {
      blog_post {
        author_link {
          ...on author {
            name
            bio
          }
        }
      }
    }
  `,
});
// Slices
const page = await client.getByUID("page", "my-page", {
  graphQuery: `
    {
      page {
        slices {
          ...on text_block {
            primary {
              title
            }
          }
          ...on image_gallery {
            items {
              image
            }
          }
        }
      }
    }
  `,
});This example fetches a the name and bio fields from a blog post’s author. It also fetches fields from the text_block and image_gallery slices.
Spread Operator
Select all fields from a set using the spread operator with ... and Fields:
const blogPost = await client.getByUID("blog_post", "my-first-post", {
  graphQuery: `
    {
      blog_post {
        ...blog_postFields
        author_link {
          ...on author {
            ...authorFields
          }
        }
      }
    }
  `,
});This example fetches all fields from the blog post and all fields from the blog post’s author.
TypeScript Support
Type fetched fields using type assertion (the as operator).
import { Content } from "@prismicio/client";
const featuredBlogPost = home.data
  .featured_blog_post as typeof home.data.featured_blog_post & {
  data: Pick<Content.BlogPostDocument["data"], "title" | "description">;
};
const title = featuredBlogPost.data.title;
//    ^ Typed as Content.BlogPostDocument['data']['title']