Lorem

How to use Gatsby's Script API with Google Analytics

In this post Iā€™ll be explaining how to add Googleā€™s ā€œnewā€ Google Analytics Property (GA4) to your Gatsby Site using the new Gatsby Script API.

Iā€™ll be demonstrating how to use the off-main-thread strategy which is powered by Builderā€™ioā€™s Partytown. I wrote about this once before on the Gatsby Blog, but that was before the release of Gatsby 4.15.0. From 4.15.0 onwards Gatsby takes care of all of the nitty-gritty for you so thereā€™s considerably less config required.

This post focuses on the Google Analytics 4 Property rather than the more recognizable Universal Analytics (UA) property. Hereā€™s a little note from Google about why you should use GA4 rather than UA.

Universal Analytics will no longer process new data in standard properties beginning July 1, 2023. Prepare now by setting up and switching over to a Google Analytics 4 property

Below are links to a minimal example repository on GitHub, and a Gatsby Cloud preview URL.

I wonā€™t go through the steps to create a GA4 Property in this post, but hereā€™s a link to Googleā€™s docs that should help you on your way.

Getting Started

Upgrade

The Script API was released as part of Gatsby 4.15.0, make sure youā€™re on at least this version or upgrade to the latest version of Gatsby. Hereā€™s a link to the Gatsby Changelog Prototype where you can see all the released versions and more details about each release.

npm install gatsby@latest

Remove Plugin

Itā€™s likely youā€™ll have been using gatsby-plugin-google-analytics, but after upgrading Gatsby you can uninstall it, and remove it from the plugins array in gatsby-config.

// gatsby-config.js

module.exports = {
  plugins: [
    ...
-   {
-     resolve: 'gatsby-plugin-google-analytics',
-     options: {
-       trackingId: 'UA-12345678-9'
-     }
-   }
  ],
};

Partytown Proxy

Since youā€™re in gatsby-config.js youā€™ll need to add the following.

Iā€™ve explained in more detail how Partytown proxies requests from Web Workers and how this usually ends up with unfathomable CORS errors in this post: How to Add Google Analytics gtag to Gatsby Using Partytown šŸŽ‰

// gatsby-config.js

module.exports = {
  plugins: [
    ...
  ],
+ partytownProxiedURLs: [`https://www.googletagmanager.com/gtag/js?id=${process.env.GATSBY_GA_MEASUREMENT_ID}`]
};

Adding Scripts

This next bit entirely depends on how youā€™ve setup your Gatsby Site.

Shared Component

Itā€™s quite common however to have a ā€œsharedā€ React component that is returned by wrapRootElement in both gatsby-browser.js and gatsby-ssr.js. Thereā€™s a little more in the Gatsby Docs here: Usage in Gatsby SSR and Browser APIs

In the example repo Iā€™ve called this Component RootElement, and it looks a bit like this.

// src/components/root-element.js

import React, { Fragment } from 'react';
import { Script } from 'gatsby';

const RootElement = ({ children }) => {
  return (
    <Fragment>
      <Script
        src={`https://www.googletagmanager.com/gtag/js?id=${process.env.GATSBY_GA_MEASUREMENT_ID}`}
        strategy='off-main-thread'
        forward={[`gtag`]}
      />
      <Script
        id='gtag-config'
        strategy='off-main-thread'
        dangerouslySetInnerHTML={{
          __html: `window.dataLayer = window.dataLayer || [];
          window.gtag = function gtag(){ window.dataLayer.push(arguments);}
          gtag('js', new Date()); 
          gtag('config', '${process.env.GATSBY_GA_MEASUREMENT_ID}', { send_page_view: false })`,
        }}
      />
      <div>{children}</div>
    </Fragment>
  );
};

export default RootElement;

The RootElement Component is returned by wrapRootElement in both gatsby-browser.js and gatsby-ssr.js which looks a bit like this.

// gatsby-browser.js

import React from 'react';
import RootElement from './src/components/root-element';

export const wrapRootElement = ({ element }) => {
  return <RootElement>{element}</RootElement>;
};
// gatsby-ssr.js

import React from 'react';
import RootElement from './src/components/root-element';

export const wrapRootElement = ({ element }) => {
  return <RootElement>{element}</RootElement>;
};

If you donā€™t ensure that gatsby-browser.js and gatsby-ssr return the same DOM elements youā€™ll likely see a React error like this.

Hydration failed because the initial UI does not match what was rendered on the server

You can read more about what that error means in the React Docs: Error Decoder

Page Views

Youā€™ll notice from the above that in the gtag config send_page_view is set to false. This is just for the initial setup but naturally youā€™ll want to fire off page_view eventsā€¦ because thatā€™s what Google Analytics is all about amirite?

Send Page Views

onRouteUpdate is one of Gatsby Browserā€™s APIā€™s and fires whenever a route change is detected. This API has a destructured parameter for the current location. This is perfect for sending to Googleā€™s page_view and will show up in your Analytics Dashboard.

Hereā€™s how Iā€™ve implemented it in gatsby-browser.js. You can see the complete src from the example repo here: gatsby-browser.js

// gatsby-browser.js

import React from 'react';
import RootElement from './src/components/root-element';


+ export const onRouteUpdate = ({ location }) => {
+  if (process.env.NODE_ENV !== 'production') {
+    return null;
+  }

+  const pagePath = location ? location.pathname + location.search + location.hash : undefined;

+  setTimeout(() => {
+    if (typeof window.gtag === 'function') {
+      window.gtag('event', 'page_view', { page_path: pagePath });
+    }
+  }, 100);

+  return true;
+ };


export const wrapRootElement = ({ element }) => {
  return <RootElement>{element}</RootElement>;
};

Google Admin Dashboard

This tripped me up but, make sure youā€™ve added your site URL to the Data Streams section of your Google Analytics Dashboard otherwise Google wonā€™t be ā€œlisteningā€ out of page_view events.

And Finally

Hereā€™s a little screenshot of my Google Analytics Realtime Overview, et voila, there I am on the map. It works, lovely stuff.

Further Reading