This article is a tutorial on how to build a blog post preview feature using Astro and Strapi, a web framework for content sites.
We will show you how to achieve this from a technical standpoint. For this, we will use the new Drafts & Publish feature that was released in Strapi 5, visit Strapi docs for more information. Furthermore, we'll make use of Astro's static generation functionality to build a blog that meets the highest performance standards so you can climb the SEO rankings.
We will build a blog post preview feature. This makes it possible to keep new blog posts private before you choose to publish them. Before you publish new posts, you can make them available as a draft. Draft posts won't show up on the home page and are only accessible to those who know the link. So what are the benefits of this? You can use this to present upcoming blog posts to your fellow team members to receive immediate feedback.
We will leverage the new Strapi 5 Strapi Document Service API and its new functions publish()
, unpublish()
, and discardDraft()
functions to allow changing visibility of blog posts directly through the Astro frontend. This will be a password-protected feature only available to authorized users.
So let's get started!
Astro is a web framework for content-driven websites, which makes it ideal for the blog we're building. It will render the blog posts and has special support for internationalization and static generation. Styling will be done with Tailwind CSS. Finally, the content will live in the powerful Strapi CMS.
All you need for this project is to have at least Node.js version 18 installed on your computer and accessible from the terminal.
Enter the following command into your terminal to check:
$ node --version
v20.11.0
If you don't have Node.js installed, head here to install it.
First, we'll need to set up Strapi. Run the command below in your terminal to set up a new Strapi project:
npx create-strapi@latest drafts-and-publish-blog
In the prompt, choose the following answers:
? Please log in or sign up. Skip
? Do you want to use Typescript ? No
? Use the default database (sqlite) ? Yes
Once the setup is complete, change into the project folder and start Strapi:
cd ./drafts-and-publish-blog
npm run develop
You will now be taken to a sign-up page in the browser. Fill out the form.
After signing up, you will be taken to the Strapi Dashboard. In the next step, we will create the collection for the blog posts with a draft or published state.
Let's create our collection. In the sidebar, click on Content-Type Builder and then Create new collection type.
Call the collection type Post. Do not press Continue just yet, as we will need to modify the advanced settings.
Navigate to the Advanced Settings and make sure the Draft & publish checkbox is selected. We'll use that flag to differentiate between Draft and Published Posts.
Now, we need to define the fields our posts should have. Add the following fields:
title
: Text, Short Text, Required fieldcontent
: Rich Text (Markdown), Required field, do not use Rich Text (Blocks)header
: Media (image only), Single Media, Optional fieldslug
: UID, attached to title
Once you're done, click the Save button. We now have the collection we need to start writing content. Let's go ahead and create two posts: one that is in the Draft state and one that is already Published.
Click on Content Manager, Post, and then Create new entry.
You can write whatever you want here. We will be using Lorem Ipsum.
Create Draft Entry
Click Save, but do not publish the post yet. We will use this one as a draft. You can add another post that you can publish immediately.
Create Published Entry
Once you're ready, click Save and Publish to publish the post. If you want, you can go ahead and create another post so that we have more than one post to show on the website.
We now have posts in a defined collection, but no way to access them from Astro. To change this, we need to ensure they can be accessed through the Strapi API. For this, navigate to Settings > Users & Permissions Plugin > Roles.
In this section, select the Public role to grant find
and findOne
permissions to the Post collection.
Ensure that no other permissions are granted to the Public role, as this would allow anyone to modify or delete your content. Once done, press Save.
Now you'll be able to access your posts through the Strapi API at http://localhost:1337/api/posts
. Try it out to make sure everything has worked. This is the API we will consume on our frontend to render the posts. With this, we're ready to set up the Astro for our blog. Be aware that you will only see Published posts here. We will deal with accessing Drafts later.
First, we'll need to set up Astro. Run the command below in your terminal to set up a new Astro site:
npm create astro@latest astro-blog
Select the following choices in the installation sequence:
astro Launch sequence initiated.
◼ dir Using astro-blog as project directory
tmpl How would you like to start your new project?
Include sample files
ts Do you plan to write TypeScript?
No
◼ No worries! TypeScript is supported in Astro by default,
but you are free to continue writing JavaScript instead.
deps Install dependencies?
Yes
git Initialize a new git repository?
No
◼ Sounds good! You can always run git init manually.
✔ Project initialized!
■ Template copied
■ Dependencies installed
next Liftoff confirmed. Explore your project!
Enter your project directory using cd ./astro-blog
Run npm run dev to start the dev server. CTRL+C to stop.
Add frameworks like react or tailwind using astro add.
Stuck? Join us at https://astro.build/chat
╭─────╮ Houston:
│ ◠ ◡ ◠ Good luck out there, astronaut! 🚀
╰─────╯
Now that we have installed Astro, change to the astro-blog
directory. Then, we want to add Tailwind CSS to style our blog. For this, first run this command in your terminal and say Yes to all choices:
npx astro add tailwind
This has automatically set up Tailwind CSS in Astro for us. However, we also need the Tailwind Typography plugin so that we can style the prose in the posts. So once again, open your terminal and run:
npm install -D @tailwindcss/typography
Modify your tailwind.config.mjs
file so that it imports the Typography plugin:
1/** @type {import('tailwindcss').Config} */
2export default {
3 content: ["./src/**/*.{astro,html,js,jsx,md,mdx,svelte,ts,tsx,vue}"],
4 theme: {
5 extend: {},
6 },
7 plugins: [require("@tailwindcss/typography")],
8};
Also, head over to the astro.config.mjs
and to enable Server-Side Rendering with output: "server"
. This allows us to change the visibility of a post whenever we want, without re-building the whole site.
1import { defineConfig } from "astro/config";
2
3import tailwind from "@astrojs/tailwind";
4
5// https://astro.build/config
6export default defineConfig({
7 output: "server",
8 integrations: [tailwind()],
9});
Now start the application using npm run dev
and visit http://localhost:4321/ to see the Astro site.
Our blog will consist of two pages:
/blog
, which will show all blog posts./blog/[slug]
, which will display a single blog post.Let's get started with building this common layout. Create a new file at /src/layouts/BlogLayout.astro
with the following content:
1---
2import Layout from "./Layout.astro";
3
4interface Props {
5 title: string;
6}
7
8const { title } = Astro.props;
9---
10
11<Layout title={title}>
12 <header class="bg-blue-600 text-white text-center py-4">
13 <h1 class="text-3xl font-bold">
14 <a href="/blog" class="text-white"> Astro Blog </a>
15 </h1>
16 </header>
17
18 <slot />
19</Layout>
See how we import the already existing layout from /src/layouts/Layout.astro
. While you're at it, open this file and remove the <style>
tag, since we're using Tailwind for everything. Afterwards, it should look like this:
1---
2interface Props {
3 title: string;
4}
5
6const { title } = Astro.props;
7---
8
9<!doctype html>
10<html lang="en">
11 <head>
12 <meta charset="UTF-8" />
13 <meta name="description" content="Astro description" />
14 <meta name="viewport" content="width=device-width" />
15 <link rel="icon" type="image/svg+xml" href="/favicon.svg" />
16 <meta name="generator" content={Astro.generator} />
17 <title>{title}</title>
18 </head>
19 <body>
20 <slot />
21 </body>
22</html>
Now we have the building blocks we need to render an overview of all blog posts. First, create a .env
file in the root of your project and add the API URL there:
STRAPI_URL=http://localhost:1337
We will use this to call the Strapi API from Astro. If you deploy your site, this will need to be set to the Strapi production URL instead of localhost
.
Create a new file at /src/pages/blog/index.astro
, which will serve as the overview page for our blog. Add this content to the file:
1---
2import BlogLayout from "../../layouts/BlogLayout.astro";
3
4const response = await fetch(
5 `${import.meta.env.STRAPI_URL}/api/posts?populate=header`
6);
7const data = await response.json();
8const posts = data.data;
9---
10
11<BlogLayout title="All Posts">
12 <main class="container mx-auto mt-8 px-4">
13 <div class="space-y-6 max-w-lg mx-auto">
14 {
15 posts.map((post) => {
16 const headerImageUrl = post.header?.url;
17 return (
18 <a
19 href={`/blog/${post.slug}`}
20 class="block bg-white rounded-lg shadow-md overflow-hidden hover:shadow-lg transition-shadow duration-300 max-w-md mx-auto"
21 >
22 {headerImageUrl && (
23 <img
24 src={`${import.meta.env.STRAPI_URL}${headerImageUrl}`}
25 alt="Article Header Image"
26 class="w-full h-48 object-cover"
27 />
28 )}
29 <div class="p-4">
30 <h2 class="text-xl font-bold mb-2">{post.title}</h2>
31 <p class="text-gray-600 text-sm">
32 Published on:{" "}
33 {new Date(post.publishedAt).toLocaleDateString()}
34 </p>
35 </div>
36 </a>
37 );
38 })
39 }
40 </div>
41 </main>
42</BlogLayout>
What this does is fetch all posts from our Strapi API. The ?populate=header
parameter tells Strapi to include the header image we've set for our posts. Astro now fetches this information and renders a page where we can view the previews for all blog posts.
Go and check it out in your browser at http://localhost:4321/blog
. When you publish a new post in Strapi, it will automatically show up on this page. You might have noticed that a 404 NOT FOUND
error occurs whenever you try to click on a blog post preview link. This is because we haven't created this page yet.
Let's get started with building the post detail view, where we will be able to see the post with all its content.
First, we'll need to install the marked
library, which will convert Strapi's Markdown into HTML:
npm install marked
Now, create a new file at /src/pages/blog/[slug].astro
. This will act as a catch-all for any post. Add the following content to the file:
1---
2import { marked } from "marked";
3import BlogLayout from "../../layouts/BlogLayout.astro";
4import Visibility from "../../components/Visibility";
5
6const { slug } = Astro.params;
7
8const publishedResponse = await fetch(
9 `${import.meta.env.STRAPI_URL}/api/posts?populate=header&filters[slug][$eq]=${slug}`
10);
11
12const publishedData = await publishedResponse.json();
13
14let post = publishedData.data[0];
15
16if (!post) {
17 return new Response("Post not found", { status: 404 });
18}
19
20const headerImage = post.header?.url;
21
22const publishDate = new Date(post.publishedAt).toLocaleDateString("en-US", {
23 year: "numeric",
24 month: "long",
25 day: "numeric",
26});
27---
28
29<BlogLayout title={post.title}>
30 <article class="prose prose-lg max-w-2xl mx-auto py-24">
31 {
32 headerImage && (
33 <img
34 src={`${import.meta.env.STRAPI_URL}${headerImage}`}
35 alt={post.title}
36 class="mb-6 w-full h-auto rounded-lg"
37 />
38 )
39 }
40 <div class="flex items-center justify-between mb-4">
41 <h1 class="mb-0">{post.title}</h1>
42 </div>
43 {publishDate && <p class="text-gray-500 mt-2">{publishDate}</p>}
44 <div set:html={marked.parse(post.content)} />
45 </article>
46</BlogLayout>
First, we fetch the post by using Strapi's filter in the API query. If it doesn't exist, we throw a 404 NOT FOUND
error. After finding the correct post, we can extract the header image and publication date.
The rest of the code is to render the post information and actual content. To turn the Markdown content into HTML, we use the marked
library. With the Tailwind classes prose prose-lg
, all elements of the article automatically get styled. So the HTML output from our Markdown parser is styled without us needing to do anything additional.
Now we have a working, fully static blog. However, we have no way to access the draft post.
We have already added the draft post to Strapi earlier, so there is no need to change anything within Strapi. Let's see how we can access drafts. By default, the API call only returns published articles, which is what we want.
However, if a user has the direct URL to a draft post, they should be able to access it. An article can have both a published and a draft version. This allows improving upon an already published article. So draft articles must be available under a different URL. So let's create a new page for this.
Create a new file under /src/pages/blog/draft/[slug].astro
. In there, add the following code:
1---
2import { marked } from "marked";
3import BlogLayout from "../../../layouts/BlogLayout.astro";
4import Discard from "../../../components/Discard";
5import Visibility from "../../../components/Visibility";
6
7const { slug } = Astro.params;
8
9const draftResponse = await fetch(
10 `${import.meta.env.STRAPI_URL}/api/posts?populate=header&filters[slug][$eq]=${slug}&status=draft`
11);
12
13const draftData = await draftResponse.json();
14
15let post = draftData.data[0];
16
17if (!post) {
18 return new Response("Draft post not found", { status: 404 });
19}
20
21const headerImage = post.header?.url;
22---
23
24<BlogLayout title={post.title}>
25 <article class="prose prose-lg max-w-2xl mx-auto py-24">
26 {
27 headerImage && (
28 <img
29 src={`${import.meta.env.STRAPI_URL}${headerImage}`}
30 alt={post.title}
31 class="mb-6 w-full h-auto rounded-lg"
32 />
33 )
34 }
35 <div class="flex items-center justify-between mb-4">
36 <h1 class="mb-0">{post.title}</h1>
37 </div>
38 <div set:html={marked.parse(post.content)} />
39 </article>
40</BlogLayout>
We specifically search for draft versions of posts. While fetching, we filter by slug so we only get the relevant post.
1const draftResponse = await fetch(
2 `${import.meta.env.STRAPI_URL}/api/posts?populate=header&filters[slug][$eq]=${slug}&status=draft`
3);
4
5const draftData = await draftResponse.json();
The rest of the logic is the same as in the initial slug route for published posts.
If we now head to /blog/draft/our-preview-post
, we can access the draft post. However, it cannot be found on the /blog
overview of all published posts. Unless you add a published version.
NOTE:
/our-preview-post
in the link above is the slug of the draft entry.Adding a Draft Label
Currently, there is no way to tell if a post is a draft or published. We want to add a label to posts that aren't published yet to clearly mark them as a draft. Let's display a small label on draft posts to mark them as such.
This is the updated code:
1<BlogLayout title={post.title}>
2 <article class="prose prose-lg max-w-2xl mx-auto py-24">
3 {
4 headerImage && (
5 <img
6 src={`${import.meta.env.STRAPI_URL}${headerImage}`}
7 alt={post.title}
8 class="mb-6 w-full h-auto rounded-lg"
9 />
10 )
11 }
12 <div class="flex items-center justify-between mb-4">
13 <h1 class="mb-0">{post.title}</h1>
14 <p
15 class="text-base font-semibold bg-yellow-100 text-yellow-800 px-3 py-1 rounded flex-shrink-0"
16 >
17 Draft
18 </p>
19 </div>
20 <div set:html={marked.parse(post.content)} />
21 </article>
22</BlogLayout>
If we now access the draft post, it is going to have a label that marks it as a draft.
For the following section, we'll leverage the Strapi Document Service API to build a password-protected Publish, Unpublish, and Draft Discard action. On every article, we will show buttons to either publish a draft or unpublish an article, as well as a discard button for drafts. Since these actions are sensitive, only users who know a secret password will be able to perform them.
We are working with a separate frontend (Astro Site) and backend (Strapi CMS), all communications go through the REST API of our Strapi instance. By default, it doesn't support the actions we want to implement. However, Strapi offers unlimited customization by writing your own code.
To get started, open up the previously created Strapi instance in your code editor. The things we have configured in the Strapi UI already created all files for our Post model. We will now modify these files. First, open up the src/api/post/services/post.js
file and write the following code:
1// src/api/post/services/post.js
2
3'use strict';
4
5/**
6 * post service
7 */
8
9const { createCoreService } = require('@strapi/strapi').factories;
10
11module.exports = createCoreService('api::post.post', ({ strapi }) => ({
12 // Extend core service with custom document management functions
13
14 async publishDocument(id) {
15 return await strapi.documents('api::post.post').publish({
16 documentId: id,
17 });
18 },
19
20 async unpublishDocument(id) {
21 return await strapi.documents('api::post.post').unpublish({
22 documentId: id,
23 });
24 },
25
26 async discardDraftDocument(id) {
27 return await strapi.documents('api::post.post').discardDraft({
28 documentId: id,
29 });
30 },
31}));
This defines the three document actions we want to implement.
We now need to call them in the controller at src/api/post/controllers/post.js
:
1// src/api/post/controllers/post.js
2
3'use strict';
4
5/**
6 * post controller
7 */
8
9const { createCoreController } = require('@strapi/strapi').factories;
10
11module.exports = createCoreController('api::post.post', ({ strapi }) => ({
12 async publish(ctx) {
13 try {
14 const { id } = ctx.params;
15 const result = await strapi.service('api::post.post').publishDocument(id);
16 ctx.send(result);
17 } catch (err) {
18 ctx.throw(500, err);
19 }
20 },
21
22
23 async unpublish(ctx) {
24 try {
25 console.log('hello hello hello')
26 const { id } = ctx.params;
27 const result = await strapi.service('api::post.post').unpublishDocument(id);
28 ctx.send(result);
29 } catch (err) {
30 ctx.throw(500, err);
31 }
32 },
33
34
35 async discard(ctx) {
36 try {
37 const { id } = ctx.params;
38 const result = await strapi.service('api::post.post').discardDraftDocument(id);
39 ctx.send(result);
40 } catch (err) {
41 ctx.throw(500, err);
42 }
43 },
44}));
And lastly, create a new file src/api/post/routes/post-actions.js
to define the new API routes for these actions:
1// src/api/post/routes/post-actions.js
2module.exports = {
3 routes: [
4 {
5 method: "POST",
6 path: "/posts/:id/publish",
7 handler: "post.publish",
8 config: {
9 auth: {
10 strategies: ["api-token"],
11 },
12 },
13 },
14 {
15 method: "POST",
16 path: "/posts/:id/unpublish",
17 handler: "post.unpublish",
18 config: {
19 auth: {
20 strategies: ["api-token"],
21 },
22 },
23 },
24 {
25 method: "DELETE",
26 path: "/posts/:id/discard",
27 handler: "post.discard",
28 config: {
29 auth: {
30 strategies: ["api-token"],
31 },
32 },
33 },
34 ],
35};
publish()
, discard()
and discardDraft()
As you can see, we have defined the strategy as api-token
. This means those endpoints can only be called with a valid Bearer Token. So, let's create this token in the UI.
Go to Settings > API Token and click Create new API Token.
Name the token whatever you like and choose the token duration you want. The important part is the permissions. If you've followed all the steps, you should see the three actions we've added in the Permissions section. Only select these three since it's best practice to provision tokens with as little access as needed. With that, create the token and copy it.
Head back to your Astro codebase and open the .env
file. Paste in the token as STRAPI_TOKEN
. While you're at it, also define a SECRET
variable and set whatever value you like. We will use this as the password to execute these actions in the frontend.
Your Astro environment file .env
should now look like this:
STRAPI_URL=http://localhost:1337
STRAPI_TOKEN=your-token
SECRET=your-secret
We cannot call our Strapi API directly from the frontend, since it requires a secret token. We can only store that token in code that runs on the server. Otherwise, we risk exposing it to all users. Therefore, we need a very simple server endpoint. Luckily, Astro supports that out of the box.
Create a new file in the Astro project at src/pages/api/content.js
with this content:
1export const prerender = false;
2
3const STRAPI_URL = import.meta.env.STRAPI_URL;
4const STRAPI_TOKEN = import.meta.env.STRAPI_TOKEN;
5const SECRET = import.meta.env.SECRET;
6
7export const POST = async ({ request }) => {
8 if (request.headers.get("Content-Type") === "application/json") {
9 const body = await request.json();
10 const { action, postId, secret } = body;
11
12 if (!action || !postId || !secret) {
13 return new Response(
14 JSON.stringify({ error: "Missing action, postId, or secret" }),
15 { status: 400 }
16 );
17 }
18
19 if (secret !== SECRET) {
20 return new Response(JSON.stringify({ error: "Invalid secret" }), {
21 status: 401,
22 });
23 }
24
25 let endpoint = "";
26 switch (action) {
27 case "publish":
28 endpoint = `${STRAPI_URL}/api/posts/${postId}/publish`;
29 break;
30 case "unpublish":
31 endpoint = `${STRAPI_URL}/api/posts/${postId}/unpublish`;
32 break;
33 case "discard":
34 endpoint = `${STRAPI_URL}/api/posts/${postId}/discard`;
35 break;
36 default:
37 return new Response(JSON.stringify({ error: "Invalid action" }), {
38 status: 400,
39 });
40 }
41
42 try {
43 const strapiResponse = await fetch(endpoint, {
44 method: action === "discard" ? "DELETE" : "POST",
45 headers: {
46 Authorization: `Bearer ${STRAPI_TOKEN}`,
47 "Content-Type": "application/json",
48 },
49 });
50
51 if (!strapiResponse.ok) {
52 throw new Error(`Strapi API error: ${strapiResponse.statusText}`);
53 }
54
55 const data = await strapiResponse.json();
56 return new Response(JSON.stringify(data), { status: 200 });
57 } catch (error) {
58 console.error("Error calling Strapi API:", error);
59 return new Response(JSON.stringify({ error: "Internal server error" }), {
60 status: 500,
61 });
62 }
63 }
64
65 return new Response(JSON.stringify({ error: "Invalid content type" }), {
66 status: 400,
67 });
68};
In this section, we check that the client has sent the correct password. If that’s the case, we forward the request to Strapi along with the secret API Token. You can think of it as a type of proxy.
Up to this point, the whole application has not had any client-side code. Everything was rendered on the server and then sent to the client as static HTML. But to send actions to our API, we need some JavaScript on the client.
Vanilla JavaScript would be all you need for this. But we want to use this opportunity to show Astro's React integration. If you feel adventurous, you can use Vanilla JavaScript or another library like Vue or Svelte.
So, let's install React. All that's needed is one Astro command:
npx astro add react
Let's create the first component at src/components/Visibility.jsx
with:
1export default function Visibility({ postId, isDraft, slug }) {
2 const handleVisibilityChange = async () => {
3 const action = isDraft ? 'publish' : 'unpublish';
4 const secret = prompt(`Please enter the secret to ${action} this post:`);
5
6 if (!secret) {
7 console.log(`${action.charAt(0).toUpperCase() + action.slice(1)} operation cancelled`);
8 return;
9 }
10
11 try {
12 const response = await fetch('/api/content', {
13 method: 'POST',
14 headers: {
15 'Content-Type': 'application/json',
16 },
17 body: JSON.stringify({
18 action: action,
19 postId: postId,
20 secret: secret,
21 }),
22 });
23
24 if (!response.ok) {
25 throw new Error(`Failed to ${action} post`);
26 }
27
28 const data = await response.json();
29 console.log(`Post ${action}ed successfully`, data);
30 // Redirect to the appropriate URL based on the new state
31 const newPath = isDraft ? `/blog/${slug}` : `/blog/draft/${slug}`;
32 window.location.href = newPath;
33 } catch (error) {
34 console.error(`Error ${action}ing post:`, error);
35 alert(`Failed to ${action} post. Please try again.`);
36 }
37 };
38
39 return (
40 <button
41 onClick={handleVisibilityChange}
42 className={`px-4 py-2 ${
43 isDraft
44 ? 'bg-emerald-500 hover:bg-emerald-600 focus:ring-emerald-500'
45 : 'bg-amber-500 hover:bg-amber-600 focus:ring-amber-500'
46 } text-per-neutral-900 font-semibold rounded-md focus:outline-none focus:ring-2 focus:ring-opacity-50 transition-all duration-200 border border-per-neutral-200 shadow-sm hover:shadow-md`}
47 >
48 {isDraft ? 'Publish' : 'Unpublish'}
49 </button>
50 );
51}
This component is a button that toggles the visibility of a blog post between published and draft states, requiring a secret for authentication and updating the server via an API call. It takes the ID of the post and whether it's published or a draft as props.
A Published entry can be unpublished, and vice versa.
Now create the second component at src/components/Discard.jsx
with:
1import React from 'react';
2
3export default function Discard({ postId }) {
4 const handleDiscard = async () => {
5 // Prompt the user for the secret
6 const secret = prompt("Please enter the secret to discard this post:");
7
8 // If the user cancels the prompt or enters an empty string, abort the operation
9 if (!secret) {
10 console.log('Discard operation cancelled');
11 return;
12 }
13
14 try {
15 const response = await fetch('/api/content', {
16 method: 'POST',
17 headers: {
18 'Content-Type': 'application/json',
19 },
20 body: JSON.stringify({
21 action: 'discard',
22 postId: postId,
23 secret: secret,
24 }),
25 });
26
27 if (!response.ok) {
28 throw new Error('Failed to discard post');
29 }
30
31 const data = await response.json();
32 // Handle successful discard
33 console.log('Post discarded successfully', data);
34 // You might want to trigger some UI update here
35 } catch (error) {
36 console.error('Error discarding post:', error);
37 // Handle error (e.g., show an error message to the user)
38 alert('Failed to discard post. Please try again.');
39 }
40 };
41
42 return (
43 <button
44 onClick={handleDiscard}
45 className="px-4 py-2 bg-rose-500 text-per-neutral-900 font-semibold rounded-md hover:bg-rose-600 focus:outline-none focus:ring-2 focus:ring-rose-500 focus:ring-opacity-50 transition-all duration-200 border border-per-neutral-200 shadow-sm hover:shadow-md"
46 >
47 Discard
48 </button>
49 );
50};
Since draft entries can be discarded, this component renders a "Discard" button that, when clicked, prompts the user for a secret and sends a request to discard a post, handling the response and potential errors.
In src/pages/blog/[slug].astro
, add the Visibility
button into your markup:
1<BlogLayout title={post.title}>
2 <article class="prose prose-lg max-w-2xl mx-auto py-24">
3 {
4 headerImage && (
5 <img
6 src={`${import.meta.env.STRAPI_URL}${headerImage}`}
7 alt={post.title}
8 class="mb-6 w-full h-auto rounded-lg"
9 />
10 )
11 }
12 <div class="flex items-center justify-between mb-4">
13 <h1 class="mb-0">{post.title}</h1>
14 </div>
15 <div class="mb-6">
16 <Visibility client:load postId={post.documentId} isDraft={false} slug={slug} />
17 </div>
18 <p class="text-per-neutral-900 mt-2">{publishDate}</p>
19 <div set:html={marked.parse(post.content)} />
20 </article>
21</BlogLayout>
Observe the client:load
directive. This tells Astro to ship the JavaScript of this component to the client. This is essential for performing the API call. If you remove it, the button will still be rendered - without any of the functionality.
In src/pages/blog/draft/[slug].astro
, do the same thing, but this time with the Discard
component as well:
1<BlogLayout title={post.title}>
2 <article class="prose prose-lg max-w-2xl mx-auto py-24">
3 {
4 headerImage && (
5 <img
6 src={`${import.meta.env.STRAPI_URL}${headerImage}`}
7 alt={post.title}
8 class="mb-6 w-full h-auto rounded-lg"
9 />
10 )
11 }
12 <div class="flex items-center justify-between mb-4">
13 <h1 class="mb-0">{post.title}</h1>
14 <p
15 class="text-base font-semibold bg-yellow-100 text-yellow-800 px-3 py-1 rounded flex-shrink-0"
16 >
17 Draft
18 </p>
19 </div>
20 <div class="flex gap-2 mb-6">
21 <Visibility client:load postId={post.documentId} isDraft={true} slug={slug} />
22 <Discard client:load postId={post.documentId} />
23 </div>
24 <div set:html={marked.parse(post.content)} />
25 </article>
26</BlogLayout>
Create a draft in the Strapi admin. Make sure you know the link of this draft, since you only want the link to a draft to be accessible to you alone. Remember, only click Save and do not click Publish since this should be a draft.
Next, visit the draft and publish it. Recall that for this to work, you will use your secret key.
When we check the Strapi admin panel, we will see that the draft post is published.
Now that we have published this draft let's unpublish it.
This is what we will see when we visit the Strapi admin panel.
Go ahead and discard the draft. If successful, we will see this in the Strapi admin dashboard.
As you can see, the draft has been discarded and is no longer in the Strapi CMS admin dashboard.
In this tutorial, we've covered how to build a Blog Post Preview feature using Strapi 5 Draft and Publish feature along with Astro. This setup allows you to manage draft and published blog posts efficiently. Thanks to Astro, the website remains fast, and SEO-friendly and Strapi's Draft and Publish feature ensures a smooth content review process.
Using Strapi's Document Service API, we can control the visibility of posts directly from the Frontend. It remains secure because all requests are password protected through Astro's API endpoint.
You can find the complete code in this GitHub repository. Do checkout Strapi docs and Astro docs to learn more about building websites using Astro and Strapi.
Self employed full-stack developer