___TERMS_OF_SERVICE___ By creating or modifying this file you agree to Google Tag Manager's Community Template Gallery Developer Terms of Service available at https://developers.google.com/tag-manager/gallery-tos (or such other URL as Google may provide), as modified from time to time. ___INFO___ { "displayName": "Facebook Pixel by Jabjab", "description": "This is an unofficial Google Tag Manager template for the Facebook Pixel. Originally developed by Simo Ahava.", "securityGroups": [], "categories": [ "ADVERTISING", "ANALYTICS" ], "id": "cvt_temp_public_id", "type": "TAG", "version": 1, "brand": { "displayName": "jabjab-online-marketing", "id": "github.com_jabjabonlinemarketing", "thumbnail": "" }, "containerContexts": [ "WEB" ] } ___TEMPLATE_PARAMETERS___ [ { "alwaysInSummary": true, "valueValidators": [ { "errorMessage": "You must provide a Pixel ID", "type": "NON_EMPTY" }, { "args": [ "^[0-9,]+$" ], "errorMessage": "Invalid Pixel ID format", "type": "REGEX" } ], "displayName": "Facebook Pixel ID(s)", "simpleValueType": true, "name": "pixelId", "type": "TEXT", "valueHint": "e.g. 12345678910" }, { "type": "CHECKBOX", "name": "enhancedEcommerce", "checkboxText": "Ecommerce dataLayer Integration", "simpleValueType": true, "help": "If you check this, then the Facebook pixel will populate \u003cstrong\u003eEvent Name\u003c/strong\u003e and \u003cstrong\u003eObject Properties\u003c/strong\u003e automatically from the last \u003ca href\u003d\"https://developers.google.com/analytics/devguides/collection/ga4/ecommerce?client_type\u003dgtm\"\u003eecommerce\u003c/a\u003e object pushed into the dataLayer array." }, { "type": "RADIO", "name": "eventName", "displayName": "Event Name", "radioItems": [ { "value": "standard", "displayValue": "Standard", "subParams": [ { "type": "SELECT", "name": "standardEventName", "macrosInSelect": false, "selectItems": [ { "displayValue": "PageView", "value": "PageView" }, { "displayValue": "AddPaymentInfo", "value": "AddPaymentInfo" }, { "displayValue": "AddToCart", "value": "AddToCart" }, { "displayValue": "AddToWishlist", "value": "AddToWishlist" }, { "displayValue": "CompleteRegistration", "value": "CompleteRegistration" }, { "displayValue": "Contact", "value": "Contact" }, { "displayValue": "CustomizeProduct", "value": "CustomizeProduct" }, { "displayValue": "Donate", "value": "Donate" }, { "displayValue": "FindLocation", "value": "FindLocation" }, { "displayValue": "InitiateCheckout", "value": "InitiateCheckout" }, { "displayValue": "Lead", "value": "Lead" }, { "displayValue": "Purchase", "value": "Purchase" }, { "displayValue": "Schedule", "value": "Schedule" }, { "displayValue": "Search", "value": "Search" }, { "displayValue": "StartTrial", "value": "StartTrial" }, { "displayValue": "SubmitApplication", "value": "SubmitApplication" }, { "displayValue": "Subscribe", "value": "Subscribe" }, { "displayValue": "ViewContent", "value": "ViewContent" } ], "simpleValueType": true, "defaultValue": "PageView" } ] }, { "value": "custom", "displayValue": "Custom", "subParams": [ { "type": "TEXT", "name": "customEventName", "displayName": "", "simpleValueType": true } ] }, { "value": "variable", "displayValue": "Variable", "subParams": [ { "type": "SELECT", "name": "variableEventName", "macrosInSelect": true, "selectItems": [], "simpleValueType": true } ] } ], "simpleValueType": true, "enablingConditions": [ { "paramName": "enhancedEcommerce", "paramValue": true, "type": "NOT_EQUALS" } ] }, { "type": "RADIO", "name": "eecEventName", "displayName": "Event Name", "radioItems": [ { "value": "eec", "displayValue": "Set automatically from dataLayer" }, { "value": "gtmvar", "displayValue": "Read from variable", "subParams": [ { "type": "SELECT", "name": "eecEventNameGTMVar", "displayName": "", "macrosInSelect": true, "selectItems": [], "simpleValueType": true } ] } ], "simpleValueType": true, "help": "The Ecommerce integration populates the Event Name automatically depending on what \u003cstrong\u003erecommended ecommerce event name\u003c/strong\u003e was last pushed into dataLayer (\"view_item\" -\u003e \"ViewContent\", \"add_to_cart\" -\u003e \"AddToCart\", \"add_to_wishlist\" -\u003e \"AddToWishlist\", \"begin_checkout\" -\u003e \"InitiateCheckout\", \"add_payment_info\" -\u003e \"AddPaymentInfo\". \"purchase\" -\u003e \"Purchase\").\n\nAlternatively you can select a GTM variable that returns one of the recommended ecommerce event names.", "enablingConditions": [ { "paramName": "enhancedEcommerce", "paramValue": true, "type": "EQUALS" } ] }, { "type": "TEXT", "name": "eecDefaultCurrency", "displayName": "Default currency if not present in the datalayer", "simpleValueType": true, "enablingConditions": [ { "paramName": "enhancedEcommerce", "paramValue": true, "type": "EQUALS" } ], "help": "The Ecommerce integration checks for the presence of ecommerce.currency or ecommerce.items[0].currency. If neither is found, this currency will be used. If this fields if left blank, USD will be used as default.", "textAsList": true }, { "type": "SELECT", "name": "consent", "displayName": "Consent Granted (GDPR)", "macrosInSelect": true, "selectItems": [ { "value": true, "displayValue": "True" }, { "value": false, "displayValue": "False" } ], "simpleValueType": true, "help": "If you set Consent Granted to \u003cstrong\u003efalse\u003c/strong\u003e, the pixel will not send any hits until a tag is fired where Consent Granted is set to \u003cstrong\u003etrue\u003c/strong\u003e. See \u003ca href\u003d\"https://developers.facebook.com/docs/facebook-pixel/implementation/gdpr/\"\u003ethis article\u003c/a\u003e for more information." }, { "simpleValueType": true, "name": "advancedMatching", "checkboxText": "Enable Advanced Matching", "type": "CHECKBOX" }, { "type": "GROUP", "name": "dataProcessingOptionsGroup", "displayName": "Data Processing Options", "groupStyle": "ZIPPY_CLOSED", "subParams": [ { "type": "LABEL", "name": "dpoInfo", "displayName": "Data Processing Options force this Facebook event to comply to regional regulations with regard to the processing and selling of user data. Read \u003ca href\u003d\"https://developers.facebook.com/docs/marketing-apis/data-processing-options\"\u003ethis\u003c/a\u003e for more information about how to configure this section." }, { "type": "CHECKBOX", "name": "dpoLDU", "checkboxText": "Limited Data Use (LDU)", "simpleValueType": true }, { "type": "TEXT", "name": "dpoCountry", "displayName": "Country", "simpleValueType": true, "defaultValue": 0, "enablingConditions": [ { "paramName": "dpoLDU", "paramValue": true, "type": "EQUALS" } ], "valueValidators": [ { "type": "NUMBER" } ] }, { "type": "TEXT", "name": "dpoState", "displayName": "State", "simpleValueType": true, "defaultValue": 0, "enablingConditions": [ { "paramName": "dpoLDU", "paramValue": true, "type": "EQUALS" } ], "valueValidators": [ { "type": "NUMBER" } ] } ] }, { "enablingConditions": [ { "paramName": "advancedMatching", "type": "EQUALS", "paramValue": true } ], "displayName": "Customer Information Data Parameters", "name": "advancedMatchingGroup", "groupStyle": "ZIPPY_CLOSED", "type": "GROUP", "subParams": [ { "displayName": "", "name": "advancedMatchingList", "simpleTableColumns": [ { "selectItems": [ { "displayValue": "City", "value": "ct" }, { "displayValue": "Country", "value": "cn" }, { "displayValue": "Date of Birth", "value": "db" }, { "displayValue": "Email", "value": "em" }, { "displayValue": "External ID", "value": "external_id" }, { "displayValue": "First Name", "value": "fn" }, { "displayValue": "Gender", "value": "ge" }, { "displayValue": "Last Name", "value": "ln" }, { "displayValue": "Phone", "value": "ph" }, { "displayValue": "State", "value": "st" }, { "displayValue": "Zip Code", "value": "zp" } ], "defaultValue": "", "displayName": "Parameter name", "name": "name", "isUnique": true, "type": "SELECT" }, { "defaultValue": "", "displayName": "Parameter value", "name": "value", "type": "TEXT" } ], "type": "SIMPLE_TABLE", "newRowButtonText": "Add parameter", "valueValidators": [ { "type": "NON_EMPTY" } ] } ] }, { "displayName": "Object Properties", "name": "objectPropertiesGroup", "groupStyle": "ZIPPY_CLOSED", "type": "GROUP", "subParams": [ { "type": "LABEL", "name": "enhancedEcommerceObject", "displayName": "\u003cstrong\u003eWarning!\u003c/strong\u003e Object properties are populated automatically based on the most recent \u003cstrong\u003eecommerce\u003c/strong\u003e object pushed into dataLayer. If you add properties here that are already set by the integration (content_type, contents, num_items, value, currency), then the properties you add here will override those set automatically by the integration!", "enablingConditions": [ { "paramName": "enhancedEcommerce", "paramValue": true, "type": "EQUALS" } ] }, { "type": "SELECT", "name": "objectPropertiesFromVariable", "displayName": "Load Properties From Variable", "macrosInSelect": true, "selectItems": [ { "value": false, "displayValue": "False" } ], "simpleValueType": true, "help": "You can use a variable that returns a JavaScript object with the properties you want to use. This object will be merged with any additional properties you add via the table below. Any conflicts will be resolved in favor of the properties you add to the table." }, { "name": "objectPropertyList", "simpleTableColumns": [ { "valueValidators": [], "defaultValue": "", "displayName": "Property Name", "name": "name", "isUnique": true, "type": "TEXT" }, { "defaultValue": "", "displayName": "Property Value", "name": "value", "type": "TEXT" } ], "type": "SIMPLE_TABLE", "newRowButtonText": "Add property" } ] }, { "displayName": "More Settings", "name": "moreSettingsGroup", "groupStyle": "ZIPPY_CLOSED", "type": "GROUP", "subParams": [ { "help": "Facebook collects some metadata (e.g. structured data) and user interactions (e.g. clicks) automatically. Check this box to disable this automatic configuration of the pixel.", "simpleValueType": true, "name": "disableAutoConfig", "checkboxText": "Disable Automatic Configuration", "type": "CHECKBOX" }, { "type": "CHECKBOX", "name": "disablePushState", "checkboxText": "Disable History Event Tracking", "simpleValueType": true, "help": "The Facebook Pixel tracks history events (pushState and replaceState) automatically as PageViews. Check this box to prevent the pixel from tracking such events automatically." }, { "type": "TEXT", "name": "eventId", "displayName": "Event ID", "simpleValueType": true, "help": "Set the Event ID parameter in case you are tracking the same event server-side as well. The Event ID can be used to deduplicate the same event if sent from multiple sources. See more \u003ca href\u003d\"https://developers.facebook.com/docs/marketing-api/conversions-api/deduplicate-pixel-and-server-events/\"\u003ehere\u003c/a\u003e." } ] } ] ___SANDBOXED_JS_FOR_WEB_TEMPLATE___ const createQueue = require('createQueue'); const callInWindow = require('callInWindow'); const aliasInWindow = require('aliasInWindow'); const copyFromWindow = require('copyFromWindow'); const setInWindow = require('setInWindow'); const injectScript = require('injectScript'); const makeTableMap = require('makeTableMap'); const makeNumber = require('makeNumber'); const getType = require('getType'); const copyFromDataLayer = require('copyFromDataLayer'); const math = require('Math'); const log = require('logToConsole'); const initIds = copyFromWindow('_fbq_gtm_ids') || []; const pixelIds = data.pixelId; const standardEventNames = ['AddPaymentInfo', 'AddToCart', 'AddToWishlist', 'CompleteRegistration', 'Contact', 'CustomizeProduct', 'Donate', 'FindLocation', 'InitiateCheckout', 'Lead', 'PageView', 'Purchase', 'Schedule', 'Search', 'StartTrial', 'SubmitApplication', 'Subscribe', 'ViewContent']; const ecommerce = copyFromDataLayer('ecommerce', 1); let lastEventName = copyFromDataLayer('event'); // Helper methods const fail = msg => { log(msg); data.gtmOnFailure(); }; const mergeObj = (obj, obj2) => { for (let key in obj2) { if (obj2.hasOwnProperty(key)) { obj[key] = obj2[key]; } } return obj; }; const parseEecObj = prod => { return { id: prod.item_id, quantity: prod.quantity }; }; // Initialize EEC integration let eventName, action, eecObjectProps; if (data.enhancedEcommerce) { if (!ecommerce) return fail('Facebook Pixel: No valid "ecommerce" object found in dataLayer'); if ("gtmvar" === data.eecEventName) { if (data.eecEventNameGTMVar) { lastEventName = data.eecEventNameGTMVar; } if (!lastEventName) { return fail('Facebook Pixel: No event name was returned by the selected GTM variable'); } } if ("view_item" === lastEventName) { eventName = 'ViewContent'; } else if ("add_to_cart" === lastEventName) { eventName = 'AddToCart'; } else if ("add_to_wishlist" === lastEventName) { eventName = 'AddToWishlist'; } else if ("begin_checkout" === lastEventName || "begin_checkout" === lastEventName) { eventName = 'InitiateCheckout'; } else if ("add_payment_info" === lastEventName) { eventName = 'AddPaymentInfo'; } else if ("purchase" === lastEventName) { eventName = 'Purchase'; } else return fail('Facebook Pixel: Most recently pushed ecommerce event name unsupported: ' + lastEventName); if (!ecommerce.items || getType(ecommerce.items) !== 'array') return fail('Facebook pixel: Most recently pushed "ecommerce" object did not have a valid "items" array.'); if (ecommerce.items.length === 0) return fail('Facebook pixel: Most recently pushed "ecommerce" object does not include product data in the "items" array.'); eecObjectProps = { content_type: 'product', contents: ecommerce.items.map(parseEecObj), value: ecommerce.items.reduce((acc, cur) => { const curVal = math.round(makeNumber(cur.price || 0) * (cur.quantity || 1) * 100) / 100; return acc + curVal; }, 0.0), currency: ecommerce.currency || ecommerce.items[0].currency || data.eecDefaultCurrency || 'USD' }; if (['InitiateCheckout', 'Purchase'].indexOf(eventName) > -1) eecObjectProps.num_items = ecommerce.items.reduce((acc,cur) => { return acc + makeNumber(cur.quantity || 1); }, 0); } // Build the fbq() command arguments const cidParams = data.advancedMatchingList && data.advancedMatchingList.length ? makeTableMap(data.advancedMatchingList, 'name', 'value') : {}; const objectProps = data.objectPropertyList && data.objectPropertyList.length ? makeTableMap(data.objectPropertyList, 'name', 'value') : {}; const objectPropsFromVar = getType(data.objectPropertiesFromVariable) === 'object' ? data.objectPropertiesFromVariable : {}; const mergedObjectProps = mergeObj(objectPropsFromVar, objectProps); const finalObjectProps = mergeObj(eecObjectProps || {}, mergedObjectProps); eventName = eventName || (data.eventName === 'custom' ? data.customEventName : (data.eventName === 'variable' ? data.variableEventName : data.standardEventName)); const command = standardEventNames.indexOf(eventName) === -1 ? 'trackSingleCustom' : 'trackSingle'; const consent = data.consent === false || data.consent === 'false' ? 'revoke' : 'grant'; // Utility function to use either fbq.queue[] // (if the FB SDK hasn't loaded yet), or fbq.callMethod() // if the SDK has loaded. const getFbq = () => { // Return the existing 'fbq' global method if available let fbq = copyFromWindow('fbq'); if (fbq) { return fbq; } // Initialize the 'fbq' global method to either use // fbq.callMethod or fbq.queue) setInWindow('fbq', function() { const callMethod = copyFromWindow('fbq.callMethod.apply'); if (callMethod) { callInWindow('fbq.callMethod.apply', null, arguments); } else { callInWindow('fbq.queue.push', arguments); } }); aliasInWindow('_fbq', 'fbq'); // Create the fbq.queue createQueue('fbq.queue'); // Return the global 'fbq' method, created above return copyFromWindow('fbq'); }; // Get reference to the global method const fbq = getFbq(); fbq('consent', consent); // Set Data Processing Options if (data.dpoLDU) { fbq('dataProcessingOptions', ['LDU'], makeNumber(data.dpoCountry), makeNumber(data.dpoState)); } // Handle multiple, comma-separated pixel IDs, // and initialize each ID if not done already. pixelIds.split(',').forEach(pixelId => { if (initIds.indexOf(pixelId) === -1) { // If the user has chosen to disable automatic configuration if (data.disableAutoConfig) { fbq('set', 'autoConfig', false, pixelId); } // If the user has chosen to disable pushState and replaceState tracking if (data.disablePushState) { setInWindow('fbq.disablePushState', true); } // Initialize pixel and store in global array fbq('init', pixelId, cidParams); initIds.push(pixelId); setInWindow('_fbq_gtm_ids', initIds, true); } // Call the fbq() method with the parameters defined earlier if (data.eventId) { fbq(command, pixelId, eventName, finalObjectProps, {eventID: data.eventId}); } else { fbq(command, pixelId, eventName, finalObjectProps); } }); injectScript('https://connect.facebook.net/en_US/fbevents.js', data.gtmOnSuccess, data.gtmOnFailure, 'fbPixel'); ___WEB_PERMISSIONS___ [ { "instance": { "key": { "publicId": "access_globals", "versionId": "1" }, "param": [ { "key": "keys", "value": { "type": 2, "listItem": [ { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "fbq" }, { "type": 8, "boolean": true }, { "type": 8, "boolean": true }, { "type": 8, "boolean": false } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "_fbq_gtm" }, { "type": 8, "boolean": true }, { "type": 8, "boolean": true }, { "type": 8, "boolean": false } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "_fbq" }, { "type": 8, "boolean": false }, { "type": 8, "boolean": true }, { "type": 8, "boolean": false } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "_fbq_gtm_ids" }, { "type": 8, "boolean": true }, { "type": 8, "boolean": true }, { "type": 8, "boolean": false } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "fbq.callMethod.apply" }, { "type": 8, "boolean": true }, { "type": 8, "boolean": false }, { "type": 8, "boolean": true } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "fbq.queue.push" }, { "type": 8, "boolean": false }, { "type": 8, "boolean": false }, { "type": 8, "boolean": true } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "fbq.queue" }, { "type": 8, "boolean": true }, { "type": 8, "boolean": true }, { "type": 8, "boolean": false } ] }, { "type": 3, "mapKey": [ { "type": 1, "string": "key" }, { "type": 1, "string": "read" }, { "type": 1, "string": "write" }, { "type": 1, "string": "execute" } ], "mapValue": [ { "type": 1, "string": "fbq.disablePushState" }, { "type": 8, "boolean": true }, { "type": 8, "boolean": true }, { "type": 8, "boolean": false } ] } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "inject_script", "versionId": "1" }, "param": [ { "key": "urls", "value": { "type": 2, "listItem": [ { "type": 1, "string": "https://connect.facebook.net/en_US/fbevents.js" } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "logging", "versionId": "1" }, "param": [ { "key": "environments", "value": { "type": 1, "string": "debug" } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true }, { "instance": { "key": { "publicId": "read_data_layer", "versionId": "1" }, "param": [ { "key": "allowedKeys", "value": { "type": 1, "string": "specific" } }, { "key": "keyPatterns", "value": { "type": 2, "listItem": [ { "type": 1, "string": "ecommerce" }, { "type": 1, "string": "event" } ] } } ] }, "clientAnnotations": { "isEditedByUser": true }, "isRequired": true } ] ___TESTS___ scenarios: - name: Library is injected code: |- // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('injectScript').wasCalledWith(scriptUrl, success, failure, 'fbPixel'); assertApi('gtmOnSuccess').wasCalled(); - name: fbq does not exist - method created code: |- let fbq; mock('copyFromWindow', key => { if (key === 'fbq') return fbq; }); mock('createQueue', key => {}); mock('setInWindow', (key, val) => { if (key === 'fbq') fbq = val; }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('aliasInWindow').wasCalledWith('_fbq', 'fbq'); assertApi('setInWindow').wasCalled(); assertApi('gtmOnSuccess').wasCalled(); - name: fbq exists - method copied code: |- mock('setInWindow', key => { if (key === 'fbq') fail('setInWindow called with fbq even though variable exists'); }); mock('createQueue', key => {}); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: makeTableMap called code: |- mockData.advancedMatching = true; // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('makeTableMap').wasCalledWith(mockData.advancedMatchingList, 'name', 'value'); assertApi('makeTableMap').wasCalledWith(mockData.objectPropertyList, 'name', 'value'); assertApi('gtmOnSuccess').wasCalled(); - name: Consent set code: |- mock('copyFromWindow', key => { if (key === 'fbq') return function() { if (arguments[0] === 'consent') { assertThat(arguments[1], 'Consent set incorrectly').isEqualTo('grant'); } }; }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: DPO LDU set code: |- mockData.dpoLDU = true; mockData.dpoCountry = '0'; mockData.dpoState = '0'; mock('copyFromWindow', key => { if (key === 'fbq') return function() { if (arguments[0] === 'consent') { assertThat(arguments[1], 'Consent set incorrectly').isEqualTo('grant'); } if (arguments[0] === 'dataProcessingOptions') { assertThat(arguments[1], 'LDU array value not set').isEqualTo(['LDU']); assertThat(arguments[2], 'LDU country not set').isEqualTo(0); assertThat(arguments[3], 'LDU state not set').isEqualTo(0); } }; }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: DPO LDU not set code: |- mock('copyFromWindow', key => { if (key === 'fbq') return function() { if (arguments[0] === 'consent') { assertThat(arguments[1], 'Consent set incorrectly').isEqualTo('grant'); } if (arguments[0] === 'dataProcessingOptions') { fail('dataProcessingOptions called even though DPO was not set'); } }; }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: Pixel IDs set - do not initialize code: |- mock('copyFromWindow', key => { if (key === '_fbq_gtm_ids') return ['12345', '23456']; if (key === 'fbq') return function() { if (arguments[0] === 'init') fail('init called even though pixel IDs already initialized'); }; }); // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('gtmOnSuccess').wasCalled(); - name: Pixel IDs not set - run init process code: "let index = 0;\nlet count = 0;\nlet _fbq_gtm_ids;\n\nmockData.advancedMatching\ \ = true;\nmockData.disableAutoConfig = true;\nmockData.disablePushState = true;\n\ \nmock('setInWindow', (key, val) => {\n if (key === 'fbq.disablePushState') count\ \ += 1;\n if (key === '_fbq_gtm_ids') _fbq_gtm_ids = val;\n});\n\nconst initObj\ \ = {\n ct: 'Helsinki',\n cn: 'Finland',\n external_id: 'UserId'\n};\n\nmock('copyFromWindow',\ \ key => {\n if (key === 'fbq') return function() {\n if (arguments[0] ===\ \ 'set' && arguments[1] === 'autoConfig' && arguments[2] === false) {\n assertThat(arguments[3],\ \ 'autoConfig called with incorrect pixelId').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ }\n if (arguments[0] === 'set' && arguments[1] === 'agent') {\n assertThat(arguments[2],\ \ 'agent set with invalid value').isEqualTo('tmSimo-GTM-WebTemplate');\n \ \ assertThat(arguments[3], 'agent set with invalid pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ index += 1;\n }\n if (arguments[0] === 'init') {\n assertThat(arguments[1],\ \ 'init called with incorrect pixelId').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'init called with incorrect initObj').isEqualTo(initObj);\n\ \ } \n };\n});\n\n// Call runCode to run the template's code.\nrunCode(mockData);\n\ \nassertThat(_fbq_gtm_ids, '_fbq_gtm_ids has incorrect contents').isEqualTo(mockData.pixelId.split(','));\n\ assertThat(index, 'init called incorrect number of times: ' + index).isEqualTo(2);\n\ assertThat(count, 'fbq.disablePushState called incorrect number of times: ' +\ \ count).isEqualTo(2);\n\n// Verify that the tag finished successfully.\nassertApi('gtmOnSuccess').wasCalled();" - name: Send standard event code: "const eventParams = {\n prop1: 'val1',\n prop2: 'val2'\n};\n\nlet index\ \ = 0;\nmock('copyFromWindow', key => {\n if (key === 'fbq') return function()\ \ {\n if (arguments[0] === 'trackSingle') {\n assertThat(arguments[1],\ \ 'trackSingle called with incorrect pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'trackSingle called with incorrect event name').isEqualTo(mockData.standardEventName);\n\ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(eventParams);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Send custom event code: "mockData.eventName = 'custom';\n\nconst eventParams = {\n prop1: 'val1',\n\ \ prop2: 'val2'\n};\n\nlet index = 0;\nmock('copyFromWindow', key => {\n if\ \ (key === 'fbq') return function() {\n if (arguments[0] === 'trackSingleCustom')\ \ {\n assertThat(arguments[1], 'trackSingleCustom called with incorrect pixel\ \ ID').isEqualTo(mockData.pixelId.split(',')[index]);\n assertThat(arguments[2],\ \ 'trackSingleCustom called with incorrect event name').isEqualTo(mockData.customEventName);\n\ \ assertThat(arguments[3], 'trackSingleCustom called with incorrect event\ \ parameters').isEqualTo(eventParams);\n index += 1;\n }\n };\n});\n\ \ \n// Call runCode to run the template's code.\nrunCode(mockData);\n\n//\ \ Verify that the tag finished successfully.\nassertThat(index, 'trackSingleCustom\ \ called incorrect number of times').isEqualTo(2);\nassertApi('gtmOnSuccess').wasCalled();" - name: Send variable event with standard name code: "mockData.eventName = 'variable';\nmockData.variableEventName = 'PageView';\n\ \nconst eventParams = {\n prop1: 'val1',\n prop2: 'val2'\n};\n\nlet index =\ \ 0;\nmock('copyFromWindow', key => {\n if (key === 'fbq') return function()\ \ {\n if (arguments[0] === 'trackSingle') {\n assertThat(arguments[1],\ \ 'trackSingle called with incorrect pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'trackSingle called with incorrect event name').isEqualTo(mockData.variableEventName);\n\ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(eventParams);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Send variable event with custom name code: "mockData.eventName = 'variable';\nmockData.variableEventName = 'custom';\n\ \nconst eventParams = {\n prop1: 'val1',\n prop2: 'val2'\n};\n\nlet index =\ \ 0;\nmock('copyFromWindow', key => {\n if (key === 'fbq') return function()\ \ {\n if (arguments[0] === 'trackSingleCustom') {\n assertThat(arguments[1],\ \ 'trackSingleCustom called with incorrect pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'trackSingleCustom called with incorrect event\ \ name').isEqualTo(mockData.variableEventName);\n assertThat(arguments[3],\ \ 'trackSingleCustom called with incorrect event parameters').isEqualTo(eventParams);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingleCustom called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Send event parameters from a variable code: "mockData.objectPropertiesFromVariable = {\n prop1: 'val1',\n prop2: 'val2'\n\ };\n\nlet index = 0;\nmock('copyFromWindow', key => {\n if (key === 'fbq') return\ \ function() {\n if (arguments[0] === 'trackSingle') {\n assertThat(arguments[1],\ \ 'trackSingle called with incorrect pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'trackSingle called with incorrect event name').isEqualTo(mockData.standardEventName);\n\ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(mockData.objectPropertiesFromVariable);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Enhanced Ecommerce integration fails with invalid object code: |- mockData.enhancedEcommerce = true; // Call runCode to run the template's code. runCode(mockData); // Verify that the tag finished successfully. assertApi('logToConsole').wasCalledWith('Facebook Pixel: No valid "ecommerce" object found in dataLayer'); assertApi('gtmOnFailure').wasCalled(); assertApi('gtmOnSuccess').wasNotCalled(); - name: Enhanced Ecommerce integration fails with invalid action code: "mockData.enhancedEcommerce = true;\n\nmock('copyFromDataLayer', key => {\n\ \ if (key === 'ecommerce') return {\n invalid: true\n };\n \n if (key ===\ \ \"event\") return \"view_item\";\n});\n\n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertApi('logToConsole').wasCalledWith('Facebook pixel: Most recently pushed\ \ \"ecommerce\" object did not have a valid \"items\" array.');\nassertApi('gtmOnFailure').wasCalled();\n\ assertApi('gtmOnSuccess').wasNotCalled();" - name: Enhanced Ecommerce ViewContent works code: "mockData.enhancedEcommerce = true;\nmockData.objectPropertyList = {};\n\n\ mock('copyFromDataLayer', key => {\n if (key === 'ecommerce') return {\n currency:\ \ 'EUR',\n items: mockEec.gtm.products\n };\n\n if (key === 'event') return\ \ \"view_item\";\n});\n\nlet index = 0;\nmock('copyFromWindow', key => {\n if\ \ (key === 'fbq') return function() {\n if (arguments[0] === 'trackSingle')\ \ {\n assertThat(arguments[1], 'trackSingle called with incorrect pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'trackSingle called with incorrect event name').isEqualTo('ViewContent');\n\ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(mockEec.fb);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Enhanced Ecommerce AddToCart works code: "mockData.enhancedEcommerce = true;\nmockData.objectPropertyList = {};\n\n\ mock('copyFromDataLayer', key => {\n if (key === 'ecommerce') return {\n currency:\ \ 'EUR',\n items: mockEec.gtm.products\n };\n\n\n if (key === 'event') return\ \ \"add_to_cart\";\n});\n\nlet index = 0;\nmock('copyFromWindow', key => {\n \ \ if (key === 'fbq') return function() {\n if (arguments[0] === 'trackSingle')\ \ {\n assertThat(arguments[1], 'trackSingle called with incorrect pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'trackSingle called with incorrect event name').isEqualTo('AddToCart');\n\ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(mockEec.fb);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Enhanced Ecommerce InitiateCheckout works code: "mockData.enhancedEcommerce = true;\nmockEec.fb.num_items = 3;\nmockData.objectPropertyList\ \ = {};\n\nmock('copyFromDataLayer', key => {\n if (key === 'ecommerce') return\ \ {\n currency: 'EUR',\n items: mockEec.gtm.products\n };\n\n if (key\ \ === 'event') return \"begin_checkout\";\n});\n\nlet index = 0;\nmock('copyFromWindow',\ \ key => {\n if (key === 'fbq') return function() {\n if (arguments[0] ===\ \ 'trackSingle') {\n assertThat(arguments[1], 'trackSingle called with incorrect\ \ pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n assertThat(arguments[2],\ \ 'trackSingle called with incorrect event name').isEqualTo('InitiateCheckout');\n\ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(mockEec.fb);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Enhanced Ecommerce Purchase works code: "mockData.enhancedEcommerce = true;\nmockEec.fb.num_items = 3;\nmockData.objectPropertyList\ \ = {};\n\nmock('copyFromDataLayer', key => {\n if (key === 'ecommerce') return\ \ {\n currency: 'EUR',\n items: mockEec.gtm.products\n };\n\n if (key\ \ === 'event') return \"purchase\";\n});\n\nlet index = 0;\nmock('copyFromWindow',\ \ key => {\n if (key === 'fbq') return function() {\n if (arguments[0] ===\ \ 'trackSingle') {\n assertThat(arguments[1], 'trackSingle called with incorrect\ \ pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n assertThat(arguments[2],\ \ 'trackSingle called with incorrect event name').isEqualTo('Purchase');\n \ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(mockEec.fb);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Object merge with variable and list works code: "mockData.objectPropertiesFromVariable = {\n prop1: 'var1',\n prop2: 'var2',\n\ \ prop3: 'var3'\n};\n\nconst expected = {\n prop1: 'val1',\n prop2: 'val2',\n\ \ prop3: 'var3'\n};\n\nlet index = 0;\nmock('copyFromWindow', key => {\n if\ \ (key === 'fbq') return function() {\n if (arguments[0] === 'trackSingle')\ \ {\n assertThat(arguments[1], 'trackSingle called with incorrect pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'trackSingle called with incorrect event name').isEqualTo('PageView');\n\ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(expected);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Object merge with variable, list and eec works code: "mockData.enhancedEcommerce = true;\nmockData.objectPropertiesFromVariable\ \ = {\n content_type: 'product_group'\n};\nmockData.objectPropertyList = [{\n\ \ name: 'currency',\n value: 'USD'\n}];\nmockEec.fb.num_items = 3;\nmockEec.fb.content_type\ \ = 'product_group';\nmockEec.fb.currency = 'USD';\n\nmock('copyFromDataLayer',\ \ key => {\n if (key === 'ecommerce') return {\n currency: 'EUR',\n items:\ \ mockEec.gtm.products\n };\n\n if (key === 'event') return \"purchase\";\n\ });\n\nlet index = 0;\nmock('copyFromWindow', key => {\n if (key === 'fbq') return\ \ function() {\n if (arguments[0] === 'trackSingle') {\n assertThat(arguments[1],\ \ 'trackSingle called with incorrect pixel ID').isEqualTo(mockData.pixelId.split(',')[index]);\n\ \ assertThat(arguments[2], 'trackSingle called with incorrect event name').isEqualTo('Purchase');\n\ \ assertThat(arguments[3], 'trackSingle called with incorrect event parameters').isEqualTo(mockEec.fb);\n\ \ index += 1;\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertThat(index, 'trackSingle called incorrect number of times').isEqualTo(2);\n\ assertApi('gtmOnSuccess').wasCalled();" - name: Send event ID code: "mockData.eventId = 'eventId';\n\nmock('copyFromWindow', key => {\n if (key\ \ === 'fbq') return function() {\n if (arguments[0] === 'trackSingle') {\n\ \ assertThat(arguments[4], 'eventID not included in hit').isEqualTo({eventID:\ \ mockData.eventId});\n }\n };\n});\n \n// Call runCode to run the template's\ \ code.\nrunCode(mockData);\n\n// Verify that the tag finished successfully.\n\ assertApi('gtmOnSuccess').wasCalled();" setup: "const mockData = {\n pixelId: '12345,23456',\n eventName: 'standard',\n\ \ standardEventName: 'PageView',\n customEventName: 'custom',\n variableEventName:\ \ 'standard',\n consent: true,\n advancedMatching: false,\n advancedMatchingList:\ \ [{name: 'ct', value: 'Helsinki'},{name: 'cn', value: 'Finland'},{name: 'external_id',\ \ value: 'UserId'}],\n objectPropertiesFromVariable: false,\n objectPropertyList:\ \ [{name: 'prop1', value: 'val1'},{name: 'prop2', value: 'val2'}],\n disableAutoConfig:\ \ false,\n disablePushState: false,\n enhancedEcommerce: false,\n eventId: ''\n\ };\n\nconst mockEec = {\n gtm: { \n products: [{\n item_id: 'i1',\n \ \ item_name: 'n1',\n item_category: 'c1',\n price: '1.00',\n quantity:\ \ 1\n },{\n item_id: 'i2',\n item_name: 'n2',\n item_category:\ \ 'c2',\n price: '2.00',\n quantity: 2\n }]\n },\n fb: {\n content_type:\ \ 'product',\n contents: [{\n id: 'i1',\n quantity: 1\n },{\n \ \ id: 'i2',\n quantity: 2\n }],\n currency: 'EUR',\n value: 5.00\n\ \ }\n};\n\nconst scriptUrl = 'https://connect.facebook.net/en_US/fbevents.js';\n\ \n// Create injectScript mock\nlet success, failure;\nmock('injectScript', (url,\ \ onsuccess, onfailure) => {\n success = onsuccess;\n failure = onfailure;\n \ \ onsuccess();\n});\n\nmock('copyFromWindow', key => {\n if (key === 'fbq') return\ \ () => {};\n});" ___NOTES___ Created on 18/05/2019, 21:57:16