Quick-and-useful: A DIY, self-hosted Linktree and Linkinbio clone with Astro and TinaCMS
Quick-and-useful, part 2: A self-hosted Linktree and Linkinbio clone based on Astro with a user-friendly edit interface.
Quick-and-useful, part 2: A self-hosted Linktree and Linkinbio clone based on Astro with a user-friendly edit interface.
This is a part of the quick-and-useful series - a series of hands-on guides that teach you how to build small apps using Astro and a mix of other interesting web technologies.
As much as I dislike the many “walled gardens” of the modern web, most of us, present company included, often use them. Whether it’s peer pressure, my-family-is-there, or just plain convenience, we all have our reasons. Still, we managed to devise some interesting escape hatches. Two good examples are Linktree and Linkinbio. They are simple yet effective, allowing you to have a single link in your social media profiles that leads to a page with links to your other profiles, projects, etc.
Simple ideas are often easy to replicate. So, in this post, we’ll build a self-hosted Linktree and Linkinbio clone rolled in one.
Since this is a series of posts about Astro, we’ll use it as the foundation of our app in static-site generation mode. We’ll also use TinaCMS to provide a user-friendly editing interface. Beyond that, a bit of Tailwind for styling… and that’s it.
I’ll admit I started this experiment with a much bigger scope - I wanted to build a complete multi-user clone of the products mentioned above, with a database, authentication, etc. But that made me realize that such a post would be too long and likely have to be split into multiple parts. In my book, that doesn’t qualify as “quick-and-useful”.
With problem spaces like this, having a visual editor that allows you to edit the content more visually feels better. I started looking into headless CMSs, but most don’t match this use case. Then I remembered that TinaCMS is a thing - a CMS that piggy-backs on top of your git repo. In a sense, you can think of it as a more user-friendly and visual interface for markdown files that live on GitHub (or any of the other Gits* for that matter).
TinaCMS comes with a cloud version - which translates to never having to touch your code once you’re happy with the look and feel. Tina manages your GitHub repo’s contents, which, in turn, is used to generate your site on each content change. Which can also be automated via one of the hundreds of static-site hosting services out there.
As I further developed this application, I realized that TinaCMS plays nicely with the Astro content collections API. If you’d instead stick with plain ol’ markdown, you can do that too.
Assuming you already have Node.js 18+ installed, you can get started by running the following command on your CLI:
npm create astro@latest astro-ltree
As with the previous installment, pick and choose your preferred options.
Once the app is initialized, we can install TinaCMS:
npx @tinacms/cli@latest init
This will ask you a few questions about your project. Since these steps are well documented on the official Astro docs, I won’t go into details here. Have a look at the link for more information (but you can probably get away with just following the prompts on the screen).
Note: TinaCMS also has a similar guide about integrating with Astro.
Once the process is done, you should have a tina
folder in your project root with a config file inside. This is where we’ll define our content model.
One additional piece that’s useful to underline here is the changes to package.json.
Since TinaCMS is its own thing, we’d have to run it parallel with Astro. So, the dev
and start
scripts change from:
...
"scripts": {
"dev": "astro dev",
"start": "astro dev",
...
},
...
To this:
...
"scripts": {
"dev": "tinacms dev -c \"astro dev\"",
"start": "tinacms dev -c \"astro dev\"",
...
}
...
With those in place, we can get started with modeling our data!
After some thought and experimentation, I decided to go with four collections:
Bio
- the “collection” that will hold the app’s owner’s name, bio, and avatar.Posts
- a list of posts linking to a post in a walled garden of your choice.Links
- a list of links to be displayed on the page (your blog, projects, etc.)Socials
- links to your social media profiles.As I mentioned, we have two copies of these collections to define - one for TinaCMS and one for Astro. The former is defined in tinacms/config.ts
and the latter in src/content/config.ts
to ensure we can properly use the Astro content collections API.
Since the schema is a bit verbose on both ends, I’ll split each collection into sections.
Note: I’ve changed the path for each of Tina’s collections to src/content
to ensure we can use the Astro content collections API.
bio
collectionIn Tina:
...
{
name: "bio", // schema name
label: "Bio", // label that appears in the form
path: "src/content/bio", // path to the file where the data is stored
fields: [
// the form fields as they appear in the CMS
{
name: "name",
type: "string",
label: "Name",
required: true,
isTitle: true,
},
{
name: "biodescription",
type: "rich-text",
label: "Bio",
required: true,
isBody: true,
},
{
name: "avatar",
type: "image",
label: "Avatar",
required: true,
},
],
},
...
}
More on data modeling in TinaCMS here.
In Astro:
const bioCollection = defineCollection({
schema: z.object({
name: z.string(),
avatar: z.string(),
}),
});
More on Astro content collections here.
To the keen observer - you probably noticed the biodescription
field in the TinaCMS schema and its absence in the Astro schema. This is because setting the isBody
flag on a field in TinaCMS makes it the body text of the resulting markdown file. Astro handles that automatically, so we don’t need to define it explicitly in the schema.
links
collectionIn Tina:
{
...
name: "link",
label: "Links",
path: "src/content/links",
fields: [
{
type: "string",
name: "title",
label: "Title",
isTitle: true,
required: true,
},
{
type: "string",
name: "url",
label: "URL",
required: true,
},
{
type: "number",
name: "order",
label: "Order",
required: true,
},
],
...
}
In Astro:
const linksCollection = defineCollection({
schema: z.object({
title: z.string(),
url: z.string(),
order: z.number(),
}),
});
In this case, we have a 1:1 parity between the two schemas - we do not need a body, as links are just that - a title, a URL, and the order value determining where they are rendered.
socials
collectionIn Tina:
{
...
{
name: "socials",
label: "Socials",
path: "src/content/socials",
fields: [
{
type: "string",
name: "title",
label: "Title",
isTitle: true,
required: true,
},
{
type: "string",
name: "url",
label: "URL",
required: true,
},
{
type: "number",
name: "order",
label: "Order",
required: true,
},
{
type: "string",
name: "icon",
label: "Icon",
required: true,
list: true,
ui: {
component: "select",
},
options: [
"github",
"twitter",
"linkedin",
"instagram",
"facebook",
"youtube",
"twitch",
"tiktok",
"snapchat",
"reddit",
"pinterest",
"medium",
"dev",
"dribbble",
"behance",
"codepen",
"producthunt",
"discord",
"slack",
"whatsapp",
"telegram",
"email",
],
},
],
},
...
}
…Holy verbosity, Batman! 😱 Well, yes, but if we’d like a nice UI for the social media icons, we need to define them as options. There are probably smarter ways to achieve this. PRs open 😉
In Astro:
const socialsCollection = defineCollection({
schema: z.object({
title: z.string(),
url: z.string(),
order: z.number(),
icon: z
.array(
z.enum([
"github",
"twitter",
"linkedin",
"instagram",
"facebook",
"youtube",
"twitch",
"tiktok",
"snapchat",
"reddit",
"pinterest",
"medium",
"dev",
"dribbble",
"behance",
"codepen",
"producthunt",
"discord",
"slack",
"whatsapp",
"telegram",
"email",
])
)
.length(1),
}),
});
Again, we have a 1:1 parity between the two schemas. The only difference is that we have to define the icon
field as an array of length 1 to ensure we can use it as a string in the template (and render the appropriate icon).
Both files are a bit verbose, so I won’t paste them here. You can find them in the GitHub repo for the app:
If you go to http://localhost:4321/admin/index.html, you should see something like this:
We have our data modeled and our collections configured to be consumed by Astro and edited by TinaCMS. Now, we need to render them.
Since both concepts we’re trying to replicate are single-page apps, which are pretty minimal, we’ll do the same - have one page to serve as our “Linktree” and another one to serve us our “Linkinbio”.
index
page (aka Linktree)Since we went through the trouble of neatly modeling our data in separate collections, we can use the Astro content collections API to fetch the data and render it in the template.
Linktree pages are usually a combination of a picture, name, maybe a short bio, a list of links, and some social media icons (which are also links):
---
import { getCollection } from "astro:content";
import SocialIcon from "../components/SocialIcon.astro";
// Fetch bio, links and socials
const bio = await getCollection("bio");
const links = await getCollection("links");
const socials = await getCollection("socials");
// Get the first item from bio, since that's our profile
const profile = bio[0];
// Render the contents of the bio body (the description)
const { Content } = await profile.render();
---
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
<meta name="viewport" content="width=device-width" />
<meta name="generator" content={Astro.generator} />
<title>LTree | {profile.data.name}</title>
</head>
<body>
<main class="flex flex-col items-center justify-center p-4 pt-10">
<!-- Top section: image, name, description -->
<img src={profile.data.avatar} alt="avatar" class="w-32 h-32 rounded-full" />
<h1 class="text-2xl mt-4">{profile.data.name}</h1>
<section class="text-sm max-w-[400px]">
<Content />
</section>
<!-- Navigation to get us around -->
<nav>
<ul class="flex divide-x divide-blue-700 p-2">
<li class="text-lg"><a class="block px-2 text-blue-500" href="/">Links</a></li>
<li class="text-lg"><a class="block px-2 text-blue-500" href="/postlinks">Posts</a></li>
</ul>
</nav>
<!-- Mid section: links -->
<ul class="flex flex-col gap-y-4 pt-10 min-w-[400px]">
{
links.sort((a, b) => {
if (a.data.order < b.data.order) {
return -1;
}
if (a.data.order > b.data.order) {
return 1;
}
return 0;
}).map((link) => (
<li class="border border-black border-2 w-full text-center p-4 text-xl font-semibold">
<a href={link.data.url} class="block">
{link.data.title}
</a>
</li>
))
}
</ul>
<!-- Bottom section: socials -->
<ul class="flex gap-4 items-center justify-center flex-wrap pt-10">
{
socials.sort((a, b) => {
if (a.data.order < b.data.order) {
return -1;
}
if (a.data.order > b.data.order) {
return 1;
}
return 0;
}).map((social) => (
<li class="border border-black border-2 rounded-full">
<a href={social.data.url} class="block p-4">
// This is an Astro component that renders the appropriate icon we picked from the list -> https://github.com/DBozhinovski/astro-ltree/blob/master/src/components/SocialIcon.astro
<SocialIcon id={social.data.icon[0]}>
</a>
</li>
))
}
</ul>
</main>
</body>
</html>
postlinks
page (aka Linkinbio)Linkinbio pages usually combine a picture, a name, and a list of posts that link somewhere. Again, our initial data modeling effort pays off:
---
import { getCollection } from "astro:content";
import SocialIcon from "../components/SocialIcon.astro";
// Fetch bio, socials and posts
const bio = await getCollection("bio");
const socials = await getCollection("socials");
const posts = await getCollection("posts");
// Get the first item from bio, since that's our profile
const profile = bio[0];
---
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
<meta name="viewport" content="width=device-width" />
<meta name="generator" content={Astro.generator} />
<title>LTree | {profile.data.name}</title>
</head>
<body>
<main class="flex flex-col items-center justify-center p-4 pt-10">
<!-- Top section: image and name -->
<img src={profile.data.avatar} alt="avatar" class="w-32 h-32 rounded-full" />
<h1 class="text-2xl mt-4">{profile.data.name}</h1>
<!-- Navigation to get us around -->
<nav>
<ul class="flex divide-x divide-blue-700 p-2">
<li class="text-lg"><a class="block px-2 text-blue-500" href="/">Links</a></li>
<li class="text-lg"><a class="block px-2 text-blue-500" href="/postlinks">Posts</a></li>
</ul>
</nav>
<!-- Socials -->
<ul class="flex gap-4 items-center justify-center flex-wrap pt-10">
{
socials.sort((a, b) => {
if (a.data.order < b.data.order) {
return -1;
}
if (a.data.order > b.data.order) {
return 1;
}
return 0;
}).map((social) => (
<li class="border border-black border-2 rounded-full">
<a href={social.data.url} class="block p-4">
// This is an Astro component that renders the appropriate icon we picked from the list -> https://github.com/DBozhinovski/astro-ltree/blob/master/src/components/SocialIcon.astro
<SocialIcon id={social.data.icon[0]}>
</a>
</li>
))
}
</ul>
<ul class="grid grid-cols-3 gap-1 pt-20 max-w-[640px] w-full">
{
// Finally, posts:
posts.sort((a, b) => {
if (a.data.date < b.data.date) {
return 1;
}
if (a.data.date > b.data.date) {
return -1;
}
return 0;
}).map((p) =>
<li>
<a href={p.data.url}>
<img src={p.data.image} alt={p.data.title} class="aspect-square col-span-1 object-cover" />
</a>
</li>
)
}
</ul>
</main>
</body>
</html>
With those two pages in place, you should be able to see something like this.
By now, we have a working app (which you’re welcome to clone and adapt), but the ultimate goal is to have a user-friendly, visual way to edit the content. TinaCMS is currently running in local mode, but we can deploy our project, configure Tina Cloud, and have a nice UI to edit the content.
Since the process is well documented, I’ll leave some links and go over the steps briefly. The full docs are here.
The steps, in the order I did them:
npx @tinacms/cli init backend
, and follow the instructions for the key and the token.tinacms/config.ts
file to use the .env variables.build
script in package.json
to tinacms build && astro check && astro build
.If you decide to use TypeScript when initializing the app, you’ll likely get a whole load of undefined
errors when running the full build
script. Funny enough, this one’s on Astro, not TinaCMS. The reason is that TypeScript tries to type-check the auto-generated TinaCMS scripts in public. I’ll admit, some swearing was involved, but luckily, the fix was pretty simple - add "exclude": ["public/**"]
to your tsconfig.json
file.
Finally, once you deploy these changes and they’re up and running, you should see TinaCMS running at /admin/index.html
on your deployed app. With that, we have a fully functional Linktree and Linkinbio clone with a user-friendly editing interface. So, goal achieved, I guess 🤔
But why stop there? This does seem like a good candidate for an Astro theme, doesn’t it?
Astro themes aren’t a complicated idea. Basically, if we pull out the me-specific parts of the app, we get a generic template, which we can reuse. With the approach we took, that means:
And, well, that’s it. You can find the resulting theme here.
We can always make things better, right? Here are some future ideas for the Astro theme I’d like to explore:
This was a fun little experiment that produced a useful result. I have a copy running at https://ltree.darko.io as my personal “router” for the various walled gardens of the web. If you’d like one of your own, you can follow the steps above or just use the theme and customize it to your liking.
As always, if you have any questions, comments, or suggestions, feel free to reach out on Twitter. Until next time, happy hacking! 🚀