Connecting backend and frontend
Connecting backend and frontend
Introduction
Now we will learn how to retrieve data from the backend and display it in the interface. VTEX IO uses GraphQL as a language/technology for data transfer, which makes programming our components quite simple. We will modify our Countdown component to search for the targetDate of the releaseDate
field of a VTEX product. To perform GraphQL queries in React, the Apollo Client is used, a state management lib that facilitates the integration of a GraphQL API with the front-end application.
The Apollo Client lib offers native integration with React, through hooks. Thus, making a query means using a hook that will not only perform the queries and fetch the data but will also provide caching and updating the UI state. This integration, called react-apollo
is already declared in package.json
.
Preparation
- To implement this functionality, add our countdown block on the product page and also do our tests on this page as well. To do this, do the following:
-
On your cloned theme (
store-theme
) access thestore/blocks/pdp/product.jsonc
file and, on theflex-layout.col#right-col
block add thecountdown
block, right before thebuy-button
:"product-gifts", + "countdown", "flex-layout.row#buy-button", "availability-subscriber",
-
Now, run
vtex link
on your theme again (if the process is not already running). It's done! Now our block is on the product page. Access any of these pages and see the renderedCountdown
component.
Release Date Query
-
First, create a folder in your Countdown app,
react/queries
and add aproductReleaseDate.graphql
file to it that will contain the query to be made. In particular, this query will receive a term, which will be the product slug to be retrieved at the launch date. It will call the resolverproduct
, already available through thevtex.search-graphql
app, and we will retrieve only the field we need.query productReleaseDate($slug: String) { product(slug: $slug) { releaseDate } }
Note that the query will need the slug of the product we are looking for. To do so, retrieve this information of the VTEX Product context.
-
To use this query, it is necessary to add the
vtex.search-graphql
app as a dependency on your app. We will also need to use theuseProduct
hook, exported by thevtex.product-context
the app, to retrieve the product slug that is loaded on the page. To do this, in your app'smanifest.json
, add in dependencies:"vtex.search-graphql": "0.x", "vtex.product-context": "0.x"
-
Now, it is necessary to import the
useQuery
hooks, to make the query that will return the data we described, anduseProduct
, to give us information about the current product slug. In addition, it is also necessary to import the query defined previously, which is found in the fileproductReleaseDate.graphql
.// react/Countdown.tsx import React from 'react' ... +import { useQuery } from 'react-apollo' +import useProduct from 'vtex.product-context/useProduct' +import productReleaseDate from './graphql/productReleaseDate.graphql'
It is important to higlight that there is the possibility of your IDE showing an error while importing
product-context
.The prop
targetDate
will no longer be necessary, so you can remove it. -
After that, define the query using the
productReleaseDate
imported and theuseQuery
hook, you can find the product data inuseProduct
hook. Since they are hooks, they only work inside react functional components.+ const { product } = useProduct() + const { data, loading, error } = useQuery(productReleaseDate, { + variables: { + slug: product?.linkText + }, + ssr: false + })
linkText
will be the same as'red-front-loading-washer'
, for example, when your component is rendered in this product's page. -
Now that we're using our block in pages that have the product context, it's important to test if this context exists. To do that, let's add the following code block:
if (!product) { return ( <div> <span>There is no product context.</span> </div> ) }
-
Besides, it is important to deal with the cases in which there is no data fetched when using
useQuery
and before returning the main component: loading and error In those cases, it is possible to return a span in the countdown component, such as the example below:if (loading) { return ( <div> <span>Loading...</span> </div> ) } if (error) { return ( <div> <span>Error!</span> </div> ) }
-
After sending the changes, access a product page and note that the query is working through a
console.log({data})
after callinguseQuery
, which should show something like this:{ data: { product: { releaseDate: '2019-01-01T00:00:00"', __typename: "Product" } } }
-
At last, but not least, to make Countdown set the hours for the product's
releaseDate
, change thetick
function parameter. You can also remove theprops
received in the component, as they will no longer be used.-tick(targetDate, setTime) +tick(data?.product?.releaseDate, setTime)
Result using the Red Front-Loading Washer product:
In case of having cases that the countdown is going up or with negative values, don't worry! It's related to the fact that some
releaseDate
can be in the past.
Well done!
This is the last step of the Store Block course, you did really well and we hope you've learned a lot until this moment. Congratulations!
If you want to continue learning more about how to develop using VTEX IO, we encourage you to start our next course, which focus on teaching how to develop services on top of VTEX IO.
Any questions?
See the answersheet for this step or check our [office hours] on the VTEX Developers channel(https://www.youtube.com/c/VTEXDevelopers)
Help us make this content better!
VTEX IO courses are open source. If you see something wrong, you can open a pull request!
Make a contributionUpdated 11 months ago