A Link Content Previewer with Svelte & Sapper

The inspiration for this post came from reading Ilona’s fantastic post on dev.to titled ”Frontend Shorts: How to create link content previewer with React, Vue, and Vanilla JavaScript”. She did an amazing job explaining all the details, for VanillaJS, React and Vue. Please have a look at her post for full details.

In summary, she developed a feature that displays a content preview when a user moves their mouse over a link.

Below is an animated gif Ilona created to illustrate how this looks in practice:

Link content previewer in action

Link content previewer in action

With all code in place as described in her post, the solution can be used as follows:

<a
	href="https://dev.to"
	onmouseover="showLinkPreview()"
	onmouseleave="hideLinkPreview()"
	class="link-with-preview"
	data-image="https://thepracticaldev.s3.amazonaws.com/i/6hqmcjaxbgbon8ydw93z.png"
	data-title="DEV Community 👩‍💻👨‍💻"
	data-text="Where programmers share ideas and help each other grow—A constructive and inclusive social network."
>
	dev.to.
</a>

There must be a way to dynamically create the preview… 🤔

This was my first thought when I read the blog post and looked at the source code. What if we fetched the link’s href source in the onmouseover handler and dynamically set the title, text and image?

A sample Sapper application is available on my Github profile at https://github.com/mootoday/sapper-link-preview.

Shouldn’t be too hard, eh? Use the browser’s built-in fetch in the onmouseover handler, parse the returned HTML and look for the page’s <title>, some sort of meta tag for the text and image.

Note: Continue to read before you try this out ;-)

❌ Cross-Origin Resource Sharing (CORS)

From the MDN web docs:

Cross-Origin Resource Sharing (CORS) is a mechanism that uses additional HTTP headers to tell browsers to give a web application running at one origin, access to selected resources from a different origin. A web application executes a cross-origin HTTP request when it requests a resource that has a different origin (domain, protocol, or port) from its own.

In other words, our website on https://domain-a.com can’t easily fetch content from https://domain-b.com. However, the whole point of the link content previewer is to do exactly that.

A backend service to the rescue

Alright then, if we can’t fetch the content on the client-side, let’s do it on the backend. There’s no CORS issues there and we can easily fetch any URL, parse its HTML and create the link preview.

Sapper server routes

In a Sapper application, we can use server routes to do the job for our link conten previewer.

All we need is the following server route at src/routes/link-preview.json.js:

import got from 'got';
import * as cheerio from 'cheerio';

// TODO: Expand to include other meta tags (e.g. Facebook, Twitter, etc.)
const getTitle = ($) => $('head title').text();
const getDescription = ($) => $('meta[name=description]').attr('content');
const getImgSrc = ($) => $("meta[property='og:image']").attr('content');

export const get = async (req, res) => {
	const { href } = req.query;
	const fetchResponse = await got(href);
	const $ = cheerio.load(fetchResponse.body);

	res.setHeader('Content-Type', 'application/json');
	res.end(
		JSON.stringify({
			title: getTitle($),
			description: getDescription($),
			imgSrc: getImgSrc($)
		})
	);
};

It exports a get function, which maps to a GET request to /link-preview.json. Since that endpoints is served from the same domain as the frontend application, we have no CORS issues.

We need two libraries:

  • got: A request library for Node.js
  • cheerio: An implementation of core jQuery designed for the server

In the get function, we first read the href request query parameter. The frontend can provide that when calling the endpoint with /link-preview.json?href=https://dev.to.

We send a GET request to that URL using got. Once the response comes back, we load the response body into cheerio, this helps us query the HTML in a way that’s similar to how jQuery works.

Lastly, the get function returns a JSON object with the title, description and imgSrc, all read from the HTML.

Note: As commented in the code, the getTitle, getDescription and getImgSrc functions can be extended at will to read different meta tags.

Test the backend service

In your Sapper project, run npm install and npm run dev to start the server. If you access http://localhost:3000/link-preview.json?href=https://dev.to, you’ll see the JSON response as expected.

Create a LinkPreview Svelte component

To use the above backend service, let’s create a <LinkPreview> Svelte component. The goal is to end up with the following usage of this component:

<LinkPreview href="https://dev.to">dev.to</LinkPreview>

Expressed as a user story:

As a frontend developer,
I want to use the <LinkPreview> component the same way I would use an <a> tag,
so that I can leverage all supported <a> attributes.

With these goals in mind, let’s start with a new src/components/link-preview.svelte file:

<a href="{$$props.href}" {...$$props}>
	<slot />
</a>

This gives us a <LinkPreview> component that simply wraps an <a> tag and passes all props to the underlying <a> tag. For example, <LinkPreview href="https://dev.to">dev.to</LinkPreview> renders as <a href="https://dev.to">dev.to</a>.

We could modify the existing link-preview.svelte component and add the HTML for the card and the mouseover and mouseleave event handlers. However, let’s instead create a separate component for that to encapsulate the behavior and design.

Create a src/components/link-preview-card.svelte file with basic Svelte boilerplate, plus the card HTML and CSS Ilona provided in her blog post:

<script></script>

<style>
	.card {
		width: 150px;
		font-size: 10px;
		color: black;
		position: absolute;
		z-index: 100;
		bottom: 30px;
		left: 50%;
		transform: translateX(-50%);
	}
	.card img {
		width: 150px;
	}
	.card-title {
		font-size: 14px;
	}
</style>

<div class="card">
	<img src="" class="card-img-top" />
	<div class="card-body">
		<h5 class="card-title"></h5>
		<p class="card-text"></p>
	</div>
</div>

Next, let’s define the title, description and imgSrc variables. For now, they’re empty, but we will soon use the backend service we created earlier to provide actual values. While we’re at it, we also update the HTML to use the new variables.

<script>
	let title = '';
	let description;
	let imgSrc;
</script>

<style>
	...;
</style>

<div class="card">
	<img src="{imgSrc}" alt="{title}" class="card-img-top" />
	<div class="card-body">
		<h5 class="card-title">{title}</h5>
		<p class="card-text">{description}</p>
	</div>
</div>

With that skeleton in place, it’s time to populate the three variables dynamically, based on the link’s href attribute.

We start with Svelte’s onMount() lifecycle method to call the backend service. What this means is that when the component gets mounted, we immediately call the backend service, which in turns fetches the href’s content and parses it. Once done, the preview card’s values are populated so that by the time a user moves their mouse over a link, the preview card displays the content.

<script>
	 import { onMount } from "svelte";

	 export let href;

	 ...

	 onMount(() => {
	   fetch("/link-preview.json?href=" + href)
	     .then(response => response.json())
	     .then(linkData => {
	       title = linkData.title;
	       description = linkData.description;
	       imgSrc = linkData.imgSrc;
	     });
	});
</script>

You notice the new href prop we defined. We will pass that in when we use this component.

Now, the HTML needs a few updates for better user experience. Not every website provides the meta tags our backend service looks for. For now, we simply hide the parts that are missing.

<div class="card">
	{#if imgSrc}
	<img src="{imgSrc}" alt="{title}" class="card-img-top" />
	{/if}
	<div class="card-body">
		<h5 class="card-title">{title}</h5>
		{#if description}
		<p class="card-text">{description}</p>
		{/if}
	</div>
</div>

To wrap up the development of this component, we need to indicate whether to show or hide this card. Remember, the preview card should only be displayed as long as the user’s mouse is over a particular link. The only place we can determine that is in the already created <LinkPreview> component where the <a> tag is. So let’s expose a show prop from the <LinkPreviewCard> component and wrap the entire HTML in an if statement.

<script>
	...

	export href;
	export show = false;

	...
</script>

{#if show}
<div class="card">
	{#if imgSrc}
	<img src="{imgSrc}" alt="{title}" class="card-img-top" />
	{/if}
	<div class="card-body">
		<h5 class="card-title">{title}</h5>
		{#if description}
		<p class="card-text">{description}</p>
		{/if}
	</div>
</div>
{/if}

By default, the card is hidden (export show = false;).

Use the new LinkPreviewCard component

Back in the src/components/link-preview.svelte component, we need to import the new preview card component first and add it inside the <a> tag.

<script>
	import LinkPreviewCard from './link-preview-card.svelte';
</script>

<a href="{$$props.href}" {...$$props}>
	<slot />
	<LinkPreviewCard href="{$$props.href}" />
</a>

The href prop we pass to the <LinkPreviewCard> component is the one we send to the backend service to fetch the preview content.

To properly style the preview card, the <a> tag needs a tiny bit of styling:

<style>
	a {
		position: relative;
	}
</style>

Deal with the mouseover and mouseleave events

Lastly, the preview card needs to be displayed when the mouse is over a link and hidden otherwise. We track this in a new variable:

<script>
	...

	let showPreviewCard = false;
</script>

Now, let’s add the mouseover and mouseleave handlers and pass the show prop to the <LinkPreviewCard> component.

<a {href} {...$$restProps}
  on:mouseover={() => showPreviewCard = true}
  on:mouseleave={() => showPreviewCard = false}
>
  <slot />
  <LinkPreviewCard {href} show={showPreviewCard} />
</a>

Use the LinkPreview component

The <LinkPreview> component can now be used anywhere in the application, for example in src/routes/index.svelte:

<script>
	import LinkPreview from '../components/link-preview.svelte';
</script>

<p>
	Move your mouse over the following link to see a preview of the link's content:
	<LinkPreview href="https://dev.to">dev.to.</LinkPreview>
</p>

Conclusion

Thanks to Ilona for posting the initial blog post on this topic!

As we saw in my addition, with a simple backend service, it is possible to load a link’s preview dynamically. The benefit of that is an always up-to-date preview even when a website we link to changes their title, description or main image.

If you like this kind of content, make sure to follow me on Bluesky @mootoday to get notified of new blog posts.