UNPKG

ngx-pixel

Version:

An Angular library to simplify the use of a Facebook Pixel.

199 lines (194 loc) 7.54 kB
import { ɵɵdefineInjectable, ɵɵinject, PLATFORM_ID, RendererFactory2, Injectable, Inject, Optional, NgModule } from '@angular/core'; import { NavigationEnd, Router } from '@angular/router'; import { isPlatformBrowser, DOCUMENT } from '@angular/common'; import { filter } from 'rxjs/operators'; class PixelService { constructor(config, injectedDocument, platformId, router, rendererFactory) { this.config = config; this.injectedDocument = injectedDocument; this.platformId = platformId; this.router = router; this.rendererFactory = rendererFactory; // DOCUMENT cannot be injected directly as Document type, see https://github.com/angular/angular/issues/20351 // It is therefore injected as any and then cast to Document this.doc = injectedDocument; this.renderer = rendererFactory.createRenderer(null, null); if (router) { // Log page views after router navigation ends router.events.pipe(filter(event => event instanceof NavigationEnd)).subscribe(event => { if (this.isLoaded()) { this.track('PageView'); } }); } } /** * Initialize the Pixel tracking script * - Adds the script to page's head * - Tracks first page view */ initialize(pixelId = this.config.pixelId) { if (this.isLoaded()) { console.warn('Tried to initialize a Pixel instance while another is already active. Please call `remove()` before initializing a new instance.'); return; } this.config.enabled = true; this.addPixelScript(pixelId); } /** Remove the Pixel tracking script */ remove() { this.removePixelScript(); this.config.enabled = false; } /** * Track a Standard Event as predefined by Facebook * * See {@link https://developers.facebook.com/docs/facebook-pixel/reference Facebook Pixel docs - reference} * @param eventName The name of the event that is being tracked * @param properties Optional properties of the event */ track(eventName, properties) { if (!isPlatformBrowser(this.platformId)) { return; } if (!this.isLoaded()) { console.warn('Tried to track an event without initializing a Pixel instance. Call `initialize()` first.'); return; } if (properties) { fbq('track', eventName, properties); } else { fbq('track', eventName); } } /** * Track a custom Event * * See {@link https://developers.facebook.com/docs/facebook-pixel/implementation/conversion-tracking#custom-conversions Facebook Pixel docs - custom conversions} * @param eventName The name of the event that is being tracked * @param properties Optional properties of the event */ trackCustom(eventName, properties) { if (!isPlatformBrowser(this.platformId)) { return; } if (!this.isLoaded()) { console.warn('Tried to track an event without initializing a Pixel instance. Call `initialize()` first.'); return; } if (properties) { fbq('trackCustom', eventName, properties); } else { fbq('trackCustom', eventName); } } /** * Adds the Facebook Pixel tracking script to the application * @param pixelId The Facebook Pixel ID to use */ addPixelScript(pixelId) { if (!isPlatformBrowser(this.platformId)) { return; } const pixelCode = ` var pixelCode = function(f,b,e,v,n,t,s) {if(f.fbq)return;n=f.fbq=function(){n.callMethod? n.callMethod.apply(n,arguments):n.queue.push(arguments)}; if(!f._fbq)f._fbq=n;n.push=n;n.loaded=!0;n.version='2.0'; n.queue=[];t=b.createElement(e);t.async=!0; t.src=v;s=b.getElementsByTagName(e)[0]; s.parentNode.insertBefore(t,s)}(window, document,'script', 'https://connect.facebook.net/en_US/fbevents.js'); fbq('init', '${pixelId}'); fbq('track', 'PageView');`; const scriptElement = this.renderer.createElement('script'); this.renderer.setAttribute(scriptElement, 'id', 'pixel-script'); this.renderer.setAttribute(scriptElement, 'type', 'text/javascript'); this.renderer.setProperty(scriptElement, 'innerHTML', pixelCode); this.renderer.appendChild(this.doc.head, scriptElement); } /** Remove Facebook Pixel tracking script from the application */ removePixelScript() { if (!isPlatformBrowser(this.platformId)) { return; } const pixelElement = this.doc.getElementById('pixel-script'); if (pixelElement) { pixelElement.remove(); } } /** Checks if the script element is present */ isLoaded() { if (isPlatformBrowser(this.platformId)) { const pixelElement = this.doc.getElementById('pixel-script'); return !!pixelElement; } return false; } } PixelService.ɵprov = ɵɵdefineInjectable({ factory: function PixelService_Factory() { return new PixelService(ɵɵinject("config"), ɵɵinject(DOCUMENT), ɵɵinject(PLATFORM_ID), ɵɵinject(Router, 8), ɵɵinject(RendererFactory2)); }, token: PixelService, providedIn: "root" }); PixelService.decorators = [ { type: Injectable, args: [{ providedIn: 'root' },] } ]; PixelService.ctorParameters = () => [ { type: undefined, decorators: [{ type: Inject, args: ['config',] }] }, { type: undefined, decorators: [{ type: Inject, args: [DOCUMENT,] }] }, { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }, { type: Router, decorators: [{ type: Optional }] }, { type: RendererFactory2 } ]; class PixelModule { constructor(pixel, platformId) { this.pixel = pixel; if (!PixelModule.config) { throw Error('ngx-pixel not configured correctly. Pass the `pixelId` property to the `forRoot()` function'); } if (PixelModule.config.enabled && isPlatformBrowser(platformId)) { this.pixel.initialize(); } } /** * Initiale the Facebook Pixel Module * * Add your Pixel ID as parameter */ static forRoot(config) { this.config = config; const pixelId = config.pixelId; this.verifyPixelId(pixelId); return { ngModule: PixelModule, providers: [PixelService, { provide: 'config', useValue: config }] }; } /** * Verifies the Pixel ID that was passed into the configuration. * - Checks if Pixel was initialized * @param pixelId Pixel ID to verify */ static verifyPixelId(pixelId) { // Have to verify first that all Pixel IDs follow the same 15 digit format if (pixelId === null || pixelId === undefined || pixelId.length === 0) { throw Error('Invalid Facebook Pixel ID. Did you pass the ID into the forRoot() function?'); } } } PixelModule.config = null; PixelModule.decorators = [ { type: NgModule, args: [{ imports: [], },] } ]; PixelModule.ctorParameters = () => [ { type: PixelService }, { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] } ]; /** * Generated bundle index. Do not edit. */ export { PixelModule, PixelService }; //# sourceMappingURL=ngx-pixel.js.map