Pliny provides out of the box components to enhance your static site:
- Analytics
- Google Analytics
- Plausible Analytics
- Simple Analytics
- Umami Analytics
- Posthog
- Comments
- Disqus
- Giscus
- Utterances
- Newsletter (uses Next 13 API Routes)
- Buttondown
- Convertkit
- Email Octopus
- Klaviyo
- Mailchimp
- Revue
- Command palette search with tailwind style sheet
- Algolia
- Kbar (local search)
- UI utility components
- Bleed
- Newsletter / Blog Newsletter
- Pre / Code block
- Table of Contents
as well as a bunch of MDX and contentlayer utility functions which I use to build Tailwind Nextjs Starter Blog and my own sites.
It is based on Next.js, Tailwind CSS and Contentlayer. For an example of how all the components can be used together, check out the Tailwind Nextjs Starter Blog.
Note: The previous cli and starter template have been deprecated. Please use the new components directly in your favourite Next 13 websites.
Note 2: The components are intended to be use within Next 13 app directory setup with Contentlayer. You might still be able to use the components in older websites but there's no official support for it, especially since many components are now using next/navigation
instead of next/router
.
This project is still in beta. Please report any issues or feedbacks.
npm i pliny
As many of the components are styled with tailwindcss, you will need to include the path to the library within the content
section of your tailwind config file:
module.exports = {
content: [
'./node_modules/pliny/**/*.js',
'./pages/**/*.{html,js}',
'./components/**/*.{html,js}',
],
// ...
}
The Analytics
component provides an easy interface to switch between different analytics providers. It might not be as feature rich as the official analytics providers but it should be sufficient for simple use cases.
All components default to the hosted service, but can be configured to use a self-hosted or proxied version of the script by providing the src
/ apiHost
props to the respective analytics component.
Note: As an external script will be loaded, do ensure that script-src
in the content security policy of next.config.js
has been configured to whitelist the domain.
import { Analytics, AnalyticsConfig } from 'pliny/analytics'
const analytics: AnalyticsConfig = {
// If you want to use an analytics provider you have to add it to the
// content security policy in the `next.config.js` file.
// supports Plausible, Simple Analytics, Umami, Posthog or Google Analytics.
plausibleAnalytics: {
plausibleDataDomain: '', // e.g. tailwind-nextjs-starter-blog.vercel.app
},
simpleAnalytics: {},
umamiAnalytics: {
umamiWebsiteId: '', // e.g. 123e4567-e89b-12d3-a456-426614174000
},
posthogAnalytics: {
posthogProjectApiKey: '', // e.g. 123e4567-e89b-12d3-a456-426614174000
},
googleAnalytics: {
googleAnalyticsId: '', // e.g. G-XXXXXXX
},
}
export default function Layout() {
return (
...
<Analytics analyticsConfig={analyticsConfig} />
)
}
You can also use the individual analytics components directly.
import { GA } from 'pliny/analytics/GoogleAnalytics'
const googleAnalyticsId = '' // e.g. UA-000000-2 or G-XXXXXXX
export default function Layout() {
return (
...
<GA googleAnalyticsId={googleAnalyticsId} />
)
}
import { Plausible } from 'pliny/analytics/Plausible'
const plausibleDataDomain = '' // e.g. tailwind-nextjs-starter-blog.vercel.app
export default function Layout() {
return (
...
<Plausible plausibleDataDomain={plausibleDataDomain} />
)
}
import { SimpleAnalytics } from 'pliny/analytics/SimpleAnalytics'
export default function Layout() {
return (
...
<SimpleAnalytics />
)
}
import { Umami } from 'pliny/analytics/Umami'
const umamiWebsiteId = '' // e.g. 123e4567-e89b-12d3-a456-426614174000
export default function Layout() {
return (
...
<Umami umamiWebsiteId={umamiWebsiteId} />
)
}
import { Posthog } from 'pliny/analytics/Posthog'
const posthogProjectApiKey: '', // e.g. AhnJK8392ndPOav87as450xd
export default function Layout() {
return (
...
<Posthog posthogProjectApiKey={posthogProjectApiKey} />
)
}
The Comments
component provides an easy interface to switch between different comments providers.
import { Comments, CommentsConfig } from 'pliny/comments'
import siteMetadata from '@/data/siteMetadata'
export default function BlogComments({ slug }: { slug: string }) {
return <Comments commentsConfig={commentsConfig as CommentsConfig} slug={slug} />
}
You can also use the individual comments components directly.
import { Giscus, GiscusProps } from 'pliny/comments/Giscus'
export default function BlogComments(props: GiscusProps) {
return <Giscus {...props} />
}
import { Disqus, DisqusProps } from 'pliny/comments/Disqus'
export default function BlogComments(props: DisqusProps) {
return <Disqus {...props} />
}
import { Utterances, UtterancesProps } from 'pliny/comments/Utterances'
export default function BlogComments(props: UtterancesProps) {
return <Utterances {...props} />
}
The Newsletter
component provides a Next 13 API route to integrate a newsletter subscription API with various providers. E.g. in app/api/newsletter/route.ts
import { NewsletterAPI } from 'pliny/newsletter'
import siteMetadata from '@/data/siteMetadata'
const handler = NewsletterAPI({
provider: '', // Use one of mailchimp, buttondown, convertkit, klaviyo, revue, emailOctopus
})
export { handler as GET, handler as POST }
You can then send a POST
request to the API route with a body with the email - { email: '[email protected]' }
. See the NewsletterForm
component in pliny/ui/NewsletterForm
for an example.
The Search
component provides an easy interface to switch between different search providers. If you are using algolia, you will need to import the css file as well - import 'pliny/search/algolia.css'
.
import { SearchProvider, SearchConfig } from 'pliny/search'
export default function Layout() {
return <SearchProvider searchConfig={searchConfig as SearchConfig}>...</SearchProvider>
}
You can also use the individual search components directly.
You can pass in an optional defaultActions
to kbarConfig
to customize the default actions. See Kbar documentation for more details.
import { KBarSearchProvider } from 'pliny/search/KBar'
export default function Layout() {
return <KBarSearchProvider kbarConfig={{ searchDocumentsPath: 'abc' }}>...</KBarSearchProvider>
}
Use KBarButton
to add a button which toggles the command palette on click event.
import { KBarButton } from 'pliny/search/KBarButton'
export default function SearchButton() {
return (
<KBarButton aria-label="Search Content">
<svg
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
strokeWidth={1.5}
stroke="currentColor"
className="w-6 h-6"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M21 21l-5.197-5.197m0 0A7.5 7.5 0 105.196 5.196a7.5 7.5 0 0010.607 10.607z"
/>
</svg>
</KBarButton>
)
}
import 'pliny/search/algolia.css'
import { AlgoliaSearchProvider } from 'pliny/search/Algolia'
export default function Layout() {
return (
<AlgoliaSearchProvider
algoliaConfig={{
appId: 'R2IYF7ETH7',
apiKey: '599cec31baffa4868cae4e79f180729b',
indexName: 'docsearch',
}}
>
...
</AlgoliaSearchProvider>
)
}
Use AlgoliaButton
to add a button which toggles the command palette on click event.
import { AlgoliaButton } from 'pliny/search/AlgoliaButton'
export default function SearchButton() {
return (
<AlgoliaButton aria-label="Search Content">
<svg
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
strokeWidth={1.5}
stroke="currentColor"
className="w-6 h-6"
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M21 21l-5.197-5.197m0 0A7.5 7.5 0 105.196 5.196a7.5 7.5 0 0010.607 10.607z"
/>
</svg>
</AlgoliaButton>
)
}
Add the plugins to remarkPlugins
in contentlayer or other MDX processors.
Extracts frontmatter from markdown file and adds it to the file's data object. Used to pass frontmatter fields to subsequent remark / rehype plugins.
Parses title from code block and inserts it as a sibling title node.
Converts markdown image nodes to next/image jsx.
Extracts TOC headings from markdown file and adds it to the file's data object. Alternatively, it also exports a extractTocHeadings
function which can be used within contentlayer to create a computedField
with the TOC headings.
While these can be used in any React code, they can also be passed down as MDXComponents and used within MDX files.
Useful component to break out of a constrained-width layout and fill the entire width.
Simple code block component with copy to clipboard button.
Table of contents component which can be used within a markdown file. asDisclosure
will wrap the TOC in a details
element with a summary
element. collapse
will collapse the TOC when AsDisclosure
is true. Modify the list style by passing in a ulClassName
prop. For example, if you are using Tailwind css and want to revert to the default HTML list style set ulClassName="[&_ul]:list-[revert]"
.
Newsletter form component to add a subscriber to your mailing list.