This document describes the use of Google Tag Manager (GTM) with Google Analytics 4 (GA4) on GOV.UK Publishing. Further information can be found in our Implementation record.
No analytics code is initialised and no data is gathered without user consent. For more information, see our consent documentation.
If you are a developer working on a part of GOV.UK that includes analytics, read our developer guide.
We pass three types of data to GA4.
- page views
- events
- search data
Page views happen when a page loads.
Events happen when a user interacts with certain things, for example clicking on an accordion, tab, or link.
Search data is gathered when users perform a search.
For more information about different kinds of tracking, read our overview of trackers.
We have two ways of running analytics code:
- as modules attached to elements on the page e.g. the GA4 auto tracker
<div data-module="ga4-auto-tracker">
- as code that runs on page load regardless of page content e.g. the GA4 copy tracker
For modules, our existing guidance on writing GOV.UK JavaScript modules should be followed.
For analytics code that runs on page load, the following approach should be used. Code should be attached to the window.GOVUK.analyticsGa4.analyticsModules
object and include an init
function, using the structure shown below.
When cookie consent is given, init-ga4.js
looks through the analyticsModules
object for anything with an init
function, and executes them if found. This means that analytics code will not be executed unless consent is given, and gives a standard way to add more analytics code without additional initialisation.
window.GOVUK = window.GOVUK || {}
window.GOVUK.analyticsGa4 = window.GOVUK.analyticsGa4 || {}
window.GOVUK.analyticsGa4.analyticsModules = window.GOVUK.analyticsGa4.analyticsModules || {};
(function (analyticsModules) {
'use strict'
var ExampleCode = {
init: function () {
// do analytics stuff, like send a page view
}
}
analyticsModules.ExampleCode = ExampleCode
})(window.GOVUK.analyticsGa4.analyticsModules)
General analytics code and reused functions are kept in ga4-core.js.
This includes the code that initialises Google Tag Manager and pushes data to the dataLayer, as well as shared functions for link and ecommerce tracking.
While our aim is to keep the majority of the analytics code in govuk_publishing_components
it makes sense to put analytics code specific to a certain page in the application that renders that page. GA4 analytics documentation can also be found in the following applications.
Often tracking will need to be built into components, either using an existing tracker or some custom code, or a combination of both.
Where component tracking needs additional data to be passed (for example index
or section
values to show where the component is on a page) this should be built into the component and the data required documented in the component guide.
Where component tracking can be entirely self contained (i.e. nothing needs to be passed) tracking should be enabled by default, but a disable_ga4
option should be included and documented to allow the tracking to be disabled if required. See the tabs component for an example.
All of the data sent to GTM is based on a common schema.
{
event: '',
page: {},
event_data: {},
search_results: {}
}
event
must have a specific value to activate the right trigger in GTM.
page
is defined in the ga4-page-views script.
event_data
is defined in the ga4-schemas script and used in the ga4-event-tracker script.
search_results
is defined in the ga4-ecommerce-tracker script.
For more details of how our data schemas work, read our data schemas documentation or for the structure read our Implementation record.
GTM's dataLayer has two elements - an array and an object. window.dataLayer = []
is executed when the page loads.
GOV.UK JavaScript (JS) pushes objects to the dataLayer array when certain things happen e.g. when the page loads, or a user clicks on a certain type of link. Once that happens GTM takes over. It reads the latest object in the array and passes the data found into the dataLayer object. Importantly, it only adds to the object, so data can persist from previous array pushes.
For example:
- an event happens and JS pushes
{ a: 1 }
to the dataLayer array - GTM adds this to the dataLayer object, which is now
{ a: 1 }
- another event happens and JS pushes
{ b: 1 }
to the array - GTM adds this to the dataLayer object, which is now
{ a: 1, b: 1 }
If data shouldn't persist it can be erased in a following push, for example by sending { a: 1, b: false }
, but often this overall behaviour is desirable - for example, page view data will persist in events that happen on that page, providing more context for analysts.
If the data given to GTM contains a recognised event
attribute value, GTM then sends that data on to GA4.
The dataLayer is recreated when a user navigates to another page, so no data in the dataLayer will persist between pages.