DEV Community

Cover image for ReferenceError: window is not defined - A Developer's Guide
Harshal Ranjhani for CodeParrot

Posted on • Originally published at codeparrot.ai

ReferenceError: window is not defined - A Developer's Guide

Frustrated developer meme

Ever seen this error pop up in your console and wondered what's going on? You're not alone! The infamous "window is not defined" error is one of the most common headaches for developers working with React, Next.js, or any server-side rendered (SSR) applications.

What's the Deal with this Error? ๐Ÿค”

First, let's understand what window actually is. In browser-based JavaScript, window is a global object that represents the browser window. It contains all sorts of useful stuff like:

  • window.localStorage for storing data
  • window.location for URL information
  • window.document for DOM manipulation
  • And many more browser-specific APIs

The problem? This object only exists in the browser. When your code runs on the server (like during SSR), there is no browser, and therefore no window object!

Server vs Browser Environment

Common Scenarios Where This Error Occurs ๐Ÿšจ

  1. Direct Window Access

When you try to access window properties directly in your component, especially during initial render, you'll encounter this error. This commonly happens when checking screen dimensions or browser features:

// This will break during SSR
const screenWidth = window.innerWidth;
Enter fullscreen mode Exit fullscreen mode
  1. Third-party Libraries

Many browser-specific libraries assume they're running in a client environment. When these libraries try to access window during server-side rendering, your application will crash:

// Some libraries assume window exists
import someLibrary from 'browser-only-library';
Enter fullscreen mode Exit fullscreen mode
  1. localStorage Usage

localStorage is a window property that's frequently accessed for client-side storage. Trying to use it during server rendering will trigger the error:

// This will fail on the server
const savedData = localStorage.getItem('user-data');
Enter fullscreen mode Exit fullscreen mode

How to Fix It? ๐Ÿ’ช

1. Use useEffect Hook

The most straightforward solution is to wrap your browser-specific code in a useEffect hook:

import { useEffect } from 'react';

function MyComponent() {
    useEffect(() => {
        // Safe to use window here
        const screenWidth = window.innerWidth;
        console.log('Screen width:', screenWidth);
    }, []);
    return <div>My Component</div>;
}
Enter fullscreen mode Exit fullscreen mode

2. Check if Window is Defined

Create a utility function to safely check for window:

const isClient = typeof window !== 'undefined';

function MyComponent() {
    if (isClient) {
    // Safe to use window here
    }
    return <div>My Component</div>;
}
Enter fullscreen mode Exit fullscreen mode

3. Dynamic Imports (Next.js Solution)

For Next.js applications, use dynamic imports with ssr: false:

import dynamic from 'next/dynamic';

const BrowserOnlyComponent = dynamic(
    () => import('../components/BrowserComponent'),
    { ssr: false }
);
Enter fullscreen mode Exit fullscreen mode

Pro Tips ๐ŸŒŸ

Avoid the "window is not defined" error with these battle-tested patterns:

  1. Create a Custom Hook
function useWindowSize() {
    const [size, setSize] = useState({ width: 0, height: 0 });
    useEffect(() => {
        const updateSize = () => {
            setSize({
                width: window.innerWidth,
                height: window.innerHeight
             });
        };

        window.addEventListener('resize', updateSize);
        updateSize();
        return () => window.removeEventListener('resize', updateSize);
    }, []);
    return size;
}
Enter fullscreen mode Exit fullscreen mode
  1. Handle SSR Gracefully
function MyComponent() {
    const [isMounted, setIsMounted] = useState(false);
    useEffect(() => {
        setIsMounted(true);
    }, []);

    if (!isMounted) {
        return null; // or a loading state
    }
    return <div>Browser-only content</div>;
}
Enter fullscreen mode Exit fullscreen mode

Common Gotchas to Watch Out For โš ๏ธ

  1. Forgetting about SSR: Always remember that your React code might run on the server first, leading to the "window is not defined" error.

  2. Third-party Dependencies: Some packages assume they're running in a browser. Check their SSR compatibility before using them.

  3. Conditional Imports: Don't use dynamic imports unless necessary - they can impact performance.

Testing Your Code ๐Ÿงช

Remember to test your application in both server and client environments. Here's a simple test setup:

describe('MyComponent', () => {
    it('handles server-side rendering', () => {
    // Test SSR scenario
    const { container } = render(<MyComponent />);
    expect(container).toBeInTheDocument();
});

it('handles client-side rendering', () => {
    // Mock window object
    const { container } = render(<MyComponent />);
    expect(container).toBeInTheDocument();
    });
});
Enter fullscreen mode Exit fullscreen mode

Conclusion ๐ŸŽ‰

The "window is not defined" error might seem scary at first, but it's actually pretty straightforward to handle once you understand why it happens. Remember:

  • Use useEffect for browser-specific code
  • Check if window exists before using it
  • Consider using Next.js dynamic imports for browser-only components
  • Always test both server and client scenarios

If you wish to learn more about the window object, you can read the MDN Web Docs.

Success gif

Happy coding! ๐Ÿš€

Top comments (0)