Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Bug]: Example in README > Examples > Avoid Hydration Mismatch > Images does not work on initial load #311

Open
davidde opened this issue Sep 4, 2024 · 1 comment · May be fixed by #335
Labels
bug Something isn't working triage

Comments

@davidde
Copy link

davidde commented Sep 4, 2024

What happened?

This code from the README causes Warning: Prop 'loading' did not match. Server: "null" Client: "lazy":

import Image from 'next/image'
import { useTheme } from 'next-themes'

function ThemedImage() {
  const { resolvedTheme } = useTheme()
  let src

  switch (resolvedTheme) {
    case 'light':
      src = '/light.png'
      break
    case 'dark':
      src = '/dark.png'
      break
    default:
      src = ''
      break
  }

  return <Image src={src} width={400} height={400} />
}

export default ThemedImage

It does not render the image on initial load or page refresh.

I noticed it works with the useEffect/setMounted hack, but this causes other problems like layout shift. Is there a fix for this?

Version

^0.3.0

What browsers are you seeing the problem on?

Firefox, Chrome

@davidde davidde added bug Something isn't working triage labels Sep 4, 2024
@davidde davidde changed the title [Bug]: Example in README > Examples > Avoid Hydration Mismatch > Images does not work on initial load [Bug]: Example in README > Examples > Avoid Hydration Mismatch > Images does not work on initial load Sep 4, 2024
@90PabloRomero
Copy link

try to set same size container to the image element to avoid shifting
try this approach instead

'use client'
import { useTheme } from "next-themes";
import type { ImageProps } from "next/image";
import Image from "next/image";
import { useEffect, useState } from "react";

type Props = Omit<ImageProps, "src" | "priority" | "loading"> & {
  srcLight: string;
  srcDark: string;
};

const ThemedImage = (props: Props) => {
  const { resolvedTheme } = useTheme();
  const [mounted, setMounted] = useState(false);
  const { srcLight, srcDark, ...rest } = props;

  let srcImage;

  useEffect(() => {
    setMounted(true);
  }, []);

  if (!mounted) {
    return null;
  }

  switch (resolvedTheme) {
    case "light":
      srcImage = srcLight;
      break;
    case "dark":
      srcImage = srcDark;
      break;
    default:
      srcImage = srcLight;
      break;
  }

  return (
    <Image
      src={srcImage}
      {...rest}
    />
  );
};

export default ThemedImage;

kanavbajaj added a commit to kanavbajaj/next-themes that referenced this issue Jan 4, 2025
Fixes issue pacocoursey#311

Used a conditional rendering strategy that waits until the client has mounted before rendering the Image component.
kanavbajaj added a commit to kanavbajaj/next-themes that referenced this issue Jan 4, 2025
Fixes pacocoursey#311

Used a conditional rendering strategy that waits until the client has mounted before rendering the Image component.
@kanavbajaj kanavbajaj linked a pull request Jan 4, 2025 that will close this issue
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working triage
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants