Leon Gaban
Leon Gaban

Reputation: 39018

Window is not defined in Next.js React app

In my Next.js app I can't seem to access window:

Unhandled Rejection (ReferenceError): window is not defined

componentWillMount() {
    console.log('window.innerHeight', window.innerHeight);
}

Enter image description here

Upvotes: 320

Views: 625828

Answers (30)

Eric Burel
Eric Burel

Reputation: 4914

None of the answers here is fully complete and most people will end up here because they hit the same issue but in various situations that require different solutions, so let me share the summary extracted from my paid course on the topic. I've also written a more detailed and free article.

Original question mentions an older version of React, using callbacks, but my answer is based on hooks and more relevant to current versions of React (16+) and Next (12+).

  • My component needs a browser-only library like Leaflet or jQuery, it crashes on server import

These libraries crash when being imported server-side, because they expect th window object to be immediately available.

This is a situation similar to the one described in this question.

You need what I call a Browser Component, a React component that cannot be prerendered on the server, contrary to Client components that are prerendered on the server in Next/React.

Solution: use lazy loading with next/dynamic and ssr: false to import your Browser Component. Now your component will only be imported and run in the browser.

  • My component needs to use the window object or any other browser-only code to render. Sadly, I cannot modify its code.

It's a Browser Component. Ideally, it should be rewritten to be a Client Component, but since you can't control the code you may not be able to do that.

You can use a NoSsr component. You can also use the useMounted hook (Next doc calls it "useClient") in the parent, but only if the parent is a Client Component. So NoSsr is a better choice in this case.

Relevant utility code:

export const useMounted = () => {
    const [mounted, setMounted] = useState<boolean>()
    // effects run only client-side
    // so we can detect when the component is hydrated/mounted
    // @see https://react.dev/reference/react/useEffect
    useEffect(() => {
        setMounted(true)
    }, [])
    return mounted
}

export function NoSsr({ children }: { children: React.ReactNode }) {
  const mounted = useMounted();
  if (!mounted) return null;
  return <>{children}</>;
}
  • My component needs to use the window object or any other browser-only code to render. I can modify its code.

The component should be a Client Component. Make it safe to be prerendered on the server

You can use the useMounted hook above to rewrite your component and render conditionally: you can use the window object to render only if the component is mounted.

This will turn your "Browser Component" into a "Client Component" that is safe for server-side rendering, even if renders nothing server-side or a loader.

  • My component needs a library that can be imported server-side but not used there like D3.

More broadly, this is the case where you want to do imperative modification of the DOM, and the closest to OP original question.

Make it a Client Component.

You can wrap offending code in an effect like so: useEffect(() => { console.log(window.innerHeight) }, []).

It will log/render nothing, or a loader, during server-side rendering, but that's totally fine.

  • My component code is exactly the same, but behaves slightly differently in browser and server.

This happens when using dates or localization features of JavaScript, you have the same code client-side and server-side but it generates a different result.

It should be a client component.

You can use suppressHydrationWarning or a Suspense as demonstrated in this article

Upvotes: 4

user4920718
user4920718

Reputation: 1456

Just put 'use client' on the top of ur codebase

Upvotes: -2

Snehal singh
Snehal singh

Reputation: 49

Current answer as per next 13

In my case I was importing a library that used window in it's source code without adhering to it's usage in nextjs environments.

The error in such scenarios occurs because next13 tries to pre-render your pages and while doing so when they parse their code then obviously they will raise the error belonging to class things on the browser are not defined on the server.

Easy way to solve this is to tell nextjs not to pre-render that page. Simply opt out. In order to do that from your page.tsx export export const dynamic = "force-dynamic"; which omits this page from pre-rendering.

Another way would be to raise an issue and patch the source code yourself to then create a PR to be merged.

Final code

import React from "react";

function page() {
    return (
        <div> {/*tsx/jsx*/} </div>
    );
}

export const dynamic = "force-dynamic";
export default page;

Upvotes: 1

user884321
user884321

Reputation: 477

Solution when using hooks. Credit to @luckgaer.

This hook will return null on initial(server) render when called in a client component, and therefore not throw the above stated error. All we have to do to utilize the hook in a client component is to check that it's not null and skip its usage on its first render. Example on how to use the hook in a client component is below the hook snippet.

"use client";
import { useLayoutEffect, useState } from "react";

const useWindowWidth = (): number | null => {
  if (!global?.window) return null;
  const [windowWidth, setWindowWidth] = useState(window.innerWidth);

  useLayoutEffect(() => {
    let timeoutId: NodeJS.Timeout;

    const handleResize = () => {
      clearTimeout(timeoutId);
      timeoutId = setTimeout(() => {
        setWindowWidth(window.innerWidth);
      }, 100);
    };

    window.addEventListener("resize", handleResize);
    return () => window.removeEventListener("resize", handleResize);
  }, []);

  return windowWidth;
};

export default useWindowWidth;

How to use in client components:

const windowWidth = useWindowWidth();
if (windowWidth && windowWidth >= 850)...

Upvotes: 0

Anjan Talatam
Anjan Talatam

Reputation: 3996

Next JS 13 with Pages router

Below is a code snippet on how you can create, update(use create fn for updating as well), and remove a key (here username) from localStorage


const [username, setUsername] = useState<null | string>(null);

useEffect(() => {
  if (typeof window !== "undefined" && window.localStorage) {
    let token = localStorage.getItem("username");
    setUsername(token);
  }
}, [setUsername]);

const createItem = (newUsername: string) => {
  if (typeof window !== "undefined" && window.localStorage) {
    localStorage.setItem("username", newUsername);

    let token = localStorage.getItem("username");

    setUsername(token);
  }
};

const removeItem = () => {
  if (typeof window !== "undefined" && window.localStorage) {
    localStorage.removeItem("username");
    setUsername(null);
  }
};

console.log(username, "username");

Reference

Upvotes: 5

o-faro
o-faro

Reputation: 1031

This was driving me nuts, using useEffect is only working if you define your function within it. But I don't want to define helper functions there, so I ended up doing it like this:

    'use client'
    const MyClientC = () => { 
    const desiredCalculation = helperFN.bind(this)
       ...
    }
    // helperFN.ts
    interface Window {
      innerHeight: number
      ...
    
    }
    const isBrowser = () => typeof window !== 'undefined';
    let Window: Window = {} as Window;
    if (typeof window !== 'undefined') {
      Window = window;
    }
    
    const helperFn = () => {     
        if (!isBrowser()) {
          return false;
        }
        return Window.innerHeight ...

Maybe this helps

Upvotes: 1

jfunk
jfunk

Reputation: 8122

Nowadays you can add the 'use client' directive at the top of your client side component.

https://nextjs.org/docs/getting-started/react-essentials#the-use-client-directive

Upvotes: 1

Terry Windwalker
Terry Windwalker

Reputation: 1888

For anyone who somehow cannot use hook (for example, function component):

Use setTimeout(() => yourFunctionWithWindow()); will allow it get the window instance. Guess it just need a little more time to load.

Upvotes: -2

tlt
tlt

Reputation: 387

I wrapped the general solution (if (typeof window === 'undefined') return;) in a custom hook, that I am very pleased with. It has a similiar interface to reacts useMemo hook which I really like.

import { useEffect, useMemo, useState } from "react";

const InitialState = Symbol("initial");

/**
 *
 * @param clientFactory Factory function similiar to `useMemo`. However, this function is only ever called on the client and will transform any returned promises into their resolved values.
 * @param deps Factory function dependencies, just like in `useMemo`.
 * @param serverFactory Factory function that may be called server side. Unlike the `clientFactory` function a resulting `Promise` will not be resolved, and will continue to be returned while the `clientFactory` is pending.
 */
export function useClientSideMemo<T = any, K = T>(
  clientFactory: () => T | Promise<T>,
  deps: Parameters<typeof useMemo>["1"],
  serverFactory?: () => K
) {
  const [memoized, setMemoized] = useState<T | typeof InitialState>(
    InitialState
  );

  useEffect(() => {
    (async () => {
      setMemoized(await clientFactory());
    })();
    // eslint-disable-next-line react-hooks/exhaustive-deps
  }, deps);

  return typeof window === "undefined" || memoized === InitialState
    ? serverFactory?.()
    : memoized;
}

Usage Example:

I am using it to dynamically import libaries that are not compatible with SSR in next.js, since its own dynamic import is only compatible with components.

  const renderer = useClientSideMemo(
    async () =>
      (await import("@/components/table/renderers/HighlightTextRenderer"))
        .HighlightTextRendererAlias,
    [],
    () => "text"
  );

As you can see I even implemented a fallback factory callback, so you may provide a result when initially rendering on the server aswell. In all other aspects this hook should behave similiar to reacts useMemo hook. Open to feedback.

Upvotes: 6

Inder Pal Singh
Inder Pal Singh

Reputation: 427

You can try the below code snippet for use-cases such as - to get current pathname (CurrentUrl Path)

 import { useRouter } from "next/router";

 const navigator = useRouter()
 console.log(navigator.pathname);

Upvotes: 1

Nick Weimer
Nick Weimer

Reputation: 627

I had this same issue when refreshing the page (caused by an import that didn't work well with SSR).

What fixed it for me was going to pages where this was occurring and forcing the import to be dynamic:

import dynamic from 'next/dynamic';


const SomeComponent = dynamic(()=>{return import('../Components/SomeComponent')}, {ssr: false});

//import SomeComponent from '../Components/SomeComponent'

Commenting out the original import and importing the component dynamically forces the client-side rendering of the component.

The dynamic import is covered in Nextjs's documentation here: https://nextjs.org/docs/advanced-features/dynamic-import

I got to this solution by watching the youtube video here: https://www.youtube.com/watch?v=DA0ie1RPP6g

Upvotes: 6

W.K.C
W.K.C

Reputation: 60

For Next.js version 12.1.0, I find that we can use process.title to determine whether we are in browser or in node side. Hope it helps!

export default function Projects(props) {
    console.log({ 'process?.title': process?.title });

    return (
        <div></div>
    );
}

1. From the terminal, I receive { 'process?.title': 'node' }

process.title === 'node'

2. From Chrome devtool, I revice { 'process?.title': 'browser' }

process.title === 'browser'

Upvotes: 1

lucgauer
lucgauer

Reputation: 325

global?.window && window.innerHeight

It's important to use the operator ?., otherwise the build command might crash.

Upvotes: 21

Parsa Safavi
Parsa Safavi

Reputation: 236

A bit late but you could also consider using Dynamic Imports from next turn off SSR for that component.

You can warp the import for your component inside a dynamic function and then, use the returned value as the actual component.

import dynamic from 'next/dynamic'

const BoardDynamic = dynamic(() => import('../components/Board.tsx'), {
  ssr: false,
})

<>
   <BoardDynamic />
</>

Upvotes: 13

Lakshit Nagar
Lakshit Nagar

Reputation: 337

If it is NextJS app and inside _document.js, use below:

<script dangerouslySetInnerHTML={{
        __html: `
            var innerHeight = window.innerHeight;
        `
        }} />

Upvotes: -4

Lianel
Lianel

Reputation: 159

I want to leave this approach that I found interesting for future researchers. It's using a custom hook useEventListener that can be used in so many others needs.

Note that you will need to apply a little change in the originally posted one, like I suggest here.

So it will finish like this:

import { useRef, useEffect } from 'react'

export const useEventListener = (eventName, handler, element) => {
  const savedHandler = useRef()

  useEffect(() => {
    savedHandler.current = handler
  }, [handler])

  useEffect(() => {
    element = !element ? window : element
    const isSupported = element && element.addEventListener
    if (!isSupported) return

    const eventListener = (event) => savedHandler.current(event)

    element.addEventListener(eventName, eventListener)

    return () => {
      element.removeEventListener(eventName, eventListener)
    }
  }, [eventName, element])
}

Upvotes: -2

Vinicius Pollo
Vinicius Pollo

Reputation: 109

Best solution ever

import dynamic from 'next/dynamic';

const Chart = dynamic(()=> import('react-apexcharts'), {
    ssr:false,
})

Upvotes: 12

Roj
Roj

Reputation: 1332

Here's an easy-to-use workaround that I did.

const runOnClient = (func: () => any) => {
  if (typeof window !== "undefined") {
    if (window.document.readyState == "loading") {
      window.addEventListener("load", func);
    } else {
      func();
    }
  }
};

Usage:

runOnClient(() => {
// access window as you like
})

// or async
runOnClient(async () => {
// remember to catch errors that might be raised in promises, and use the `await` keyword wherever needed
})

This is better than just typeof window !== "undefined", because if you just check that the window is not undefined, it won't work if your page was redirected to, it just works once while loading. But this workaround works even if the page was redirected to, not just once while loading.

Upvotes: 6

Darryl RN
Darryl RN

Reputation: 8228

̶A̶n̶o̶t̶h̶e̶r̶ ̶s̶o̶l̶u̶t̶i̶o̶n̶ ̶i̶s̶ ̶b̶y̶ ̶u̶s̶i̶n̶g̶ ̶p̶r̶o̶c̶e̶s̶s̶.̶b̶r̶o̶w̶s̶e̶r ̶ ̶t̶o̶ ̶j̶u̶s̶t̶ ̶e̶x̶e̶c̶u̶t̶e̶ ̶ ̶y̶o̶u̶r̶ ̶c̶o̶m̶m̶a̶n̶d̶ ̶d̶u̶r̶i̶n̶g̶ ̶r̶e̶n̶d̶e̶r̶i̶n̶g̶ ̶o̶n̶ ̶t̶h̶e̶ ̶c̶l̶i̶e̶n̶t̶ ̶s̶i̶d̶e̶ ̶o̶n̶l̶y̶.

But process object has been deprecated in Webpack5 and also NextJS, because it is a NodeJS variable for backend side only.

So we have to use back window object from the browser.

if (typeof window !== "undefined") {
  // Client-side-only code
}

Other solution is by using react hook to replace componentDidMount:

useEffect(() => {
    // Client-side-only code
})

Upvotes: 267

Seth Samuel
Seth Samuel

Reputation: 355

You can define a state var and use the window event handle to handle changes like so.

const [height, setHeight] = useState();

useEffect(() => {
    if (!height) setHeight(window.innerHeight - 140);
    window.addEventListener("resize", () => {
        setHeight(window.innerHeight - 140);
    });
}, []);

Upvotes: 0

Alexander Staroselsky
Alexander Staroselsky

Reputation: 38757

Move the code from componentWillMount() to componentDidMount():

componentDidMount() {
  console.log('window.innerHeight', window.innerHeight);
}

In Next.js, componentDidMount() is executed only on the client where window and other browser specific APIs will be available. From the Next.js wiki:

Next.js is universal, which means it executes code first server-side, then client-side. The window object is only present client-side, so if you absolutely need to have access to it in some React component, you should put that code in componentDidMount. This lifecycle method will only be executed on the client. You may also want to check if there isn't some alternative universal library which may suit your needs.

Along the same lines, componentWillMount() will be deprecated in v17 of React, so it effectively will be potentially unsafe to use in the very near future.

Upvotes: 103

Mohamed Jakkariya
Mohamed Jakkariya

Reputation: 1627

Date: 06/08/2021

Check if the window object exists or not and then follow the code along with it.

 function getSelectedAddress() {
    if (typeof window === 'undefined') return;

    // Some other logic
 }

Upvotes: 1

Ismoil Shokirov
Ismoil Shokirov

Reputation: 3011

The error occurs because window is not yet available, while component is still mounting. You can access window object after component is mounted.

You can create a very useful hook for getting dynamic window.innerHeight or window.innerWidth

const useDeviceSize = () => {

  const [width, setWidth] = useState(0)
  const [height, setHeight] = useState(0)

  const handleWindowResize = () => {
    setWidth(window.innerWidth);
    setHeight(window.innerHeight);
  }

  useEffect(() => {
    // component is mounted and window is available
    handleWindowResize();
    window.addEventListener('resize', handleWindowResize);
    // unsubscribe from the event on component unmount
    return () => window.removeEventListener('resize', handleWindowResize);
  }, []);

  return [width, height]

}

export default useDeviceSize 

Use case:

const [width, height] = useDeviceSize();

Upvotes: 32

San-Mak
San-Mak

Reputation: 11

For such cases, Next.js has Dynamic Import.

A module that includes a library that only works in the browser, it's suggested to use Dynamic Import. Refer

Upvotes: 1

U.A
U.A

Reputation: 3373

I have to access the hash from the URL so I come up with this

const hash = global.window && window.location.hash;

Upvotes: 10

Kate
Kate

Reputation: 1247

With No SSR

https://nextjs.org/docs/advanced-features/dynamic-import#with-no-ssr

import dynamic from 'next/dynamic'

const DynamicComponentWithNoSSR = dynamic(
  () => import('../components/hello3'),
  { ssr: false }
)

function Home() {
  return (
    <div>
      <Header />
      <DynamicComponentWithNoSSR />
      <p>HOME PAGE is here!</p>
    </div>
  )
}

export default Home

Upvotes: 92

crispengari
crispengari

Reputation: 9321

I was facing the same problem when i was developing a web application in next.js This fixed my problem, you have to refer to refer the window object in a life cycle method or a react Hook. For example lets say i want to create a store variable with redux and in this store i want to use a windows object i can do it as follows:

let store
useEffect(()=>{
    store = createStore(rootReducers,   window.__REDUX_DEVTOOLS_EXTENSION__ && 
    window.__REDUX_DEVTOOLS_EXTENSION__())
 }, [])
 ....

So basically, when you are working with window's object always use a hook to play around or componentDidMount() life cycle method

Upvotes: 3

RealScatman
RealScatman

Reputation: 376

In the constructor of your class Component you can add

if (typeof window === 'undefined') {
    global.window = {}
}

Example:

import React, { Component } from 'react'

class MyClassName extends Component {

    constructor(props){
        super(props)
        ...
        if (typeof window === 'undefined') {
            global.window = {}
        }
}

This will avoid the error (in my case, the error would occur after I would click reload of the page).

Upvotes: 10

Rotareti
Rotareti

Reputation: 53763

If you use React Hooks you can move the code into the Effect Hook:

import * as React from "react";

export const MyComp = () => {

  React.useEffect(() => {
    // window is accessible here.
    console.log("window.innerHeight", window.innerHeight);
  }, []);

  return (<div></div>)
}

The code inside useEffect is only executed on the client (in the browser), thus it has access to window.

Upvotes: 116

RAVINDER MAHAJAN
RAVINDER MAHAJAN

Reputation: 214

componentWillMount() lifecycle hook works both on server as well as client side. In your case server would not know about window or document during page serving, the suggestion is to move the code to either

Solution 1:

componentDidMount()

Or, Solution 2

In case it is something that you only want to perform in then you could write something like:

componentWillMount() {
    if (typeof window !== 'undefined') {
        console.log('window.innerHeight', window.innerHeight);
    }
}

Upvotes: 15

Related Questions