A set of components and utilities to work faster with DatoCMS in React environments. Integrates seamlessy with DatoCMS's GraphQL Content Delivery API and Real-time Updates API.
- Demos
- Installation
- Live real-time updates
- Progressive/responsive image
- Social share, SEO and Favicon meta tags
- Structured text
For fully working examples take a look at our examples directory.
Live demo: https://react-datocms-example.netlify.com/
npm install react-datocms
useQuerySubscription
is a React hook that you can use to implement client-side updates of the page as soon as the content changes. It uses DatoCMS's Real-time Updates API to receive the updated query results in real-time, and is able to reconnect in case of network failures.
Live updates are great both to get instant previews of your content while editing it inside DatoCMS, or to offer real-time updates of content to your visitors (ie. news site).
- TypeScript ready;
- Compatible with vanilla React, Next.js and pretty much any other React-based solution;
Import useQuerySubscription
from react-datocms
and use it inside your components like this:
const {
data: QueryResult | undefined,
error: ChannelErrorData | null,
status: ConnectionStatus,
} = useQuerySubscription(options: Options);
prop | type | required | description | default |
---|---|---|---|---|
enabled | boolean | ❌ | Whether the subscription has to be performed or not | true |
query | string | ✅ | The GraphQL query to subscribe | |
token | string | ✅ | DatoCMS API token to use | |
variables | Object | ❌ | GraphQL variables for the query | |
preview | boolean | ❌ | If true, the Content Delivery API with draft content will be used | false |
environment | string | ❌ | The name of the DatoCMS environment where to perform the query | defaults to primary environment |
initialData | Object | ❌ | The initial data to use on the first render | |
reconnectionPeriod | number | ❌ | In case of network errors, the period (in ms) to wait to reconnect | 1000 |
fetcher | a fetch-like function | ❌ | The fetch function to use to perform the registration query | window.fetch |
eventSourceClass | an EventSource-like class | ❌ | The EventSource class to use to open up the SSE connection | window.EventSource |
baseUrl | string | ❌ | The base URL to use to perform the query | https://graphql-listen.datocms.com |
The status
property represents the state of the server-sent events connection. It can be one of the following:
connecting
: the subscription channel is trying to connectconnected
: the channel is open, we're receiving live updatesclosed
: the channel has been permanently closed due to a fatal error (ie. an invalid query)
prop | type | description |
---|---|---|
code | string | The code of the error (ie. INVALID_QUERY ) |
message | string | An human friendly message explaining the error |
response | Object | The raw response returned by the endpoint, if available |
import React from "react";
import { useQuerySubscription } from "react-datocms";
const App: React.FC = () => {
const { status, error, data } = useQuerySubscription({
enabled: true,
query: `
query AppQuery($first: IntType) {
allBlogPosts {
slug
title
}
}`,
variables: { first: 10 },
token: "YOUR_API_TOKEN",
});
const statusMessage = {
connecting: "Connecting to DatoCMS...",
connected: "Connected to DatoCMS, receiving live updates!",
closed: "Connection closed",
};
return (
<div>
<p>Connection status: {statusMessage[status]}</p>
{error && (
<div>
<h1>Error: {error.code}</h1>
<div>{error.message}</div>
{error.response && (
<pre>{JSON.stringify(error.response, null, 2)}</pre>
)}
</div>
)}
{data && (
<ul>
{data.allBlogPosts.map((blogPost) => (
<li key={blogPost.slug}>{blogPost.title}</li>
))}
</ul>
)}
</div>
);
};
<Image />
is a React component specially designed to work seamlessly with DatoCMS’s responsiveImage
GraphQL query that optimizes image loading for your sites.
- TypeScript ready;
- Compatible with IE11;
- CSS-in-JS ready;
- Compatible with any GraphQL library (Apollo, graphql-hooks, graphql-request, etc.);
- Usable both client and server side;
- Compatible with vanilla React, Next.js and pretty much any other React-based solution;
- Offer WebP version of images for browsers that support the format
- Generate multiple smaller images so smartphones and tablets don’t download desktop-sized images
- Efficiently lazy load images to speed initial page load and save bandwidth
- Use either blur-up or background color techniques to show a preview of the image while it loads
- Hold the image position so your page doesn’t jump while images load
- Import
Image
fromreact-datocms
and use it in place of the regular<img />
tag - Write a GraphQL query to your DatoCMS project using the
responsiveImage
query
The GraphQL query returns multiple thumbnails with optimized compression. The Image
component automatically sets up the “blur-up” effect as well as lazy loading of images further down the screen.
For a fully working example take a look at our examples directory.
import React from "react";
import { Image } from "react-datocms";
const Page = ({ data }) => (
<div>
<h1>{data.blogPost.title}</h1>
<Image data={data.blogPost.cover.responsiveImage} />
</div>
);
const query = gql`
query {
blogPost {
title
cover {
responsiveImage(
imgixParams: { fit: crop, w: 300, h: 300, auto: format }
) {
# HTML5 src/srcset/sizes attributes
srcSet
webpSrcSet
sizes
src
# size information (post-transformations)
width
height
aspectRatio
# SEO attributes
alt
title
# background color placeholder or...
bgColor
# blur-up placeholder, JPEG format, base64-encoded
base64
}
}
}
}
`;
export default withQuery(query)(Page);
prop | type | required | description | default |
---|---|---|---|---|
data | ResponsiveImage object |
✅ | The actual response you get from a DatoCMS responsiveImage GraphQL query. |
|
className | string | ❌ | Additional CSS className for root node | null |
style | CSS properties | ❌ | Additional CSS rules to add to the root node | null |
pictureClassName | string | ❌ | Additional CSS class for the image inside the inner <picture /> tag |
null |
pictureStyle | CSS properties | ❌ | Additional CSS rules to add to the image inside the inner <picture /> tag |
null |
fadeInDuration | integer | ❌ | Duration (in ms) of the fade-in transition effect upoad image loading | 500 |
intersectionThreshold | float | ❌ | Indicate at what percentage of the placeholder visibility the loading of the image should be triggered. A value of 0 means that as soon as even one pixel is visible, the callback will be run. A value of 1.0 means that the threshold isn't considered passed until every pixel is visible. | 0 |
intersectionMargin | string | ❌ | Margin around the placeholder. Can have values similar to the CSS margin property (top, right, bottom, left). The values can be percentages. This set of values serves to grow or shrink each side of the placeholder element's bounding box before computing intersections. | "0px 0px 0px 0px" |
lazyLoad | Boolean | ❌ | Wheter enable lazy loading or not | true |
explicitWidth | Boolean | ❌ | Wheter the image wrapper should explicitely declare the width of the image or keep it fluid | false |
The data
prop expects an object with the same shape as the one returned by responsiveImage
GraphQL call. It's up to you to make a GraphQL query that will return the properties you need for a specific use of the <Image>
component.
- The minimum required properties for
data
are:aspectRatio
,width
,sizes
,srcSet
andsrc
; alt
andtitle
, while not mandatory, are all highly suggested, so remember to use them!- You either want to add the
webpSrcSet
field or specify{ auto: format }
in yourimgixParams
, to automatically use WebP images in browsers that support the format; - If you provide both the
bgColor
andbase64
property, the latter will take precedence, so just avoiding querying both fields at the same time, it will only make the response bigger 😉
Here's a complete recap of what responsiveImage
offers:
property | type | required | description |
---|---|---|---|
aspectRatio | float | ✅ | The aspect ratio (width/height) of the image |
width | integer | ✅ | The width of the image |
height | integer | ✅ | The height of the image |
sizes | string | ✅ | The HTML5 sizes attribute for the image |
srcSet | string | ✅ | The HTML5 srcSet attribute for the image |
src | string | ✅ | The fallback src attribute for the image |
webpSrcSet | string | ❌ | The HTML5 srcSet attribute for the image in WebP format, for browsers that support the format |
alt | string | ❌ | Alternate text (alt ) for the image |
title | string | ❌ | Title attribute (title ) for the image |
bgColor | string | ❌ | The background color for the image placeholder |
base64 | string | ❌ | A base64-encoded thumbnail to offer during image loading |
Just like the image component, renderMetaTags()
is a helper specially designed to work seamlessly with DatoCMS’s _seoMetaTags
and faviconMetaTags
GraphQL queries so that you can handle proper SEO in your pages with a simple one-liner.
- TypeScript ready;
- Compatible with any GraphQL library (Apollo, graphql-hooks, graphql-request, etc.);
- Usable both client and server side;
- Compatible with vanilla React, Next.js and pretty much any other React-based solution;
renderMetaTags()
takes an array of Tag
s in the exact form they're returned by the following DatoCMS GraphQL API queries:
_seoMetaTags
query on any record, orfaviconMetaTags
on the global_site
object.
You can concat
multiple array of Tag
s together and pass them to a single renderMetaTags()
call.
For a working example take a look at our examples directory.
import React from "react";
import { renderMetaTags } from "react-datocms";
const Page = ({ data }) => (
<div>
<Helmet>{renderMetaTags(data.page.seo.concat(data.site.favicon))}</Helmet>
<h1>{data.page.title}</h1>
</div>
);
const query = gql`
query {
page: homepage {
title
seo: _seoMetaTags {
attributes
content
tag
}
}
site: _site {
favicon: faviconMetaTags {
attributes
content
tag
}
}
}
`;
export default withQuery(query)(Page);
<StructuredText />
is a React component that you can use to render the value contained inside a DatoCMS Structured Text field type.
import React from "react";
import { StructuredText } from "react-datocms";
const Page = ({ data }) => {
// data.blogPost.content ->
// {
// value: {
// schema: "dast",
// document: {
// type: "root",
// children: [
// {
// type: "heading",
// level: 1,
// children: [
// {
// type: "span",
// value: "Hello ",
// },
// {
// type: "span",
// marks: ["strong"],
// value: "world!",
// },
// ],
// },
// ],
// },
// },
// }
return (
<div>
<h1>{data.blogPost.title}</h1>
<StructuredText data={data.blogPost.content} />
{/* -> <h1>Hello <strong>world!</strong></h1> */}
</div>
);
};
const query = gql`
query {
blogPost {
title
content {
value
}
}
}
`;
export default withQuery(query)(Page);
You can also pass custom renderers for special nodes (inline records, record links and blocks) as an optional parameter like so:
import React from "react";
import { StructuredText, Image } from "react-datocms";
const Page = ({ data }) => {
// data.blogPost.content ->
// {
// value: {
// schema: "dast",
// document: {
// type: "root",
// children: [
// {
// type: "heading",
// level: 1,
// children: [
// { type: "span", value: "Welcome onboard " },
// { type: "inlineItem", item: "324321" },
// ],
// },
// {
// type: "paragraph",
// children: [
// { type: "span", value: "So happy to have " },
// {
// type: "itemLink",
// item: "324321",
// children: [
// {
// type: "span",
// marks: ["strong"],
// value: "this awesome humang being",
// },
// ]
// },
// { type: "span", value: " in our team!" },
// ]
// },
// { type: "block", item: "1984559" }
// ],
// },
// },
// links: [
// {
// id: "324321",
// __typename: "TeamMemberRecord",
// firstName: "Mark",
// slug: "mark-smith",
// },
// ],
// blocks: [
// {
// id: "324321",
// __typename: "ImageRecord",
// image: {
// responsiveImage: { ... },
// },
// },
// ],
// }
return (
<div>
<h1>{data.blogPost.title}</h1>
<StructuredText
data={data.blogPost.content}
renderInlineRecord={({ record }) => {
switch (record.__typename) {
case "TeamMemberRecord":
return <a href={`/team/${record.slug}`}>{record.firstName}</a>;
default:
return null;
}
}}
renderLinkToRecord={({ record, children, transformedMeta }) => {
switch (record.__typename) {
case "TeamMemberRecord":
return (
<a {...transformedMeta} href={`/team/${record.slug}`}>
{children}
</a>
);
default:
return null;
}
}}
renderBlock={({ record }) => {
switch (record.__typename) {
case "ImageRecord":
return <Image data={record.image.responsiveImage} />;
default:
return null;
}
}}
/>
{/*
Final result:
<h1>Welcome onboard <a href="/team/mark-smith">Mark</a></h1>
<p>So happy to have <a href="/team/mark-smith">this awesome humang being</a> in our team!</p>
<img src="https://www.datocms-assets.com/205/1597757278-austin-distel-wd1lrb9oeeo-unsplash.jpg" alt="Our team at work" />
*/}
</div>
);
};
const query = gql`
query {
blogPost {
title
content {
value
links {
__typename
... on TeamMemberRecord {
id
firstName
slug
}
}
blocks {
__typename
... on ImageRecord {
id
image {
responsiveImage(imgixParams: { fit: crop, w: 300, h: 300, auto: format }) {
srcSet
webpSrcSet
sizes
src
width
height
aspectRatio
alt
title
base64
}
}
}
}
}
}
}
`;
export default withQuery(query)(Page);
prop | type | required | description | default |
---|---|---|---|---|
data | StructuredTextGraphQlResponse | DastNode |
✅ | The actual field value you get from DatoCMS | |
renderInlineRecord | ({ record }) => ReactElement | null |
Only required if document contains inlineItem nodes |
Convert an inlineItem DAST node into React |
[] |
renderLinkToRecord | ({ record, children }) => ReactElement | null |
Only required if document contains itemLink nodes |
Convert an itemLink DAST node into React |
null |
renderBlock | ({ record }) => ReactElement | null |
Only required if document contains block nodes |
Convert a block DAST node into React |
null |
metaTransformer | ({ node, meta }) => Object | null |
❌ | Transform link and itemLink meta property into HTML props |
See function |
customRules | Array<RenderRule> |
❌ | Customize how document is converted in JSX (use renderRule() to generate) |
null |
renderText | (text: string, key: string) => ReactElement | string | null |
❌ | Convert a simple string text into React | (text) => text |