Source: src/js/models/analytics/GoogleAnalytics.js

  1. define(["models/analytics/Analytics"], function (Analytics) {
  2. /**
  3. * @class GoogleAnalytics
  4. * @classdesc A model that connects with an analytics service to record user
  5. * interactions with the app
  6. * @classcategory Models/Analytics
  7. * @extends Backbone.Model
  8. * @constructs
  9. * @extends Analytics
  10. * @since 2.25.0
  11. */
  12. var GoogleAnalytics = Analytics.extend(
  13. /** @lends GoogleAnalytics.prototype */
  14. {
  15. /**
  16. * The name of this Model
  17. * @name GoogleAnalytics#type
  18. * @type {string}
  19. * @readonly
  20. */
  21. type: "GoogleAnalytics",
  22. /**
  23. * @inheritdoc
  24. */
  25. setupAnalytics: function () {
  26. try {
  27. // If analytics is already set up, do nothing
  28. if (this.ready()) return;
  29. const tagId = this.getKey();
  30. // If there is no tag ID, do nothing
  31. if (!tagId) return;
  32. // Create a script element and set its src to the GA4 library
  33. const script = document.createElement("script");
  34. script.async = true;
  35. script.src = "https://www.googletagmanager.com/gtag/js?id=" + tagId;
  36. // Append the script element to the head of document
  37. document.head.appendChild(script);
  38. // Create global gtag() function
  39. window.dataLayer = window.dataLayer || [];
  40. window.gtag = function () {
  41. dataLayer.push(arguments);
  42. };
  43. // Initialize GA4 with tag ID
  44. window.gtag("js", new Date());
  45. window.gtag("config", tagId);
  46. } catch (e) {
  47. console.error("Error setting up Google Analytics", e);
  48. }
  49. },
  50. /**
  51. * @inheritdoc
  52. */
  53. getKey: function () {
  54. return MetacatUI.appModel.get("googleAnalyticsKey");
  55. },
  56. /**
  57. * @inheritdoc
  58. */
  59. track: window.gtag,
  60. /**
  61. * @inheritdoc
  62. */
  63. ready: function () {
  64. return this.getKey() && typeof this.track === "function";
  65. },
  66. /**
  67. * @inheritdoc
  68. */
  69. trackException: function (message, id, fatal) {
  70. if (!this.ready()) return;
  71. this.track("event", "exception", {
  72. description: this.createExceptionDescription(message, id),
  73. fatal: fatal,
  74. });
  75. },
  76. /**
  77. * @inheritdoc
  78. */
  79. trackEvent: function (category, action, label, value) {
  80. if (!this.ready()) return;
  81. // prepare the event parameters
  82. var params = {
  83. event_category: category,
  84. event_label: label,
  85. value: value,
  86. };
  87. // send the event
  88. this.track("event", action, params);
  89. },
  90. /**
  91. * @inheritdoc
  92. */
  93. trackPageView: function (path, title) {
  94. if (!this.ready()) return;
  95. if (!path) path = (window.location.pathname || "/").replace("#", "");
  96. if (!title) title = document.title;
  97. this.track("config", this.getKey(), {
  98. page_path: path,
  99. page_title: title,
  100. });
  101. },
  102. },
  103. );
  104. return GoogleAnalytics;
  105. });