UNPKG

@aws-amplify/analytics

Version:

Analytics category of aws-amplify

113 lines (111 loc) 4.45 kB
'use strict'; // Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. // SPDX-License-Identifier: Apache-2.0 Object.defineProperty(exports, "__esModule", { value: true }); exports.PageViewTracker = void 0; const core_1 = require("@aws-amplify/core"); const utils_1 = require("@aws-amplify/core/internals/utils"); const logger = new core_1.ConsoleLogger('PageViewTracker'); const DEFAULT_EVENT_NAME = 'pageView'; const DEFAULT_APP_TYPE = 'singlePage'; const DEFAULT_URL_PROVIDER = () => { return window.location.origin + window.location.pathname; }; const PREV_URL_STORAGE_KEY = 'aws-amplify-analytics-prevUrl'; class PageViewTracker { constructor(eventRecorder, options) { this.options = {}; this.trackerActive = true; this.eventRecorder = eventRecorder; this.spaTrackingActive = false; this.handleLocationChange = this.handleLocationChange.bind(this); this.configure(eventRecorder, options); } configure(eventRecorder, options) { this.eventRecorder = eventRecorder; // Clean up any existing listeners this.cleanup(); // Apply defaults this.options = { appType: options?.appType ?? DEFAULT_APP_TYPE, attributes: options?.attributes ?? undefined, eventName: this.options?.eventName ?? DEFAULT_EVENT_NAME, urlProvider: this.options?.urlProvider ?? DEFAULT_URL_PROVIDER, }; // Configure SPA or MPA page view tracking if ((0, utils_1.isBrowser)()) { if (this.options.appType === 'singlePage') { this.setupSPATracking(); } else { this.setupMPATracking(); } this.trackerActive = true; } } cleanup() { // No-op if document listener is not active if (!this.trackerActive) { return; } // Clean up SPA page view listeners if (this.spaTrackingActive) { window.history.pushState = this.originalPushState; window.history.replaceState = this.originalReplaceState; this.pushStateProxy?.revoke(); this.replaceStateProxy?.revoke(); window.removeEventListener('popstate', this.handleLocationChange); this.spaTrackingActive = false; } } setupSPATracking() { if (!this.spaTrackingActive) { // Configure proxies on History APIs this.pushStateProxy = Proxy.revocable(window.history.pushState, { apply: (target, thisArg, args) => { target.apply(thisArg, args); this.handleLocationChange(); }, }); this.replaceStateProxy = Proxy.revocable(window.history.replaceState, { apply: (target, thisArg, args) => { target.apply(thisArg, args); this.handleLocationChange(); }, }); this.originalPushState = window.history.pushState; this.originalReplaceState = window.history.replaceState; window.history.pushState = this.pushStateProxy.proxy; window.history.replaceState = this.replaceStateProxy.proxy; window.addEventListener('popstate', this.handleLocationChange); sessionStorage.removeItem(PREV_URL_STORAGE_KEY); this.spaTrackingActive = true; } } setupMPATracking() { this.handleLocationChange(); } handleLocationChange() { const currentUrl = this.options.urlProvider(); const eventName = this.options.eventName || DEFAULT_EVENT_NAME; if (this.urlHasChanged()) { sessionStorage.setItem(PREV_URL_STORAGE_KEY, currentUrl); // Assemble attribute list const attributes = Object.assign({ url: currentUrl, }, this.options.attributes); logger.debug('Recording automatically tracked page view event', { eventName, attributes, }); this.eventRecorder(eventName, attributes); } } urlHasChanged() { const prevUrl = sessionStorage.getItem(PREV_URL_STORAGE_KEY); const currUrl = this.options.urlProvider(); return currUrl !== prevUrl; } } exports.PageViewTracker = PageViewTracker; //# sourceMappingURL=PageViewTracker.js.map