Skip to content

oetherington/bluesky-embed-react

Repository files navigation

Bluesky Embed React

npm Storybook CI Coverage License

Lightweight configurable React component for embedding posts, profiles and feeds from Bluesky.

Bluesky Embed React is an independent project, not affiliated with Bluesky.

Demo and Examples

https://etherington.xyz/bluesky-embed-react

Light mode embed

Dark mode embed

Install

npm install bluesky-embed-react

or

yarn add bluesky-embed-react

Basic Usage

Importing

import {
	BlueskyPost,
	BlueskyProfilePosts,
	BlueskyConfigProvider,
} from "bluesky-embed-react";

Embed a post

<BlueskyPost userHandle="bsky.app" postId="3l6oveex3ii2l" />
  • userHandle The username or DID of the user who created the post
  • postId The ID of the post to embed

For instance, in the post https://bsky.app/profile/bsky.app/post/3l3t5pvpm222b the userHandle is "bsky.app" and the postId is "3l3t5pvpm222b".

Embed a users profile feed

<BlueskyProfilePosts userHandle="bsky.app" />
  • userHandle The username or DID of the user to embed
  • pageSize The number of posts to display (and fetch if using infinite loading)
  • infiniteLoad Enable inifite loading when scrolling to the end of the list

Configuration

Advanced configuration can be done by wrapping the components in a BlueskyConfigProvider:

<BlueskyConfigProvider hideAvatars>
	<BlueskyPost userHandle="bsky.app" postId="3l6oveex3ii2l" />
</BlueskyConfigProvider>

The props are of type BlueskyConfig which has the following properties:

  • app The base URL to use for outward links (default https://bsky.app)
  • service The base URL for the API (default https://public.api.bsky.app)
  • openLinksInNewTab Open links in place or in a new tab (default false)
  • avatarSize The size of user avatars in pixels (default 42)
  • hideAvatars Whether or not to hide user avatars (default false)
  • hideEmbeds Whether or not to hide embedded media and links (default false)
  • textPrimaryColor CSS color string for primary text (default light-dark(#0b0f14, #f1f3f5))
  • textSecondaryColor CSS color string for secondary text (default light-dark(#42576c, #aebbc9))
  • anchorColor CSS color string for links (default light-dark(#1083fe, #208bfe))
  • backgroundColor CSS color string for post backgrounds (default light-dark(#fff, #161e27))
  • borderColor CSS color string for borders (default light-dark(#d4dbe2, #2e4052))
  • loadingShimmer CSS color string for the loading shimmer effect (default linear-gradient(100deg, light-dark(#d5d5d5, #aaa) 40%, light-dark(#dbdbdb, #bdbdbd) 50%, light-dark(#d5d5d5, #aaa) 60%))
  • fontFamily CSS font stack to apply to all posts (default InterVariable, Inter, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji")
  • fontSize CSS font size for main text (default 15px)
  • embedFontSize CSS font size for embedded media descriptions (default 14px)
  • fontWeight CSS font weight for main text (default 400)
  • titleFontWeight CSS font weight for title text (default 600)
  • lineHeight CSS line height for all text (default 140%)
  • grid Grid size in pixels used for calculating all margins and padding (default 8px)
  • borderRadius CSS border radius applied to posts and loaders (default 6px)
  • width CSS width applied to posts (default 600px)
  • formatShortDate Function to format dates into a short format
  • formatLongDate Function to format dates into a long format

Switching between light mode and dark mode

The default colors use the CSS light-dark color function to automatically switch colors based on the current operating system setting.

To enable this behaviour you need to set the CSS color-scheme property to light dark:

:root {
	color-scheme: light dark;
}

You can also override the operating system to choose one theme or the other by setting color-scheme to either light or dark. If you don't set color-scheme then the browser will default to the light theme, even if the user has set their operating system to dark mode.

For more fine grained theme control you can set the color options manually in a BlueskyConfig provider (see above).

SSR

You can use the raw API to fetch data during SSR. For instance, in NextJS:

import {
	BlueskyPostsList,
	getBlueskyClient,
	getBlueskyProfilePosts,
} from "bluesky-embed-react";

export default function Page({ profile, posts }) {
	return <BlueskyPostsList profile={profile} posts={posts} />;
}

export const getStaticProps = async () => {
	const client = getBlueskyClient();
	const [profile, posts] = await Promise.all([
		getBlueskyProfile(client, "bsky.app"),
		getBlueskyProfilePosts(client, "bsky.app"),
	]);
	return {
		props: {
			profile,
			posts,
		},
	};
};

License

MIT © oetherington. See the included COPYING file for details.