import { Injectable, Inject, isDevMode } from '@angular/core'; import { NGX_GOOGLE_ANALYTICS_SETTINGS_TOKEN } from '../tokens/ngx-google-analytics-settings-token'; import { DOCUMENT } from '@angular/common'; import { NGX_GTAG_FN } from '../tokens/ngx-gtag-token'; import * as i0 from "@angular/core"; export class GoogleAnalyticsService { constructor(settings, _document, _gtag) { this.settings = settings; this._document = _document; this._gtag = _gtag; } get document() { return this._document; } throw(err) { if ((this.settings.enableTracing || isDevMode()) && console && console.error) { console.error(err); } } /** @todo Change this to `Object.fromEntity()` in the future... */ toKeyValue(map) { return (map.size > 0) ? Array.from(map).reduce((obj, [key, value]) => Object.defineProperty(obj, key, { value, enumerable: true }), {}) : undefined; } /** * Call native GA Tag */ gtag(...args) { try { this._gtag(...args.filter(x => x !== undefined)); } catch (err) { this.throw(err); } } /** * Send an event trigger to GA. It is the same as call: * ```js * gtag('event', 'video_auto_play_start', { * 'event_label': 'My promotional video', * 'event_category': 'video_auto_play' * }); * ``` * * @param action 'video_auto_play_start' * @param category 'video_auto_play' * @param label 'My promotional video' * @param value An value to measure something * @param interaction If user interaction is performed */ event(action, category, label, value, interaction, options) { try { const opt = new Map(); if (category) { opt.set('event_category', category); } if (label) { opt.set('event_label', label); } if (value) { opt.set('value', value); } if (interaction !== undefined) { opt.set('interaction', interaction); } if (options) { Object .entries(options) .map(([key, value]) => opt.set(key, value)); } const params = this.toKeyValue(opt); if (params) { this.gtag('event', action, params); } else { this.gtag('event', action); } } catch (error) { this.throw(error); } } /** * Send an page view event. This is the same as * * ```js * gtag('config', 'GA_TRACKING_ID', { * 'page_title' : 'Homepage', * 'page_path': '/home' * }); * ``` * * The tracking ID is injected automatically by Inject Token NGX_GOOGLE_ANALYTICS_SETTINGS_TOKEN * * @param path /home * @param title Homepage * @param location '{ page_location }' * @param options '{ ... custom dimentions }' */ pageView(path, title, location, options) { try { const opt = new Map([['page_path', path]]); if (title) { opt.set('page_title', title); } if (location || this.document) { opt.set('page_location', (location || this.document.location.href)); } if (options) { Object .entries(options) .map(([key, value]) => opt.set(key, value)); } this.gtag('config', this.settings.trackingCode, this.toKeyValue(opt)); } catch (error) { this.throw(error); } } /** * Send an event to report a App Page View. It is the same as * * ```js * gtag('event', 'screen_view', { * 'app_name': 'myAppName', * 'screen_name' : 'Home' * }); * * ``` * * @param screen 'screen_name' * @param appName 'app_name' * @param appId 'app_id' * @param appVersion 'app_version' * @param installerId 'app_installer_id' */ appView(screen, appName, appId, appVersion, installerId) { try { const opt = new Map([['screen_name', screen], ['app_name', appName]]); if (appId) { opt.set('app_id', appId); } if (appVersion) { opt.set('app_version', appVersion); } if (installerId) { opt.set('app_installer_id', installerId); } this.gtag('event', 'screen_view', this.toKeyValue(opt)); } catch (error) { this.throw(error); } } /** * Defines persistent values on GoogleAnalytics * * @see https://developers.google.com/analytics/devguides/collection/gtagjs/setting-values * * ```js * gtag('set', { * 'currency': 'USD', * 'country': 'US' * }); * ``` */ set(...options) { try { this._gtag('set', ...options); } catch (err) { this.throw(err); } } /** * Send an event to GA to report an application error. It is the same as * * ```js * gtag('event', 'exception', { * 'description': 'error_description', * 'fatal': false // set to true if the error is fatal * }); * ``` * * @param description 'error_description' * @param fatal set to true if the error is fatal */ exception(description, fatal) { try { const opt = new Map(); if (description) { opt.set('description', description); } if (fatal) { opt.set('fatal', fatal); } const params = this.toKeyValue(opt); if (params) { this.gtag('event', 'exception', this.toKeyValue(opt)); } else { this.gtag('event', 'exception'); } } catch (error) { this.throw(error); } } } GoogleAnalyticsService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.2", ngImport: i0, type: GoogleAnalyticsService, deps: [{ token: NGX_GOOGLE_ANALYTICS_SETTINGS_TOKEN }, { token: DOCUMENT }, { token: NGX_GTAG_FN }], target: i0.ɵɵFactoryTarget.Injectable }); GoogleAnalyticsService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.0.2", ngImport: i0, type: GoogleAnalyticsService, providedIn: 'root' }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.2", ngImport: i0, type: GoogleAnalyticsService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: function () { return [{ type: undefined, decorators: [{ type: Inject, args: [NGX_GOOGLE_ANALYTICS_SETTINGS_TOKEN] }] }, { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT] }] }, { type: undefined, decorators: [{ type: Inject, args: [NGX_GTAG_FN] }] }]; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"google-analytics.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-google-analytics/src/lib/services/google-analytics.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, isDevMode } from '@angular/core';\nimport { NGX_GOOGLE_ANALYTICS_SETTINGS_TOKEN } from '../tokens/ngx-google-analytics-settings-token';\nimport { IGoogleAnalyticsSettings } from '../interfaces/i-google-analytics-settings';\nimport { GaActionEnum } from '../enums/ga-action.enum';\nimport { DOCUMENT } from '@angular/common';\nimport { NGX_GTAG_FN } from '../tokens/ngx-gtag-token';\nimport { GtagFn } from '../types/gtag.type';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class GoogleAnalyticsService {\n\n  private get document(): Document {\n    return this._document;\n  }\n\n  constructor(\n    @Inject(NGX_GOOGLE_ANALYTICS_SETTINGS_TOKEN) private readonly settings: IGoogleAnalyticsSettings,\n    @Inject(DOCUMENT) private readonly _document: any,\n    @Inject(NGX_GTAG_FN) private readonly _gtag: GtagFn\n  ) { }\n\n  private throw(err: Error) {\n    if ((this.settings.enableTracing || isDevMode()) && console && console.error) {\n      console.error(err);\n    }\n  }\n\n  /** @todo Change this to `Object.fromEntity()` in the future... */\n  private toKeyValue(map: Map<string, any>): { [param: string]: any } | void {\n    return (map.size > 0)\n      ? Array.from(map).reduce(\n        (obj, [key, value]) => Object.defineProperty(obj, key, { value, enumerable: true }),\n        {}\n      )\n      : undefined;\n  }\n\n  /**\n   * Call native GA Tag\n   */\n  gtag(...args: any[]) {\n    try {\n      this._gtag(...args.filter(x => x !== undefined));\n    } catch (err) {\n      this.throw(err);\n    }\n  }\n\n  /**\n   * Send an event trigger to GA. It is the same as call:\n   * ```js\n   * gtag('event', 'video_auto_play_start', {\n   *   'event_label': 'My promotional video',\n   *   'event_category': 'video_auto_play'\n   * });\n   * ```\n   *\n   * @param action 'video_auto_play_start'\n   * @param category 'video_auto_play'\n   * @param label 'My promotional video'\n   * @param value An value to measure something\n   * @param interaction If user interaction is performed\n   */\n  event(action: GaActionEnum | string, category?: string, label?: string, value?: number, interaction?: boolean, options?: Object) {\n    try {\n      const opt = new Map<string, any>();\n      if (category) {\n        opt.set('event_category', category);\n      }\n      if (label) {\n        opt.set('event_label', label);\n      }\n      if (value) {\n        opt.set('value', value);\n      }\n      if (interaction !== undefined) {\n        opt.set('interaction', interaction);\n      }\n      if (options) {\n        Object\n          .entries(options)\n          .map(([key, value]) => opt.set(key, value));\n      }      \n      const params = this.toKeyValue(opt);\n      if (params) {\n        this.gtag('event', action as string, params);\n      } else {\n        this.gtag('event', action as string);\n      }\n    } catch (error) {\n      this.throw(error);\n    }\n  }\n\n  /**\n   * Send an page view event. This is the same as\n   *\n   * ```js\n   * gtag('config', 'GA_TRACKING_ID', {\n   *   'page_title' : 'Homepage',\n   *   'page_path': '/home'\n   * });\n   * ```\n   *\n   * The tracking ID is injected automatically by Inject Token NGX_GOOGLE_ANALYTICS_SETTINGS_TOKEN\n   *\n   * @param path /home\n   * @param title Homepage\n   * @param location '{ page_location }'\n   * @param options '{ ... custom dimentions }'\n   */\n  pageView( path: string, title?: string, location?: string, options?: Object) {\n    try {\n      const opt = new Map<string, any>([['page_path', path]]);\n      if (title) {\n        opt.set('page_title', title);\n      }\n      if (location || this.document) {\n        opt.set('page_location', (location || this.document.location.href));\n      }\n      if (options) {\n        Object\n          .entries(options)\n          .map(([key, value]) => opt.set(key, value));\n      }\n      this.gtag('config', this.settings.trackingCode, this.toKeyValue(opt));\n    } catch (error) {\n      this.throw(error);\n    }\n  }\n\n  /**\n   * Send an event to report a App Page View. It is the same as\n   *\n   * ```js\n   * gtag('event', 'screen_view', {\n   *   'app_name': 'myAppName',\n   *   'screen_name' : 'Home'\n   * });\n   *\n   * ```\n   *\n   * @param screen 'screen_name'\n   * @param appName 'app_name'\n   * @param appId 'app_id'\n   * @param appVersion 'app_version'\n   * @param installerId 'app_installer_id'\n   */\n  appView(screen: string, appName: string, appId?: string, appVersion?: string, installerId?: string) {\n    try {\n      const opt = new Map<string, any>([['screen_name', screen], ['app_name', appName]]);\n      if (appId) {\n        opt.set('app_id', appId);\n      }\n      if (appVersion) {\n        opt.set('app_version', appVersion);\n      }\n      if (installerId) {\n        opt.set('app_installer_id', installerId);\n      }\n      this.gtag('event', 'screen_view', this.toKeyValue(opt));\n    } catch (error) {\n      this.throw(error);\n    }\n  }\n\n  /**\n   * Defines persistent values on GoogleAnalytics\n   *\n   * @see https://developers.google.com/analytics/devguides/collection/gtagjs/setting-values\n   *\n   * ```js\n   * gtag('set', {\n   *   'currency': 'USD',\n   *   'country': 'US'\n   * });\n   * ```\n   */\n  set(...options: Array<any>) {\n    try {\n      this._gtag('set', ...options);\n    } catch (err) {\n      this.throw(err);\n    }\n  }\n\n  /**\n   * Send an event to GA to report an application error. It is the same as\n   *\n   * ```js\n   * gtag('event', 'exception', {\n   *   'description': 'error_description',\n   *   'fatal': false   // set to true if the error is fatal\n   * });\n   * ```\n   *\n   * @param description 'error_description'\n   * @param fatal set to true if the error is fatal\n   */\n  exception(description?: string, fatal?: boolean) {\n    try {\n      const opt = new Map<string, any>();\n      if (description) {\n        opt.set('description', description);\n      }\n      if (fatal) {\n        opt.set('fatal', fatal);\n      }\n      const params = this.toKeyValue(opt);\n      if (params) {\n        this.gtag('event', 'exception', this.toKeyValue(opt));\n      } else {\n        this.gtag('event', 'exception');\n      }\n    } catch (error) {\n      this.throw(error);\n    }\n  }\n}\n"]}