From 6106ab46e25e8cf40548b0d005815770d3733f72 Mon Sep 17 00:00:00 2001 From: Terence Tuhinanshu Date: Fri, 16 Jun 2023 18:39:46 +0000 Subject: [PATCH 1/3] Update to Google Analytics 4 Replace Google Analytics with Gtag.js, update production account to latest. Closes #3581 --- src/mmw/apps/core/templates/head.html | 14 ++++++-------- src/mmw/mmw/settings/production.py | 2 +- 2 files changed, 7 insertions(+), 9 deletions(-) diff --git a/src/mmw/apps/core/templates/head.html b/src/mmw/apps/core/templates/head.html index 4f0f309ac..52b8debfe 100644 --- a/src/mmw/apps/core/templates/head.html +++ b/src/mmw/apps/core/templates/head.html @@ -12,15 +12,13 @@ - + + diff --git a/src/mmw/mmw/settings/production.py b/src/mmw/mmw/settings/production.py index 79c2e4e29..38003a6d3 100644 --- a/src/mmw/mmw/settings/production.py +++ b/src/mmw/mmw/settings/production.py @@ -98,7 +98,7 @@ GOOGLE_MAPS_API_KEY = 'AIzaSyCXdkywU7rps_i1CeKqWxlBi97vyGeXsqk' # Stroud account -GOOGLE_ANALYTICS_ACCOUNT = 'UA-47047573-15' +GOOGLE_ANALYTICS_ACCOUNT = 'G-R98FWXKCFY' # django-cookies-samesite SESSION_COOKIE_SAMESITE = 'None' # Allows for cross site embedding into LARA From 2d9888a9f672b120e3bcdf122264d9ea2383f5b3 Mon Sep 17 00:00:00 2001 From: Terence Tuhinanshu Date: Fri, 16 Jun 2023 18:57:00 +0000 Subject: [PATCH 2/3] Skip reporting virtual routes to Google Analytics These are now reported automatically via changes to window.location and window.history. See: https://developers.google.com/analytics/devguides/collection/ga4/views?client_type=gtag --- src/mmw/js/src/routes.js | 8 -------- 1 file changed, 8 deletions(-) diff --git a/src/mmw/js/src/routes.js b/src/mmw/js/src/routes.js index b085cd70b..6be086251 100644 --- a/src/mmw/js/src/routes.js +++ b/src/mmw/js/src/routes.js @@ -21,11 +21,3 @@ router.addRoute('projects(/)', ProjectsController, 'projects'); router.addRoute('error(/:type)(/)', ErrorController, 'error'); router.addRoute('sign-up(/)', SignUpController, 'signUp'); router.addRoute('sign-up/sso(/:provider)(/:username)(/:first_name)(/:last_name)(/)', SignUpController, 'ssoSignUp'); - -router.on('route', function() { - // Allow Google Analytics to track virtual pageloads following approach in - // https://developers.google.com/analytics/devguides/collection/ - // analyticsjs/single-page-applications - window.ga('set', 'page', window.location.pathname); - window.ga('send', 'pageview'); -}); From e9e1a32ba3c183cab5d2d34cb6c905434c281a6a Mon Sep 17 00:00:00 2001 From: Terence Tuhinanshu Date: Fri, 16 Jun 2023 19:27:21 +0000 Subject: [PATCH 3/3] Replace ga.send with dataLayer.push Instead of sending custom events to Google Analytics, Tag Manager expects us to add them to a dataLayer managed in the browser, from where it can pull them on demand. These are tagged with `mmw`, which must also be configured in Google Tag Manager to work properly. --- src/mmw/js/src/analyze/models.js | 3 ++- src/mmw/js/src/core/testUtils.js | 2 +- src/mmw/js/src/core/utils.js | 17 +++++++++++++++++ src/mmw/js/src/draw/views.js | 18 +++++++++--------- src/mmw/js/src/modeling/controllers.js | 2 +- src/mmw/js/src/modeling/models.js | 6 +++--- 6 files changed, 33 insertions(+), 15 deletions(-) diff --git a/src/mmw/js/src/analyze/models.js b/src/mmw/js/src/analyze/models.js index 75be23e1b..19e06e28d 100644 --- a/src/mmw/js/src/analyze/models.js +++ b/src/mmw/js/src/analyze/models.js @@ -98,7 +98,8 @@ var AnalyzeTaskModel = coreModels.TaskModel.extend({ var gaEvent = self.get('name') + '-analyze', gaLabel = utils.isInDrb(aoi) ? 'drb-aoi' : 'national-aoi', gaAoiSize = turfArea(aoi) / 1000000; - window.ga('send', 'event', 'Analyze', gaEvent, gaLabel, parseInt(gaAoiSize)); + + utils.gtm('Analyze', gaEvent, gaLabel, parseInt(gaAoiSize)); var isWkaoi = utils.isWKAoIValid(wkaoi), taskHelper = { diff --git a/src/mmw/js/src/core/testUtils.js b/src/mmw/js/src/core/testUtils.js index a11e261ed..2a372cb72 100644 --- a/src/mmw/js/src/core/testUtils.js +++ b/src/mmw/js/src/core/testUtils.js @@ -3,7 +3,7 @@ var $ = require('jquery'); // Mock the Google Analytics function for tests -window.ga = function() {}; +window.dataLayer = { push: function() {} }; // Should be called after each unit test. function resetApp(app) { diff --git a/src/mmw/js/src/core/utils.js b/src/mmw/js/src/core/utils.js index 3ab57907c..d13d26ec9 100644 --- a/src/mmw/js/src/core/utils.js +++ b/src/mmw/js/src/core/utils.js @@ -800,6 +800,23 @@ var utils = { return result; }, + + // Convenience method for Google Tag Manager that mimics the + // old Google Analytics API + gtm: function(eventCategory, eventAction, eventLabel, eventValue) { + var event = { + 'event': 'mmw', + 'eventCategory': eventCategory, + 'eventAction': eventAction, + 'eventLabel': eventLabel, + }; + + if (eventValue) { + event.eventValue = eventValue; + } + + window.dataLayer.push(event); + }, }; module.exports = utils; diff --git a/src/mmw/js/src/draw/views.js b/src/mmw/js/src/draw/views.js index e5cd69019..defc397b4 100644 --- a/src/mmw/js/src/draw/views.js +++ b/src/mmw/js/src/draw/views.js @@ -470,7 +470,7 @@ var AoIUploadView = Marionette.ItemView.extend({ } this.addPolygonToMap(polygon); - window.ga('send', 'event', GA_AOI_CATEGORY, 'aoi-create', 'geojson'); + coreUtils.gtm(GA_AOI_CATEGORY, 'aoi-create', 'geojson'); }, handleShpZip: function(zipfile) { @@ -485,7 +485,7 @@ var AoIUploadView = Marionette.ItemView.extend({ }) .catch(_.bind(self.handleShapefileError, self)); - window.ga('send', 'event', GA_AOI_CATEGORY, 'aoi-create', 'shapefile'); + coreUtils.gtm(GA_AOI_CATEGORY, 'aoi-create', 'shapefile'); }, reprojectAndAddFeature: function(shp, prj) { @@ -718,8 +718,8 @@ var SelectBoundaryView = DrawToolBaseView.extend({ codeToLayer[layerCode] = ol; grid.on('click', function(e) { - window.ga('send', 'event', GA_AOI_CATEGORY, 'aoi-create', 'boundary-' + layerCode); - window.ga('send', 'event', GA_AOI_CATEGORY, 'boundary-aoi-create', e.data.name); + coreUtils.gtm(GA_AOI_CATEGORY, 'aoi-create', 'boundary-' + layerCode); + coreUtils.gtm(GA_AOI_CATEGORY, 'boundary-aoi-create', e.data.name); getShapeAndAnalyze(e, self.model, ofg, grid, layerCode, shortDisplay); }); @@ -831,7 +831,7 @@ var DrawAreaView = DrawToolBaseView.extend({ .then(function(shape) { addLayer(shape); navigateToAnalyze(); - window.ga('send', 'event', GA_AOI_CATEGORY, 'aoi-create', 'freedraw'); + coreUtils.gtm(GA_AOI_CATEGORY, 'aoi-create', 'freedraw'); }).fail(function(message) { revertLayer(); displayAlert(message, modalModels.AlertTypes.error); @@ -848,7 +848,7 @@ var DrawAreaView = DrawToolBaseView.extend({ var point = L.marker(latlng).toGeoJSON(), box = utils.getSquareKmBoxForPoint(point); - window.ga('send', 'event', GA_AOI_CATEGORY, 'aoi-create', 'squarekm'); + coreUtils.gtm(GA_AOI_CATEGORY, 'aoi-create', 'squarekm'); return box; }).then(validateShape).then(function(polygon) { addLayer(polygon, '1 Square Km'); @@ -937,7 +937,7 @@ var WatershedDelineationView = DrawToolBaseView.extend({ utils.placeMarker(map) .then(function(latlng) { - window.ga('send', 'event', GA_AOI_CATEGORY, 'aoi-create', 'rwd-' + dataSource); + coreUtils.gtm(GA_AOI_CATEGORY, 'aoi-create', 'rwd-' + dataSource); return validatePointWithinDataSourceBounds(latlng, dataSource); }) .then(function(latlng) { @@ -1140,7 +1140,7 @@ var DrainageAreaView = DrawToolBaseView.extend({ App.map.set('areaOfInterestAdditionals', additionalShapes); addLayer(response.area_of_interest, 'Point-based Drainage Area'); navigateToAnalyze(); - window.ga('send', 'event', GA_AOI_CATEGORY, 'aoi-create', 'drainage-point'); + coreUtils.gtm(GA_AOI_CATEGORY, 'aoi-create', 'drainage-point'); }).fail(function(message) { revertLayer(); displayAlert(message, modalModels.AlertTypes.error); @@ -1179,7 +1179,7 @@ var DrainageAreaView = DrawToolBaseView.extend({ App.map.set('areaOfInterestAdditionals', additionalShapes); addLayer(response.area_of_interest, 'Stream-based Drainage Area'); navigateToAnalyze(); - window.ga('send', 'event', GA_AOI_CATEGORY, 'aoi-create', 'drainage-stream'); + coreUtils.gtm(GA_AOI_CATEGORY, 'aoi-create', 'drainage-stream'); }).fail(function(message) { revertLayer(); displayAlert(message, modalModels.AlertTypes.error); diff --git a/src/mmw/js/src/modeling/controllers.js b/src/mmw/js/src/modeling/controllers.js index 80fef3f60..b41ad5c29 100644 --- a/src/mmw/js/src/modeling/controllers.js +++ b/src/mmw/js/src/modeling/controllers.js @@ -113,7 +113,7 @@ var ModelingController = { }, makeNewProject: function(modelPackage) { - window.ga('send', 'event', constants.GA.MODEL_CATEGORY, constants.GA.MODEL_CREATE_EVENT, modelPackage); + utils.gtm(constants.GA.MODEL_CATEGORY, constants.GA.MODEL_CREATE_EVENT, modelPackage); var project; if (settings.get('itsi_embed')) { diff --git a/src/mmw/js/src/modeling/models.js b/src/mmw/js/src/modeling/models.js index 47d1f2f6e..64c9f351d 100644 --- a/src/mmw/js/src/modeling/models.js +++ b/src/mmw/js/src/modeling/models.js @@ -1297,8 +1297,8 @@ var ScenarioModel = Backbone.Model.extend({ modelPackage = App.currentProject.get('model_package'), modKeyName = modelPackage === utils.GWLFE ? 'modKey' : 'value'; - window.ga('send', 'event', constants.GA.MODEL_CATEGORY, - modelPackage + constants.GA.MODEL_MOD_EVENT, modification.get(modKeyName)); + utils.gtm(constants.GA.MODEL_CATEGORY, + modelPackage + constants.GA.MODEL_MOD_EVENT, modification.get(modKeyName)); // For GWLFE, first remove existing mod with the same key since it // doesn't make sense to have multiples of the same type of BMP. @@ -1750,7 +1750,7 @@ var ScenariosCollection = Backbone.Collection.extend({ inputs: currentConditions.get('inputs').toJSON(), }); - window.ga('send', 'event', constants.GA.MODEL_CATEGORY, constants.GA.MODEL_SCENARIO_EVENT, modelPackage); + utils.gtm(constants.GA.MODEL_CATEGORY, constants.GA.MODEL_SCENARIO_EVENT, modelPackage); this.add(scenario); this.setActiveScenarioByCid(scenario.cid);