Shopify Checkout UI: Remix and ChatGPT
Introduction
In this tutorial, we will create a Shopify Checkout UI Extension, which allows merchants to include a personalized message on their checkout page. We will achieve this by building an API using ChatGPT.
However, our primary goal is to evoke a nostalgic feeling from the 2000s, when you were using Word and Clippy™ would appear, offering assistance while you tried to write a document.
The Goal
Our approach involves creating a Checkout UI extension that establishes a connection with the app’s API. This API will interact with ChatGPT to generate a personalized message. Subsequently, this message will be presented to the customer on the checkout page.
Getting Started
A great starting point is the Shopify documentation. You can access the Checkout UI Extension documentation here.
Building the App
To begin, let’s create a new app using the Shopify CLI. We’ll name the app “Clippify” and utilize Remix for the application. The first command will take a few minutes to complete.
$ yarn create "@shopify/app"
$ cd clippify
$ yarn shopify app generate extension
We select Checkout UI
as Type of extension, we put clippify-checkout-ui
as the name of the extension, and we will
use Typescript React
.
Fixing typing issue and eslint
Maybe it’s just me, but out of the box I had an error with the React components. I had to add the following
to extension/clippify-checkout-ui/package.json
:
{
"devDependencies": {
"@types/react": "^17.0.0",
"typescript": "^5.0.0",
"@shopify/eslint-plugin": "^42.1.0"
}
}
I also had an issue with eslint, so I had to add the following to .eslintrc.js
:
plugins: ["@shopify/eslint-plugin"]
After the Generation
With the generation process complete, let’s take a look at the project structure that lies before us:
clippify
├── Dockerfile
├── README.md
├── app
│ ├── db.server.js
│ ├── entry.server.jsx
│ ├── root.jsx
│ ├── routes
│ │ ├── app._index.jsx
│ │ ├── app.additional.jsx
│ │ ├── app.jsx
│ │ ├── auth.$.jsx
│ │ ├── auth.login
│ │ │ ├── error.server.jsx
│ │ │ └── route.jsx
│ │ └── webhooks.jsx
│ └── shopify.server.js
├── extensions
│ └── clippify-checkout-ui
│ ├── README.md
│ ├── locales
│ ├── package.json
│ ├── shopify.extension.toml
│ ├── src
│ │ └── Checkout.tsx
│ └── tsconfig.json
├── package.json
├── prisma
├── public
│ └── favicon.ico
├── remix.config.js
├── shopify.app.toml
├── shopify.web.toml
├── tsconfig.json
└── yarn.lock
The Checkout UI
Creating the Block with Picture and Message
Our goal is to design a block featuring an image and a message, ideally situated in the bottom right corner of the
checkout page. As we explore the
available extension points,
we find
that purchase.checkout.block.render
aligns perfectly with our requirements. Lucky for us, this extension is already included in the template.
To accomplish our goal, we need to identify the most suitable components for the task. The list of available components can be found here.
In our strategy, we will leverage the Grid component to carefully position the image and message elements.
Our Clippy™ Assistant!
Let’s find a mascot! We’ll head over to Lottie Files to discover a cute animation.
Who could resist a puppy tiger? Take a look:
Modifying Checkout.tsx
As a first step, we will simply add a straightforward message and our mascot. While we’re there, we’ll also retrieve the
cart lines with the useCartLines
hook (we will need them later):
function Extension() {
const cartLines = useCartLines()
// later we will use a metaobject to retrieve the picture asset. (Couldn't find a way to make the local path work)
return (
<Grid columns={['30%', 'fill']} spacing={'none'}>
<View>
<Image source="https://filippi.dev/tutorials/clippify/tiger.gif"/>
</View>
<View>
<TextBlock>
It looks like you want to buy {cartLines[0].merchandise.title}!
</TextBlock>
</View>
</Grid>
);
}
Testing the App
We have now some code we can test, let’s start the app:
$ yarn dev
This process will generate a link enabling us to install the app on our development store. extension.
Not bad for a first result, the position needs to be fixed, the message display is not great but we’ll get there. Our focus can now shift to creating the right message.
The API
During the generation of the app, a remix project with a server is created. We’ll leverage this server to establish an API that simply provides a message. Our parameters for this API will include the customer (which could be null) and the cart lines.
ChatGPT Integration
To obtain the ChatGPT code, the most effective approach is to visit the Playground. Through experimentation, we can fine-tune our desired sentence and then extract the corresponding code. Here’s what we came up with:
const {Configuration, OpenAIApi} = require("openai");
const configuration = new Configuration({
apiKey: process.env.OPENAI_API_KEY,
});
const openai = new OpenAIApi(configuration);
const response = await openai.createChatCompletion({
model: "gpt-4",
messages: [
{
"role": "user",
"content": "You are a shop assistant that is helping a customer. Their name is\nFabio. You want them to be happy about their purchase.\\\\n They have purchased White shirt and Yellow Tie. And you should explain how good they go together. Say\nthat in a brief way.\n"
}
],
temperature: 1,
max_tokens: 256,
top_p: 1,
frequency_penalty: 0,
presence_penalty: 0,
});
We will replace the parameters in our code and that’s it.
Remix
We create a new file app/routes/ai.jsx
which define a straightforward GET
endpoint, returning a message.
For the purpose of this tutorial (and to save time), we will not implement a request handler for the entire cart and
customer. Instead, we will focus on the specific parameters we need for the current task.
export const loader = async ({request}) => {
const url = new URL(request.url);
const firstName = url.searchParams.get("firstName");
const title1 = url.searchParams.get("title1");
const title2 = url.searchParams.get("title2");
const customerName = firstName ? "Their name is " + firstName + "." : ""
let cartMessage;
if (title2) {
cartMessage = `They have purchased ${title1} and ${title2}. And you should explain how good they go together.`
} else {
cartMessage = `They have purchased ${title1} and you should find a good reason to use it.`
}
const prompt = `You are a shop assistant that is helping a customer. ${customerName} You want them to be happy about their purchase.
${cartMessage} Say that in a short sentence suggesting that they should definitely but it. Be assertive.`
let res = await getChatStream({ // this is a wrapper of the previous code
message: prompt
}).then(res => res.choices[0]?.message?.content || "Sorry, your taste is too good, I'm speechless.");
return new Response(res, {headers: withCors()}); // we need to add CORS headers, I'll explain down here
};
Putting It All Together
Now, we can retrieve the generated message and display it. However, fetching the right URL of our API is a bit more complex than it might initially seem.
Fetch Call
As explained here, the checkout UI has a few limitations concerning network access. In summary:
- The UI runs in a Web Worker, resulting in a null origin. (This is why CORS headers are necessary for the API.)
- The UI lacks a secure method to authorize requests.
- App Proxy cannot be utilized for password-protected stores.
In a real production app, I would utilize the App Proxy. Unfortunately, it’s unavailable in a development store, forcing me to hardcode the URL (environment variables aren’t available neither). The lack of authentication is not a real problem, since we aren’t handling sensitive information.
const cartLines = useCartLines()
const {buyerIdentity} = useApi();
const [message, setMessage] = useState(null);
const firstName = buyerIdentity?.customer?.current?.firstName || "";
const title1 = cartLines[0]?.merchandise?.title;
const title2 = cartLines[1]?.merchandise?.title || "";
useMemo(() => {
fetch(`https://mydeployedapp/ai?firstName=${encodeURIComponent(firstName)}&title1=${encodeURIComponent(title1)}&title2=${encodeURIComponent(title2)}`)
.then(res => res.text()).then(text => {
setMessage(text);
})
}, [firstName, title1, title2]);
Completing the UI
At this stage, you might assume we’re almost there, just a bit of CSS, and we’re finished! However, that’s not entirely accurate. The React API (based on remote-ui) of the checkout UI only permits us to build functional components on top of the existing ones. Unfortunately, CSS isn’t supported. We’re limited to using the predefined parameters of the components, which hampers our ability to create the exact interface we desire.
Maybe this will change in the future, maybe not.
Image Generation Process
To solve that problem we will generate an image, on the server, with the message! This image will then be returned as a base64-encoded string.
If the concept of displaying text within an image doesn’t evoke a sense of nostalgia from the 2000s, I’m not sure what will.
To achieve this, we will utilize CanvasJS for image generation. The code responsible for creating the image is quite extensive, and you can review it in the repository. However, it’s not a critical aspect of the tutorial. I find the result quite satisfactory:
Final UI
The final UI is quite straightforward; our main task is to display the GIF along with the message (picture).
By default, the image will contain a default message, which will be replaced once the fetch operation is successful.
const [message, setMessage] = useState("");
return <Grid columns={["30%", "fill"]}>
<View>
<Image source="https://filippi.dev/tutorials/clippify/tiger.gif"/>
</View>
<View>
<Image source={message}/>
</View>
</Grid>
Demo
Deploy
We are deploying our app on Heroku. Following (this)[https://shopify.dev/docs/apps/deployment/web]
$ heroku login
$ heroku container:login
$ heroku create -a clippify -s container
$ yarn shopify app env show
We create a heroku.yml
file with the following content:
build:
docker:
web: Dockerfile
config:
SHOPIFY_API_KEY: 6.......................c
Because we are using canvasjs
that uses glibc
we can’t use alpine
(unless we install the glibc
package). So we
are
modifying Dockerfile
replacing node:18-alpine
with node:18-bullseye-slim
and
adding RUN apt-get update && apt-get install -y -q libfontconfig1
to install the fonts.
We need to update the heroku environment variables with the ones from the shopify app and our open api key.
$ heroku config:set -a clippify SCOPES=write_products SHOPIFY_APP_URL=<SHOPIFY_APP_URL> SHOPIFY_API_KEY=<SHOPIFY_API_KEY> SHOPIFY_API_SECRET=<SHOPIFY_API_SECRET>
$ heroku config:set -a clippify OPEN_API_KEY=<my open ai api key>
$ git push heroku
The app is now on heroku. We can add the block in our checkout and test it.
Positioning the block
I struggled a lot to make the app appear in the block list under the checkout customization. The simplified deployment introduced a “Development Store Preview” that needs to be OFF when installing the app to make it visible in the checkout block list. To understand that took about 20% of the time spent on this project. But now our puppy is in the right spot…
The Result
Pretty, pretty, pretty good, not bad!
Conclusion
It was really nice to work on Checkout UI, it’s fun. The UI limitations are a real struggle, and when
the checkout.liquid
customization will come to an end it will be fun to see how shops will React.
The Code
This is the repository if you want the full code: https://github.com/faaabio1618/clippify
Clippy is a registered trademark of Microsoft Corporation in the United States and/or other countries. The use of the Clippy image is for parody purposes only and is not endorsed by Microsoft. I’m not affiliated with Microsoft (buy I can send a CV if asked).