Next.js Discord

Discord Forum

Changelog Page Predernering

Answered
Scale parasitoid posted this in #help-forum
Open in Discord
Scale parasitoidOP
Hey! I recently released a changelog page on my website @ https://termed.me/changelog. However I'm having some issues with seemingly predernging it? Applications like discord and general embedding tools can't seem to get the metadata to add embeds however other pages do already work. When initially loading the page there is a completely white screen before other content gets rendered. I took inspiration from https://raycast.com/changelog who also use NextJS for their changelog compoents so I'm not really sure what I'm doing wrong/differently. Thanks a lot in advance!

Notes: I pre Serialize everything on the server side as shown in one of the images, and the chanelog entry function is very simple simply using mdx-remote. Any idea whats going wrong?

heres package.json incase that helps in any way
{
  "name": "termed-me",
  "version": "0.1.0",
  "private": true,
  "scripts": {
    "dev": "next dev",
    "build": "next build",
    "start": "next start",
    "lint": "next lint"
  },
  "prisma": {
    "schema": "src/prisma/schema.prisma"
  },
  "dependencies": {
    "@headlessui/react": "^2.2.0",
    "@next/mdx": "^15.1.7",
    "@prisma/client": "latest",
    "@radix-ui/react-dialog": "^1.1.4",
    "@radix-ui/react-scroll-area": "^1.2.1",
    "@radix-ui/react-select": "^2.1.6",
    "@radix-ui/react-slot": "^1.1.0",
    "@radix-ui/react-tabs": "^1.1.1",
    "@stripe/stripe-js": "^5.6.0",
    "@vercel/analytics": "^1.5.0",
    "@vercel/og": "^0.6.4",
    "axios": "^1.7.7",
    "class-variance-authority": "latest",
    "clsx": "latest",
    "date-fns": "^4.1.0",
    "framer-motion": "^11.15.0",
    "gray-matter": "^4.0.3",
    "gsap": "latest",
    "lenis": "latest",
    "locomotive-aos": "latest",
    "lucide-react": "^0.462.0",
    "motion": "latest",
    "next": "latest",
    "next-mdx-remote": "^5.0.0",
    "next-nprogress-bar": "latest",
    "prisma": "latest",
    "react": "latest",
    "react-confetti-explosion": "^2.1.2",
    "react-dom": "latest",
    "react-dropzone": "^14.3.5",
    "react-icons": "latest",
    "react-medium-image-zoom": "^5.2.12",
    "recharts": "^2.15.0",
    "server-only": "latest",
    "sonner": "^1.7.0",
    "stripe": "^17.6.0",
    "tailwind-merge": "latest",
    "tailwindcss-animate": "latest"
  },
  "devDependencies": {
    "@types/node": "latest",
    "@types/react": "latest",
    "@types/react-dom": "latest",
    "autoprefixer": "latest",
    "postcss": "latest",
    "prettier": "latest",
    "prettier-plugin-tailwindcss": "latest",
    "tailwindcss": "3.4.17",
    "typescript": "latest"
  }
}
Answered by Scale parasitoid
Basically I just had to forwardRef the components that I was passing to MDX, for whatever reason that fixed the useState being null in it. Once that happened I rebuilt it and now when navigating to the page everything works as expected!
View full answer

70 Replies

Scale parasitoidOP
Let me know if I need to give you any more information at all
Edit: The next config didn't build so I reverted to an older one
import createMDX from "@next/mdx";

/** @type {import('next').NextConfig} */
const nextConfig = {
    pageExtensions: ["js", "jsx", "mdx", "ts", "tsx"],
    images: {
        remotePatterns: [
            {
                protocol: "https",
                hostname: "r2.resynced.design",
            },
        ],
    },
};

const withMDX = createMDX({
    options: {
        remarkPlugins: [],
        rehypePlugins: [],
    },
});

export default withMDX(nextConfig);
Example:
please ping me in any responses
Scale parasitoidOP
Got this error aswell
@Scale parasitoid please ping me in any responses
Mallow bee
I get a 500 error on initial page load
you'll need to check server logs to see exactly what that error is
@Mallow bee I get a 500 error on initial page load
Scale parasitoidOP
ohh ok lemme lookinto it
a
⨯ [TypeError: Cannot read properties of null (reading 'useState')] {
digest: '3480115524'
}
I got this error
Mallow bee
that's often caused by multiple versions of react, are you doing anything weird like a monorepo or importing react components from a library?
@Mallow bee that's often caused by multiple versions of react, are you doing anything weird like a monorepo or importing react components from a library?
Scale parasitoidOP
Well no, my package.json is above in the message and its a single website in the repo. all im using is using next-mdx-remote
@Scale parasitoid Got this error aswell
Scale parasitoidOP
Yeah I'm getting this error in dev mode here but im super confused as to why
do you need the components or anything?
Mallow bee
try npm ls react react-dom
just to make sure it installed the same versions, can be iffy when they're both floating at "latest"
@Mallow bee just to make sure it installed the same versions, can be iffy when they're both floating at "latest"
Scale parasitoidOP
now
i MIGHT be stupid
but thats a lot of instances of react and react dom
Mallow bee
that's all ok, it's deduped them
as long as they're all 18.3.1 then no issue there
It shows them like that because for example every radix package also imports react
@Mallow bee It shows them like that because for example every radix package also imports react
Scale parasitoidOP
yeah they seem to all be 18.3.1
Mallow bee
Is it possible you're breaking the rules of hooks in your component? most common reason is an early return before you call a hook
or conditional hook or running one in a loop
the react-hooks eslint plugin is good for catching these, it's pretty much the only eslint plugin I use
Scale parasitoidOP
well this is the main entry component., evverything else just passes data down to it
import { MDXRemote } from "next-mdx-remote";
import Link from "next/link";
import { format } from "date-fns";
import type { ChangelogEntry as ChangelogEntryType } from "@/utils/changelog";

const components = {
    h2: (props: any) => (
        <h2 className="mb-4 mt-8 text-xl font-semibold" {...props} />
    ),
    p: (props: any) => <p className="my-4" {...props} />,
    img: (props: any) => (
        <img
            className="pointer-events-none my-6 w-full select-none rounded-lg"
            {...props}
            alt={props.alt || "Changelog image"}
        />
    ),
    a: (props: any) => (
        <a className="text-blue-500 hover:underline" {...props} />
    ),
    ul: (props: any) => (
        <ul className="my-4 list-disc space-y-2 pl-5" {...props} />
    ),
    li: (props: any) => <li className="pl-2" {...props} />,
    code: (props: any) => (
        <code
            className="rounded bg-gray-800 px-1 py-0.5 font-mono text-sm"
            {...props}
        />
    ),
};

interface ChangelogEntryProps {
    entry: ChangelogEntryType;
    mdxSource: any;
    isDetailed?: boolean;
}

export const ChangelogEntry = ({
    entry,
    mdxSource,
    isDetailed = false,
}: ChangelogEntryProps) => {
    const formattedDate = format(new Date(entry.date), "MMMM d, yyyy");

    return (
        <article className="changelog-entry mb-16">
            <div className="changelog-meta mb-4 flex items-center gap-3">
                <Link
                    href={`/changelog/${entry.version}`}
                    className="pill-version"
                >
                    v{entry.version}
                </Link>
                <span className="text-gray-400">{formattedDate}</span>
            </div>
            <div className="changelog-content text-gray-200">
                <MDXRemote {...mdxSource} components={components} />
            </div>
        </article>
    );
};
so the other pages just find all the posts or find a specific posts and passes it down to that
Mallow bee
yeah that's all ok
it's only the changelog and only on initial page load
client side navigation is fine
Scale parasitoidOP
yeah i genuinely have no idea whats going on
imma install the eslint thing to see if I can figure out whats wrong but from what I can see and AI and such I can't find any glaring issues
however there needs to be an issue since other nextjs sites can do it without the intial 500
btw whats ur eslint config, I never use it
import globals from "globals";
import pluginJs from "@eslint/js";
import tseslint from "typescript-eslint";
import pluginReact from "eslint-plugin-react";

/** @type {import('eslint').Linter.Config[]} */
export default [
    { files: ["**/*.{js,mjs,cjs,ts,jsx,tsx}"] },
    { languageOptions: { globals: globals.browser } },
    pluginJs.configs.recommended,
    ...tseslint.configs.recommended,
    pluginReact.configs.flat.recommended,
];
rebuilt w eslint and it still worked :/ although it did have this warning ⨯ ESLint: Config (unnamed): Key "extends": This appears to be in eslintrc format rather than flat config format.
Mallow bee
currently just this with eslint-plugin-react-hooks and eslint 8
{
  "extends": ["next/typescript", "plugin:react-hooks/recommended"]
}
since you'll have to tell your IDE to use the flat config too
it's a hassle
many people made the judgement call that eslint 8 -> 9 wasn't any easier than to switch to biome
@Mallow bee since you'll have to tell your IDE to use the flat config too
Scale parasitoidOP
do you have any more ideas for what the issue couold be
@Scale parasitoid do you have any more ideas for what the issue couold be
Mallow bee
you can reproduce it locally?
I would delete half the code until it stops breaking
then undo, narrow your targets to the code you just undid, and delete half of that
bug binary search has never failed me
@Mallow bee you can reproduce it locally?
Scale parasitoidOP
well yeah I just get a error saying that I might have an issue with multiple versions of react but i dont
Warning: Invalid hook call. Hooks can only be called inside of the body of a function component. This could happen for one of the following reasons:
1. You might have mismatching versions of React and the renderer (such as React DOM)
2. You might be breaking the Rules of Hooks
3. You might have more than one copy of React in the same app
See https://reactjs.org/link/invalid-hook-call for tips about how to debug and fix this problem.
dis one
Mallow bee
yeah that's probably the one causing the 500 error server-side
delete code until you stop getting that on the changelog page
Scale parasitoidOP
Hey I found a fix for it! Thanks a lot for the support really helped push me in the right direction
That’s not the answer to the question
@Scale parasitoid Hey I found a fix for it! Thanks a lot for the support really helped push me in the right direction
Mallow bee
anything you can share for the next person who finds the thread?
Yes pls share what you found
Mallow bee
otherwise this happens
@joulev That’s not the answer to the question
Scale parasitoidOP
OH LMAO Sorry i thought I had written a message with it
Scale parasitoidOP
Basically I just had to forwardRef the components that I was passing to MDX, for whatever reason that fixed the useState being null in it. Once that happened I rebuilt it and now when navigating to the page everything works as expected!
Answer
Thanks!
@Mallow bee Ah interesting, forwardRef is no longer necessary in React 19, I wonder if that would have fixed it itself
Scale parasitoidOP
yeah it might have, i genuinely have no idea why that fixes it but im not complaining :D
Mallow bee
it's possible your MDX library is expecting react 19 now?
I haven't updated any of my mdx in a long time but I never had to wrap them in forwardRef
anyway if you do upgrade to react 19 eventually (which I recommend, it's only upside) then see if it still works after removing the forwardRef