Track Single Page Applications with GTM and GA4

Photo de profil de Lucas Rollin Lucas Rollin

Updated: Monday, March 24, 2025

What is a Single Page Application (SPA)?

A Single Page Application is a kind of website that loads all the code required for navigation from the very first page. As the name suggests, it’s an application that consists of a single page.

SPAs are created from the Javascript frameworks such as Angular, React, Vue.js or Next.JS (non-exhaustive list).

Google Analytics 4 offers a default feature to track SPAs, but this may not work for your SPA. That’s why I’m going to show you 3 different methods in this blog post.

What is the History API in Javascript?

Without going into the technical details, the idea here is to understand how SPAs work.

The history object is a global Javascript variable used by most SPAs to manipulate browsing history.

The global Javascript history variable
The global Javascript history variable

If you’re working on an old SPA, it may not use the Javascript History API, in which case you’ll need to install Google Analytics 4 via method 3.

3 ways to track Single Page Applications with GTM and GA4

Method 1: Leave GA4 with default settings

Let’s see what happens if I install a new GA4 property on all pages.

I now receive History events (gtm.historyChange-v2) and History Change events (page_view).

The gtm.historyChange-v2 event is triggered as many times as the gtm.historyChange event, because they listen to the same events (those of the History API).

However, gtm.historyChange-v2 is managed by GA4, whereas gtm.historyChange is managed by GTM.

This is why gtm.historyChange-v2 then triggers a page_view event (named History Change in GTM).

This page_view event then tracks all pages viewed using the history API.

This feature is enabled by default in GA4’s enhanced measurement, and can be disabled if you feel it’s necessary. If you wish to pass more information in the page_view event such as a dynamic page title, you will need to use method 3.

The default configuration of a GA4 property on page views linked to browsing history
The default configuration of a GA4 property on page views linked to browsing history

Method 2: Track page views with the History Change trigger in GTM

Disable GA4 automatic detection

To use this method, I disable page tracking when changing the browsing history in GA4’s enhanced metrics settings.

Disabling page views linked to browsing history
Disabling page views linked to browsing history

I now only have the gtm.historyChange event, the gtm.historyChange-v2 event is no longer available in the dataLayer.

Disable page_view event in the Google tag

Disabling page views in Google tag
Disabling page views in Google tag

The page_view event

On first page load


Page_view event on first page load
page_view event on first page load

When history changes

Overload page_location and page_referrer parameters in page_view event
Overload page_location and page_referrer parameters in page_view event.

Other events

Overload page_location and page_referrer parameters in page_view event
Overload page_location and page_referrer parameters in add_to_cart.

Method 3: Track page views via the dataLayer


Method 3 involves asking a developer to send a page_view event in the dataLayer on the first page load AND when the user changes view in the SPA.

dataLayer.push(
    {
        event: "page_view",
        virtual_page_title: document.title,
        virtual_page_location: document.location.protocol + '//' +
                    document.location.hostname +
                    document.location.pathname +
                    document.location.search,
        virtual_page_referrer: document.referrer
    }
);

Once the dataLayer has been set up by the developer, you need to update all my GA4 events to override the page_title, page_location and page_referrer parameters using the variables sent in the dataLayer.


Disable GA4 automatic detection

To use this method, I disable page tracking when changing the browsing history in the GA4 enhanced metrics settings.

Disabling page views linked to browsing history
Disabling page views linked to browsing history

Disable page_view event in the Google tag

Disabling page views in Google tag
Disabling page views in Google tag

The page_view event

page_view event after implementation of dataLayer by developer
Page_view event after implementation of dataLayer by developer

Other events

add_to_cart event after implementation of the dataLayer by the developer
Add_to_cart event after implementation of the dataLayer by the developer.