Adding Comments to a Docusaurus Blog

The features that docusaurus comes with out of the box are great, but there's one piece that's missing, especially for a blog - comments.

The features that docusaurus comes with out of the box are great, but there's one piece that's missing, especially for a blog - comments.

That being said, it doesn't really surprise me, considering the number of options available for comments and the fact that docusaurus is primarily for documentation.

Nevertheless, I went searching for some options and decided to add Talkyard to my site.

Setup

I decided to self-host a talkyard server, so the first thing I did was setup a VM inside of Azure, and point a domain name toward the public IP of the VM.

Another thing that will be helpful is to have an email service. I already had a SendGrid account, so that was taken care of.

⚠️
When setting up the talkyard server, be sure to add your blog domain to the "Allow embedding from" settings

Install Talkyard

Once the initial server was setup, it was time to install talkyard. There's not much to explain for this, and the instructions are super easy to follow

Swizzling the BlogPostPage

In order to get comments to show up, we need a way to add them to every blog post. To do this, we can swizzle the BlogPostPage by running

npm run swizzle @docusaurus/theme-classic BlogPostPage -- --eject

This will add an index.js file to /src/theme/BlogPostPage and blogs will now use this component instead of the default component.

# Folder contents
- blog
- docs
- src
  - theme
    - BlogPostPage
      - index.js
⚠️
When ejecting the BlogPostPage, it means that we'll now need to maintain that piece, and it could break during future upgrades.

Create a Comment Component

Since Docusaurus is built with React, we can add a comments by creating a new React component. To do this, add an index.tsx file under src/components/TalkyardComments

# Folder contents
- blog
- docs
- src
  - components
    - TalkyardComments
      - index.tsx

Next, you can add the following code. Be sure to replace TALKYARD_SERVER_URL and TALKYARD_SCRIPT_URL with appropriate values from the Embedded comments section of the Talkyard admin settings.

import React, { useEffect } from "react";

export default function TalkyardComments() {
  // Runs when the page loads
  useEffect(() => {
    // When loading the talkyard script, there's a "talkyard-session" iframe
    // that adds a white bar to the bottom of the screen, under the footer.
    // This function looks for that element, and hides it
    const hideTalkyardIframe = (id) => {
      try {
        let iframe = document.getElementById("talkyard-session");
        if (iframe.style.display !== "none") iframe.style.display = "none";

        if (id && iframe.style.display === "none") clearInterval(id);
      } catch (e) {
        console.log("error:", e);
      }
    };

    // Add the talkyard scripts to the page
    const serverScript = document.createElement("script");
    serverScript.id = "talkyard_url";
    serverScript.innerText = "talkyardServerUrl='<TALKYARD_SERVER_URL>;";
    (
      document.getElementsByTagName("head")[0] ||
      document.getElementsByTagName("body")[0]
    ).appendChild(serverScript);

    const talkyardScript = document.createElement("script");
    talkyardScript.async = true;
    talkyardScript.defer = true;
    talkyardScript.id = "talkyard_script";
    talkyardScript.src = "<TALKYARD_SCRIPT_URL>";
    (
      document.getElementsByTagName("head")[0] ||
      document.getElementsByTagName("body")[0]
    ).appendChild(talkyardScript);

    // Try hiding the talkyard iFrame after it's loaded via the scripts above
    let interval = setInterval(() => hideTalkyardIframe(interval), 100);

    // When leaving the page, remove all of the elements that were added by
    // Talkyard. Not doing this will cause duplicate elements to show up for
    // for every blog post you visit
    return () => {
      try {
        document.getElementById("talkyard_script").remove();
        document.getElementById("talkyard_url").remove();
        document.getElementById("talkyard-session").remove();
        document.getElementById("ed-editor-wrapper").remove();
        document.getElementById("ed-editor-placeholder").remove();
        clearInterval(interval);
      } catch (e) {
        console.log("Error:", e);
      }
    };
  }, []);

  // The actual comments that show up on the page
  return (
    <div className="talkyard-comments" style={{ marginTop: "45px" }}>
      <noscript>Please enable Javascript to view comments.</noscript>
      <p style={{ marginTop: "25px", opacity: "0.9", fontSize: "96%" }}>
        Comments powered by
        <a href="https://www.talkyard.io">Talkyard</a>.
      </p>
    </div>
  );
}

Add Comments to the BlogPostPage

Now that we have a component for displaying comments, we can update the BlogPostPage component that we swizzled earlier.

Within the BlogPostPage component, there's a BlogPostContent component. That's where we want to add our new Talkyard component:

function BlogPostPageContent(props) {
  const { content: BlogPostContents, sidebar } = props;
  const { assets, metadata } = BlogPostContents;
  const { nextItem, prevItem, frontMatter } = metadata;
  const {
    hide_table_of_contents: hideTableOfContents,
    toc_min_heading_level: tocMinHeadingLevel,
    toc_max_heading_level: tocMaxHeadingLevel,
  } = frontMatter;

  return (
    <BlogLayout
      sidebar={sidebar}
      toc={
        !hideTableOfContents &&
        BlogPostContents.toc &&
        BlogPostContents.toc.length > 0 ? (
          <TOC
            toc={BlogPostContents.toc}
            minHeadingLevel={tocMinHeadingLevel}
            maxHeadingLevel={tocMaxHeadingLevel}
          />
        ) : undefined
      }
    >
      <BlogPostItem
        frontMatter={frontMatter}
        assets={assets}
        metadata={metadata}
        isBlogPostPage
      >
        <BlogPostContents />
      </BlogPostItem>

      {/* I put the comments here: Under the actual post, but before the next
      and previous buttons. */}
      <TalkyardComments />

      {(nextItem || prevItem) && (
        <BlogPostPaginator nextItem={nextItem} prevItem={prevItem} />
      )}
    </BlogLayout>
  );
}

Conclusion

So there you have it - Talkyard comments added to a Docusaurus blog. And since we created a component, we can easily add comments to our documents, pages, or anywhere else.