@walkeros/web-destination-gtag 0.4.2 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/shared/gtag.ts","../src/ga4/config.ts","../src/shared/parameters.ts","../src/shared/mapping.ts","../src/ga4/push.ts","../src/ads/config.ts","../src/ads/push.ts","../src/gtm/config.ts","../src/gtm/push.ts","../src/index.ts","../src/types/index.ts"],"sourcesContent":["// Gtag script loading and initialization utilities\n\nconst loadedScripts = new Set<string>();\n\n// For testing: allow resetting loaded scripts\nexport function resetLoadedScripts(): void {\n loadedScripts.clear();\n}\n\nexport function addScript(\n id: string,\n src = 'https://www.googletagmanager.com/gtag/js?id=',\n document: Document = globalThis.document,\n): void {\n // Prevent loading the same script multiple times\n if (loadedScripts.has(id)) return;\n\n const script = document.createElement('script');\n script.src = src + id;\n document.head.appendChild(script);\n loadedScripts.add(id);\n}\n\nexport function initializeGtag(window: Window): Gtag.Gtag | undefined {\n const w = window;\n\n // Setup dataLayer if not exists\n w.dataLayer = w.dataLayer || [];\n\n // Setup gtag function if not exists\n if (!w.gtag) {\n w.gtag = function () {\n (w.dataLayer as unknown[]).push(arguments);\n };\n }\n\n return w.gtag;\n}\n","import type { WalkerOS, Logger } from '@walkeros/core';\nimport type { GA4Settings } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { addScript, initializeGtag } from '../shared/gtag';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function initGA4(\n settings: GA4Settings,\n loadScript: boolean | undefined,\n env: DestinationWeb.Env | undefined,\n logger: Logger.Instance,\n): void {\n const { window, document } = getEnv(env);\n const { measurementId, transport_url, server_container_url, pageview } =\n settings;\n\n if (!measurementId) logger.throw('Config settings ga4.measurementId missing');\n\n // Load the gtag script\n if (loadScript) addScript(measurementId, undefined, document as Document);\n\n // Initialize gtag infrastructure\n initializeGtag(window as Window);\n\n const gtagSettings: WalkerOS.AnyObject = {};\n\n // custom transport_url\n if (transport_url) gtagSettings.transport_url = transport_url;\n\n // custom server_container_url\n if (server_container_url)\n gtagSettings.server_container_url = server_container_url;\n\n // disable pageviews\n if (pageview === false) gtagSettings.send_page_view = false;\n\n const gtag = window.gtag as Gtag.Gtag;\n gtag('js', new Date());\n\n // gtag init call\n gtag('config', measurementId, gtagSettings);\n}\n","import type { WalkerOS } from '@walkeros/core';\nimport type { Include, Parameters } from '../types';\n\nexport function getParamsInclude(\n event: WalkerOS.DeepPartialEvent,\n include: Include,\n): Parameters {\n const params: Parameters = {};\n\n // Check for the 'all' group to add each group\n if (include.includes('all'))\n include = [\n 'context',\n 'data',\n 'event',\n 'globals',\n 'source',\n 'user',\n 'version',\n ];\n\n include.forEach((groupName) => {\n let group = event[groupName as keyof Omit<WalkerOS.Event, 'all'>] || {};\n\n // Create a fake group for event properties\n if (groupName == 'event')\n group = {\n id: event.id,\n timing: event.timing,\n trigger: event.trigger,\n entity: event.entity,\n action: event.action,\n group: event.group,\n count: event.count,\n };\n\n Object.entries(group).forEach(([key, val]) => {\n // Different value access for context\n if (groupName == 'context') val = (val as WalkerOS.OrderedProperties)[0];\n\n params[`${groupName}_${key}`] = val;\n });\n });\n\n return params;\n}\n","// Shared mapping utilities for all Google tools\nimport type { WalkerOS, Collector } from '@walkeros/core';\nimport { getMappingValue, assign, isObject } from '@walkeros/core';\nimport type { BaseSettings, Config } from '../types';\n\nexport function normalizeEventName(\n eventName: string,\n snakeCase = true,\n): string {\n if (!snakeCase) return eventName;\n\n return eventName.replace(/\\s+/g, '_').toLowerCase();\n}\n\nexport async function getData(\n event: WalkerOS.Event,\n data: WalkerOS.AnyObject | undefined,\n config: Config,\n toolSettings: BaseSettings | undefined,\n collector: Collector.Instance,\n): Promise<WalkerOS.AnyObject> {\n const baseData = isObject(data) ? data : {};\n\n // Get config-level mapped data\n const configMappedData = config.data\n ? await getMappingValue(event, config.data, { collector })\n : {};\n\n // Get tool-specific mapped data\n const toolMappedData = toolSettings?.data\n ? await getMappingValue(event, toolSettings.data, { collector })\n : {};\n\n // Ensure all values are objects before merging\n const configData = isObject(configMappedData) ? configMappedData : {};\n const toolData = isObject(toolMappedData) ? toolMappedData : {};\n\n // Merge with proper priority: base <- config <- tool\n return assign(assign(baseData, configData), toolData);\n}\n","import type { WalkerOS, Logger } from '@walkeros/core';\nimport type { GA4Settings, GA4Mapping, Parameters } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { isObject } from '@walkeros/core';\nimport { getParamsInclude } from '../shared/parameters';\nimport { normalizeEventName } from '../shared/mapping';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function pushGA4Event(\n event: WalkerOS.Event,\n settings: GA4Settings,\n mapping: GA4Mapping = {},\n data: WalkerOS.AnyObject,\n env: DestinationWeb.Env | undefined,\n logger: Logger.Instance,\n): void {\n const { window } = getEnv(env);\n\n if (!settings.measurementId)\n logger.throw('Config settings ga4.measurementId missing');\n\n const eventData = isObject(data) ? data : {};\n\n const paramsInclude = getParamsInclude(\n event,\n // Add data to include by default\n mapping.include || settings.include || ['data'],\n );\n\n const eventParams: Parameters = {\n ...paramsInclude,\n ...eventData,\n };\n\n // Event name (snake_case default)\n let eventName = event.name; // Assume custom mapped name\n if (settings.snakeCase !== false) {\n // Use snake case if not disabled\n eventName = normalizeEventName(eventName);\n }\n\n // Set the GA4 stream id\n eventParams.send_to = settings.measurementId;\n\n // Debug mode\n if (settings.debug) eventParams.debug_mode = true;\n\n const gtag = window.gtag as Gtag.Gtag;\n gtag('event', eventName, eventParams);\n}\n","import type { Logger } from '@walkeros/core';\nimport type { AdsSettings } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { addScript, initializeGtag } from '../shared/gtag';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function initAds(\n settings: AdsSettings,\n loadScript: boolean | undefined,\n env: DestinationWeb.Env | undefined,\n logger: Logger.Instance,\n): void {\n const { window, document } = getEnv(env);\n const { conversionId } = settings;\n\n if (!conversionId) logger.throw('Config settings ads.conversionId missing');\n\n // Default currency value\n if (!settings.currency) settings.currency = 'EUR';\n\n if (loadScript) addScript(conversionId, undefined, document as Document);\n\n // Initialize gtag infrastructure\n initializeGtag(window as Window);\n\n const gtag = window.gtag as Gtag.Gtag;\n gtag('js', new Date());\n\n // gtag init call\n gtag('config', conversionId);\n}\n","import type { WalkerOS, Logger } from '@walkeros/core';\nimport type { AdsSettings, AdsMapping } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { isObject } from '@walkeros/core';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function pushAdsEvent(\n event: WalkerOS.Event,\n settings: AdsSettings,\n mapping: AdsMapping = {},\n data: WalkerOS.AnyObject,\n mappingName: string | undefined,\n env: DestinationWeb.Env | undefined,\n logger: Logger.Instance,\n): void {\n const { conversionId, currency } = settings;\n const eventData = isObject(data) ? data : {};\n\n // Use label from mapping settings, fallback to mappingName for backward compatibility\n const conversionLabel = mapping.label || mappingName;\n if (!conversionLabel) logger.throw('Config mapping ads.label missing');\n\n const params: Gtag.CustomParams = {\n send_to: `${conversionId}/${conversionLabel}`,\n currency: currency || 'EUR',\n ...eventData,\n };\n\n const { window } = getEnv(env);\n const gtag = window.gtag as Gtag.Gtag;\n gtag('event', 'conversion', params);\n}\n","import type { Logger } from '@walkeros/core';\nimport type { GTMSettings } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { getEnv } from '@walkeros/web-core';\n\nconst defaultDataLayer = 'dataLayer';\nconst defaultDomain = 'https://www.googletagmanager.com/gtm.js?id=';\n\nexport function initGTM(\n settings: GTMSettings,\n loadScript?: boolean,\n env?: DestinationWeb.Env,\n _logger?: Logger.Instance,\n): void {\n const { window, document } = getEnv(env);\n const { containerId, dataLayer, domain } = settings;\n const dataLayerName = dataLayer || defaultDataLayer;\n\n // Initialize the dataLayer (default or custom name)\n if (dataLayerName === defaultDataLayer) {\n window.dataLayer = window.dataLayer || [];\n } else {\n window[dataLayerName] = (window[dataLayerName] as unknown[]) || [];\n }\n\n // Get the appropriate dataLayer array\n const dataLayerArray = window[dataLayerName] as unknown[];\n\n dataLayerArray.push({\n 'gtm.start': new Date().getTime(),\n event: 'gtm.js',\n });\n\n // Load the gtm script and container\n if (loadScript && containerId)\n addScript(\n containerId,\n domain || defaultDomain,\n dataLayerName,\n document as Document,\n );\n}\n\nfunction addScript(\n containerId: string,\n src: string,\n dataLayerName: string,\n document: Document = globalThis.document,\n) {\n const dl = dataLayerName != defaultDataLayer ? '&l=' + dataLayerName : '';\n const script = document.createElement('script');\n script.src = src + containerId + dl;\n document.head.appendChild(script);\n}\n","import type { WalkerOS, Logger } from '@walkeros/core';\nimport type { GTMSettings, GTMMapping } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { isObject } from '@walkeros/core';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function pushGTMEvent(\n event: WalkerOS.Event,\n settings: GTMSettings,\n mapping: GTMMapping = {},\n data: WalkerOS.AnyObject,\n env?: DestinationWeb.Env,\n _logger?: Logger.Instance,\n): void {\n const { window } = getEnv(env);\n const obj = { event: event.name }; // Use the name mapping by default\n\n (window.dataLayer as unknown[]).push({\n ...obj,\n ...(isObject(data) ? data : event),\n });\n}\n","import type { WalkerOS, On, Collector, Logger } from '@walkeros/core';\nimport type { Settings, Destination, ConsentMapping } from './types';\nimport { initGA4, pushGA4Event } from './ga4';\nimport { initAds, pushAdsEvent } from './ads';\nimport { initGTM, pushGTMEvent } from './gtm';\nimport { getData } from './shared/mapping';\nimport { initializeGtag } from './shared/gtag';\nimport { getEnv } from '@walkeros/web-core';\n\n// Types\nexport * as DestinationGtag from './types';\n\n// Track whether default consent has been set\nlet defaultConsentSet = false;\n\n// For testing: allow resetting consent state\nexport function resetConsentState(): void {\n defaultConsentSet = false;\n}\n\n// Default consent mapping: walkerOS consent groups → gtag consent parameters\nconst DEFAULT_CONSENT_MAPPING: ConsentMapping = {\n marketing: ['ad_storage', 'ad_user_data', 'ad_personalization'],\n functional: ['analytics_storage'],\n};\n\nexport const destinationGtag: Destination = {\n type: 'google-gtag',\n\n config: { settings: {} },\n\n init({ config, env, logger }) {\n const { settings = {} as Partial<Settings>, loadScript } = config;\n const { ga4, ads, gtm } = settings;\n\n // Fail if no tools are configured\n if (!ga4?.measurementId && !ads?.conversionId && !gtm?.containerId) {\n logger.throw(\n 'Config settings missing. Set ga4.measurementId, ads.conversionId, or gtm.containerId',\n );\n }\n\n // Initialize GA4 if configured\n if (ga4?.measurementId) {\n initGA4(ga4, loadScript, env, logger);\n }\n\n // Initialize Google Ads if configured\n if (ads?.conversionId) {\n initAds(ads, loadScript, env, logger);\n }\n\n // Initialize GTM if configured\n if (gtm?.containerId) {\n initGTM(gtm, loadScript, env, logger);\n }\n\n return config;\n },\n\n async push(event, { config, mapping = {}, data, env, collector, logger }) {\n const { settings = {} } = config;\n const { ga4, ads, gtm } = settings;\n const eventMapping = mapping.settings || {};\n\n // Resolve data for each tool with proper priority\n const ga4Data = await getData(\n event,\n data as WalkerOS.AnyObject,\n config,\n ga4,\n collector,\n );\n const adsData = await getData(\n event,\n data as WalkerOS.AnyObject,\n config,\n ads,\n collector,\n );\n const gtmData = await getData(\n event,\n data as WalkerOS.AnyObject,\n config,\n gtm,\n collector,\n );\n\n // Push to GA4 if configured\n if (ga4?.measurementId) {\n pushGA4Event(event, ga4, eventMapping.ga4, ga4Data, env, logger);\n }\n\n // @TODO: Fix condition - should check for mapping.settings?.ads?.label || mapping.name\n // Currently requires mapping.name even when label is provided via settings.ads.label\n if (ads?.conversionId && mapping.name) {\n pushAdsEvent(\n event,\n ads,\n eventMapping.ads,\n adsData,\n mapping.name,\n env,\n logger,\n );\n }\n\n // Push to GTM if configured\n if (gtm?.containerId) {\n pushGTMEvent(event, gtm, eventMapping.gtm, gtmData, env, logger);\n }\n },\n\n on(type, context) {\n // Only handle consent events\n if (type !== 'consent' || !context) return;\n\n const consent = context;\n\n // Access config directly from this destination instance\n const settings = this.config?.settings || {};\n const { como = true } = settings;\n\n // Skip if consent mode is disabled\n if (!como) return;\n\n // Ensure gtag is available\n const { window } = getEnv(this.env);\n const gtag = initializeGtag(window as Window);\n if (!gtag) return;\n\n // Determine consent mapping to use\n const consentMapping: ConsentMapping =\n como === true ? DEFAULT_CONSENT_MAPPING : como;\n\n // If this is the first consent call, set default to denied for all parameters\n if (!defaultConsentSet) {\n // Get all possible gtag parameters from the mapping\n const allGtagParams = new Set<string>();\n Object.values(consentMapping).forEach((params) => {\n const paramArray = Array.isArray(params) ? params : [params];\n paramArray.forEach((param) => allGtagParams.add(param));\n });\n\n // Only call default if we have parameters to set\n if (allGtagParams.size > 0) {\n const defaultConsent: Record<string, 'denied'> = {};\n allGtagParams.forEach((param) => {\n defaultConsent[param] = 'denied';\n });\n\n // Call default with all denied\n gtag('consent', 'default', defaultConsent);\n }\n\n defaultConsentSet = true;\n }\n\n // Build gtag consent object for update\n const gtagConsent: Record<string, 'granted' | 'denied'> = {};\n\n // Map walkerOS consent to gtag consent parameters for update\n Object.entries(consent as WalkerOS.Consent).forEach(\n ([walkerOSGroup, granted]) => {\n const gtagParams = consentMapping[walkerOSGroup];\n if (!gtagParams) return;\n\n const params = Array.isArray(gtagParams) ? gtagParams : [gtagParams];\n const consentValue = granted ? 'granted' : 'denied';\n\n params.forEach((param) => {\n gtagConsent[param] = consentValue;\n });\n },\n );\n\n // Only proceed if we have consent parameters to update\n if (Object.keys(gtagConsent).length === 0) return;\n\n // Always use update after the initial default\n gtag('consent', 'update', gtagConsent);\n },\n};\n\nexport default destinationGtag;\n","import type {\n Mapping as WalkerOSMapping,\n Destination as CoreDestination,\n} from '@walkeros/core';\nimport type { DestinationWeb } from '@walkeros/web-core';\n\n// Extend Window interface with specific dataLayer typing\ninterface WindowWithDataLayer extends Window {\n dataLayer: unknown[];\n gtag?: Gtag.Gtag;\n [key: string]: unknown; // For custom dataLayer names and other properties\n}\n\ndeclare global {\n interface Window {\n gtag?: Gtag.Gtag;\n [key: string]: unknown; // For custom dataLayer names and other properties\n }\n}\n\n// Gtag-specific environment interface\nexport interface Env extends DestinationWeb.Env {\n window: {\n gtag: Gtag.Gtag;\n dataLayer: unknown[];\n };\n document: {\n createElement: (tagName: string) => {\n src: string;\n async?: boolean;\n setAttribute: (name: string, value: string) => void;\n removeAttribute: (name: string) => void;\n };\n head: {\n appendChild: (node: unknown) => void;\n };\n };\n}\n\n// Consent mode configuration\nexport type ConsentMode =\n | false // Disable consent mode\n | true // Use default mapping\n | ConsentMapping; // Custom mapping\n\nexport interface ConsentMapping {\n [walkerOSConsentGroup: string]: string | string[];\n}\n\n// Type bundle\nexport type Types = CoreDestination.Types<Settings, Mapping, Env, InitSettings>;\n\nexport type Destination = DestinationWeb.Destination<Types>;\nexport type Config = DestinationWeb.Config<Types>;\n\n// Base settings interface with data field\nexport interface BaseSettings {\n data?: WalkerOSMapping.Value | WalkerOSMapping.Values;\n}\n\n// Unified settings for all Google tools\nexport interface Settings {\n // Consent mode configuration\n como?: ConsentMode;\n // GA4 settings\n ga4?: GA4Settings;\n // Google Ads settings\n ads?: AdsSettings;\n // GTM settings\n gtm?: GTMSettings;\n}\n\n// InitSettings: user input (all optional)\nexport type InitSettings = Partial<Settings>;\n\n// GA4-specific settings\nexport interface GA4Settings extends BaseSettings {\n measurementId: string;\n debug?: boolean;\n include?: Include;\n pageview?: boolean;\n server_container_url?: string;\n snakeCase?: boolean;\n transport_url?: string;\n}\n\n// Google Ads specific settings\nexport interface AdsSettings extends BaseSettings {\n conversionId: string;\n currency?: string;\n}\n\n// Google Ads specific mapping\nexport interface AdsMapping {\n label?: string; // Conversion label for this specific event\n}\n\n// GTM specific settings\nexport interface GTMSettings extends BaseSettings {\n containerId: string;\n dataLayer?: string;\n domain?: string;\n}\n\n// Unified mapping interface\nexport interface Mapping {\n ga4?: GA4Mapping;\n ads?: AdsMapping;\n gtm?: GTMMapping;\n}\n\n// GA4-specific mapping\nexport interface GA4Mapping {\n include?: Include;\n}\n\n// GTM specific mapping\nexport interface GTMMapping {}\n\nexport type Rule = WalkerOSMapping.Rule<Mapping>;\nexport type Rules = WalkerOSMapping.Rules<Rule>;\nexport type Param = WalkerOSMapping.Value;\n\nexport type Include = Array<\n | 'all'\n | 'context'\n | 'data'\n | 'event'\n | 'globals'\n | 'source'\n | 'user'\n | 'version'\n>;\n\nexport type Parameters = Gtag.ControlParams &\n Gtag.EventParams &\n Gtag.CustomParams;\n"],"mappings":"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","names":["document","document","getEnv","getEnv","document","getEnv","getEnv","document","addScript","getEnv","getEnv","getEnv"]}
1
+ {"version":3,"sources":["../src/shared/gtag.ts","../src/ga4/config.ts","../src/shared/parameters.ts","../src/shared/mapping.ts","../src/ga4/push.ts","../src/ads/config.ts","../src/ads/push.ts","../src/gtm/config.ts","../src/gtm/push.ts","../src/index.ts","../src/types/index.ts"],"sourcesContent":["// Gtag script loading and initialization utilities\n\nconst loadedScripts = new Set<string>();\n\n// For testing: allow resetting loaded scripts\nexport function resetLoadedScripts(): void {\n loadedScripts.clear();\n}\n\nexport function addScript(\n id: string,\n src = 'https://www.googletagmanager.com/gtag/js?id=',\n document: Document = globalThis.document,\n): void {\n // Prevent loading the same script multiple times\n if (loadedScripts.has(id)) return;\n\n const script = document.createElement('script');\n script.src = src + id;\n document.head.appendChild(script);\n loadedScripts.add(id);\n}\n\nexport function initializeGtag(window: Window): Gtag.Gtag | undefined {\n const w = window;\n\n // Setup dataLayer if not exists\n w.dataLayer = w.dataLayer || [];\n\n // Setup gtag function if not exists\n if (!w.gtag) {\n w.gtag = function () {\n (w.dataLayer as unknown[]).push(arguments);\n };\n }\n\n return w.gtag;\n}\n","import type { WalkerOS, Logger } from '@walkeros/core';\nimport type { GA4Settings } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { addScript, initializeGtag } from '../shared/gtag';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function initGA4(\n settings: GA4Settings,\n loadScript: boolean | undefined,\n env: DestinationWeb.Env | undefined,\n logger: Logger.Instance,\n): void {\n const { window, document } = getEnv(env);\n const { measurementId, transport_url, server_container_url, pageview } =\n settings;\n\n if (!measurementId) logger.throw('Config settings ga4.measurementId missing');\n\n // Load the gtag script\n if (loadScript) addScript(measurementId, undefined, document as Document);\n\n // Initialize gtag infrastructure\n initializeGtag(window as Window);\n\n const gtagSettings: WalkerOS.AnyObject = {};\n\n // custom transport_url\n if (transport_url) gtagSettings.transport_url = transport_url;\n\n // custom server_container_url\n if (server_container_url)\n gtagSettings.server_container_url = server_container_url;\n\n // disable pageviews\n if (pageview === false) gtagSettings.send_page_view = false;\n\n const gtag = window.gtag as Gtag.Gtag;\n gtag('js', new Date());\n\n // gtag init call\n gtag('config', measurementId, gtagSettings);\n}\n","import type { WalkerOS } from '@walkeros/core';\nimport type { Include, Parameters } from '../types';\n\nexport function getParamsInclude(\n event: WalkerOS.DeepPartialEvent,\n include: Include,\n): Parameters {\n const params: Parameters = {};\n\n // Check for the 'all' group to add each group\n if (include.includes('all'))\n include = [\n 'context',\n 'data',\n 'event',\n 'globals',\n 'source',\n 'user',\n 'version',\n ];\n\n include.forEach((groupName) => {\n let group = event[groupName as keyof Omit<WalkerOS.Event, 'all'>] || {};\n\n // Create a fake group for event properties\n if (groupName == 'event')\n group = {\n id: event.id,\n timing: event.timing,\n trigger: event.trigger,\n entity: event.entity,\n action: event.action,\n group: event.group,\n count: event.count,\n };\n\n Object.entries(group).forEach(([key, val]) => {\n // Different value access for context\n if (groupName == 'context') val = (val as WalkerOS.OrderedProperties)[0];\n\n params[`${groupName}_${key}`] = val;\n });\n });\n\n return params;\n}\n","// Shared mapping utilities for all Google tools\nimport type { WalkerOS, Collector } from '@walkeros/core';\nimport { getMappingValue, assign, isObject } from '@walkeros/core';\nimport type { BaseSettings, Config } from '../types';\n\nexport function normalizeEventName(\n eventName: string,\n snakeCase = true,\n): string {\n if (!snakeCase) return eventName;\n\n return eventName.replace(/\\s+/g, '_').toLowerCase();\n}\n\nexport async function getData(\n event: WalkerOS.Event,\n data: WalkerOS.AnyObject | undefined,\n config: Config,\n toolSettings: BaseSettings | undefined,\n collector: Collector.Instance,\n): Promise<WalkerOS.AnyObject> {\n const baseData = isObject(data) ? data : {};\n\n // Get config-level mapped data\n const configMappedData = config.data\n ? await getMappingValue(event, config.data, { collector })\n : {};\n\n // Get tool-specific mapped data\n const toolMappedData = toolSettings?.data\n ? await getMappingValue(event, toolSettings.data, { collector })\n : {};\n\n // Ensure all values are objects before merging\n const configData = isObject(configMappedData) ? configMappedData : {};\n const toolData = isObject(toolMappedData) ? toolMappedData : {};\n\n // Merge with proper priority: base <- config <- tool\n return assign(assign(baseData, configData), toolData);\n}\n","import type { WalkerOS, Logger } from '@walkeros/core';\nimport type { GA4Settings, GA4Mapping, Parameters } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { isObject } from '@walkeros/core';\nimport { getParamsInclude } from '../shared/parameters';\nimport { normalizeEventName } from '../shared/mapping';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function pushGA4Event(\n event: WalkerOS.Event,\n settings: GA4Settings,\n mapping: GA4Mapping = {},\n data: WalkerOS.AnyObject,\n env: DestinationWeb.Env | undefined,\n logger: Logger.Instance,\n): void {\n const { window } = getEnv(env);\n\n if (!settings.measurementId)\n logger.throw('Config settings ga4.measurementId missing');\n\n const eventData = isObject(data) ? data : {};\n\n const paramsInclude = getParamsInclude(\n event,\n // Add data to include by default\n mapping.include || settings.include || ['data'],\n );\n\n const eventParams: Parameters = {\n ...paramsInclude,\n ...eventData,\n };\n\n // Event name (snake_case default)\n let eventName = event.name; // Assume custom mapped name\n if (settings.snakeCase !== false) {\n // Use snake case if not disabled\n eventName = normalizeEventName(eventName);\n }\n\n // Set the GA4 stream id\n eventParams.send_to = settings.measurementId;\n\n // Debug mode\n if (settings.debug) eventParams.debug_mode = true;\n\n const gtag = window.gtag as Gtag.Gtag;\n gtag('event', eventName, eventParams);\n}\n","import type { Logger } from '@walkeros/core';\nimport type { AdsSettings } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { addScript, initializeGtag } from '../shared/gtag';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function initAds(\n settings: AdsSettings,\n loadScript: boolean | undefined,\n env: DestinationWeb.Env | undefined,\n logger: Logger.Instance,\n): void {\n const { window, document } = getEnv(env);\n const { conversionId } = settings;\n\n if (!conversionId) logger.throw('Config settings ads.conversionId missing');\n\n // Default currency value\n if (!settings.currency) settings.currency = 'EUR';\n\n if (loadScript) addScript(conversionId, undefined, document as Document);\n\n // Initialize gtag infrastructure\n initializeGtag(window as Window);\n\n const gtag = window.gtag as Gtag.Gtag;\n gtag('js', new Date());\n\n // gtag init call\n gtag('config', conversionId);\n}\n","import type { WalkerOS, Logger } from '@walkeros/core';\nimport type { AdsSettings, AdsMapping } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { isObject } from '@walkeros/core';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function pushAdsEvent(\n event: WalkerOS.Event,\n settings: AdsSettings,\n mapping: AdsMapping = {},\n data: WalkerOS.AnyObject,\n mappingName: string | undefined,\n env: DestinationWeb.Env | undefined,\n logger: Logger.Instance,\n): void {\n const { conversionId, currency } = settings;\n const eventData = isObject(data) ? data : {};\n\n // Use label from mapping settings, fallback to mappingName for backward compatibility\n const conversionLabel = mapping.label || mappingName;\n if (!conversionLabel) logger.throw('Config mapping ads.label missing');\n\n const params: Gtag.CustomParams = {\n send_to: `${conversionId}/${conversionLabel}`,\n currency: currency || 'EUR',\n ...eventData,\n };\n\n const { window } = getEnv(env);\n const gtag = window.gtag as Gtag.Gtag;\n gtag('event', 'conversion', params);\n}\n","import type { Logger } from '@walkeros/core';\nimport type { GTMSettings } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { getEnv } from '@walkeros/web-core';\n\nconst defaultDataLayer = 'dataLayer';\nconst defaultDomain = 'https://www.googletagmanager.com/gtm.js?id=';\n\nexport function initGTM(\n settings: GTMSettings,\n loadScript?: boolean,\n env?: DestinationWeb.Env,\n _logger?: Logger.Instance,\n): void {\n const { window, document } = getEnv(env);\n const { containerId, dataLayer, domain } = settings;\n const dataLayerName = dataLayer || defaultDataLayer;\n\n // Initialize the dataLayer (default or custom name)\n if (dataLayerName === defaultDataLayer) {\n window.dataLayer = window.dataLayer || [];\n } else {\n window[dataLayerName] = (window[dataLayerName] as unknown[]) || [];\n }\n\n // Get the appropriate dataLayer array\n const dataLayerArray = window[dataLayerName] as unknown[];\n\n dataLayerArray.push({\n 'gtm.start': new Date().getTime(),\n event: 'gtm.js',\n });\n\n // Load the gtm script and container\n if (loadScript && containerId)\n addScript(\n containerId,\n domain || defaultDomain,\n dataLayerName,\n document as Document,\n );\n}\n\nfunction addScript(\n containerId: string,\n src: string,\n dataLayerName: string,\n document: Document = globalThis.document,\n) {\n const dl = dataLayerName != defaultDataLayer ? '&l=' + dataLayerName : '';\n const script = document.createElement('script');\n script.src = src + containerId + dl;\n document.head.appendChild(script);\n}\n","import type { WalkerOS, Logger } from '@walkeros/core';\nimport type { GTMSettings, GTMMapping } from '../types';\nimport type { DestinationWeb } from '@walkeros/web-core';\nimport { isObject } from '@walkeros/core';\nimport { getEnv } from '@walkeros/web-core';\n\nexport function pushGTMEvent(\n event: WalkerOS.Event,\n settings: GTMSettings,\n mapping: GTMMapping = {},\n data: WalkerOS.AnyObject,\n env?: DestinationWeb.Env,\n _logger?: Logger.Instance,\n): void {\n const { window } = getEnv(env);\n const obj = { event: event.name }; // Use the name mapping by default\n\n (window.dataLayer as unknown[]).push({\n ...obj,\n ...(isObject(data) ? data : event),\n });\n}\n","import type { WalkerOS, On, Collector, Logger } from '@walkeros/core';\nimport type { Settings, Destination, ConsentMapping } from './types';\nimport { initGA4, pushGA4Event } from './ga4';\nimport { initAds, pushAdsEvent } from './ads';\nimport { initGTM, pushGTMEvent } from './gtm';\nimport { getData } from './shared/mapping';\nimport { initializeGtag } from './shared/gtag';\nimport { getEnv } from '@walkeros/web-core';\n\n// Types\nexport * as DestinationGtag from './types';\n\n// Track whether default consent has been set\nlet defaultConsentSet = false;\n\n// For testing: allow resetting consent state\nexport function resetConsentState(): void {\n defaultConsentSet = false;\n}\n\n// Default consent mapping: walkerOS consent groups → gtag consent parameters\nconst DEFAULT_CONSENT_MAPPING: ConsentMapping = {\n marketing: ['ad_storage', 'ad_user_data', 'ad_personalization'],\n functional: ['analytics_storage'],\n};\n\nexport const destinationGtag: Destination = {\n type: 'google-gtag',\n\n config: { settings: {} },\n\n init({ config, env, logger }) {\n const { settings = {} as Partial<Settings>, loadScript } = config;\n const { ga4, ads, gtm } = settings;\n\n // Fail if no tools are configured\n if (!ga4?.measurementId && !ads?.conversionId && !gtm?.containerId) {\n logger.throw(\n 'Config settings missing. Set ga4.measurementId, ads.conversionId, or gtm.containerId',\n );\n }\n\n // Initialize GA4 if configured\n if (ga4?.measurementId) {\n initGA4(ga4, loadScript, env, logger);\n }\n\n // Initialize Google Ads if configured\n if (ads?.conversionId) {\n initAds(ads, loadScript, env, logger);\n }\n\n // Initialize GTM if configured\n if (gtm?.containerId) {\n initGTM(gtm, loadScript, env, logger);\n }\n\n return config;\n },\n\n async push(event, { config, mapping = {}, data, env, collector, logger }) {\n const { settings = {} } = config;\n const { ga4, ads, gtm } = settings;\n const eventMapping = mapping.settings || {};\n\n // Resolve data for each tool with proper priority\n const ga4Data = await getData(\n event,\n data as WalkerOS.AnyObject,\n config,\n ga4,\n collector,\n );\n const adsData = await getData(\n event,\n data as WalkerOS.AnyObject,\n config,\n ads,\n collector,\n );\n const gtmData = await getData(\n event,\n data as WalkerOS.AnyObject,\n config,\n gtm,\n collector,\n );\n\n // Push to GA4 if configured\n if (ga4?.measurementId) {\n pushGA4Event(event, ga4, eventMapping.ga4, ga4Data, env, logger);\n }\n\n // @TODO: Fix condition - should check for mapping.settings?.ads?.label || mapping.name\n // Currently requires mapping.name even when label is provided via settings.ads.label\n if (ads?.conversionId && mapping.name) {\n pushAdsEvent(\n event,\n ads,\n eventMapping.ads,\n adsData,\n mapping.name,\n env,\n logger,\n );\n }\n\n // Push to GTM if configured\n if (gtm?.containerId) {\n pushGTMEvent(event, gtm, eventMapping.gtm, gtmData, env, logger);\n }\n },\n\n on(type, context) {\n // Only handle consent events\n if (type !== 'consent' || !context.data) return;\n\n const consent = context.data as WalkerOS.Consent;\n const settings = (context.config?.settings || {}) as Partial<Settings>;\n const { como = true } = settings;\n\n // Skip if consent mode is disabled\n if (!como) return;\n\n // Ensure gtag is available\n const { window } = getEnv(context.env);\n const gtag = initializeGtag(window as Window);\n if (!gtag) return;\n\n // Determine consent mapping to use\n const consentMapping: ConsentMapping =\n como === true ? DEFAULT_CONSENT_MAPPING : como;\n\n // If this is the first consent call, set default to denied for all parameters\n if (!defaultConsentSet) {\n // Get all possible gtag parameters from the mapping\n const allGtagParams = new Set<string>();\n Object.values(consentMapping).forEach((params) => {\n const paramArray = Array.isArray(params) ? params : [params];\n paramArray.forEach((param) => allGtagParams.add(param));\n });\n\n // Only call default if we have parameters to set\n if (allGtagParams.size > 0) {\n const defaultConsent: Record<string, 'denied'> = {};\n allGtagParams.forEach((param) => {\n defaultConsent[param] = 'denied';\n });\n\n // Call default with all denied\n gtag('consent', 'default', defaultConsent);\n }\n\n defaultConsentSet = true;\n }\n\n // Build gtag consent object for update\n const gtagConsent: Record<string, 'granted' | 'denied'> = {};\n\n // Map walkerOS consent to gtag consent parameters for update\n Object.entries(consent).forEach(([walkerOSGroup, granted]) => {\n const gtagParams = consentMapping[walkerOSGroup];\n if (!gtagParams) return;\n\n const params = Array.isArray(gtagParams) ? gtagParams : [gtagParams];\n const consentValue = granted ? 'granted' : 'denied';\n\n params.forEach((param) => {\n gtagConsent[param] = consentValue;\n });\n });\n\n // Only proceed if we have consent parameters to update\n if (Object.keys(gtagConsent).length === 0) return;\n\n // Always use update after the initial default\n gtag('consent', 'update', gtagConsent);\n },\n};\n\nexport default destinationGtag;\n","import type {\n Mapping as WalkerOSMapping,\n Destination as CoreDestination,\n} from '@walkeros/core';\nimport type { DestinationWeb } from '@walkeros/web-core';\n\n// Extend Window interface with specific dataLayer typing\ninterface WindowWithDataLayer extends Window {\n dataLayer: unknown[];\n gtag?: Gtag.Gtag;\n [key: string]: unknown; // For custom dataLayer names and other properties\n}\n\ndeclare global {\n interface Window {\n gtag?: Gtag.Gtag;\n [key: string]: unknown; // For custom dataLayer names and other properties\n }\n}\n\n// Gtag-specific environment interface\nexport interface Env extends DestinationWeb.Env {\n window: {\n gtag: Gtag.Gtag;\n dataLayer: unknown[];\n };\n document: {\n createElement: (tagName: string) => {\n src: string;\n async?: boolean;\n setAttribute: (name: string, value: string) => void;\n removeAttribute: (name: string) => void;\n };\n head: {\n appendChild: (node: unknown) => void;\n };\n };\n}\n\n// Consent mode configuration\nexport type ConsentMode =\n | false // Disable consent mode\n | true // Use default mapping\n | ConsentMapping; // Custom mapping\n\nexport interface ConsentMapping {\n [walkerOSConsentGroup: string]: string | string[];\n}\n\n// Type bundle\nexport type Types = CoreDestination.Types<Settings, Mapping, Env, InitSettings>;\n\nexport type Destination = DestinationWeb.Destination<Types>;\nexport type Config = DestinationWeb.Config<Types>;\n\n// Base settings interface with data field\nexport interface BaseSettings {\n data?: WalkerOSMapping.Value | WalkerOSMapping.Values;\n}\n\n// Unified settings for all Google tools\nexport interface Settings {\n // Consent mode configuration\n como?: ConsentMode;\n // GA4 settings\n ga4?: GA4Settings;\n // Google Ads settings\n ads?: AdsSettings;\n // GTM settings\n gtm?: GTMSettings;\n}\n\n// InitSettings: user input (all optional)\nexport type InitSettings = Partial<Settings>;\n\n// GA4-specific settings\nexport interface GA4Settings extends BaseSettings {\n measurementId: string;\n debug?: boolean;\n include?: Include;\n pageview?: boolean;\n server_container_url?: string;\n snakeCase?: boolean;\n transport_url?: string;\n}\n\n// Google Ads specific settings\nexport interface AdsSettings extends BaseSettings {\n conversionId: string;\n currency?: string;\n}\n\n// Google Ads specific mapping\nexport interface AdsMapping {\n label?: string; // Conversion label for this specific event\n}\n\n// GTM specific settings\nexport interface GTMSettings extends BaseSettings {\n containerId: string;\n dataLayer?: string;\n domain?: string;\n}\n\n// Unified mapping interface\nexport interface Mapping {\n ga4?: GA4Mapping;\n ads?: AdsMapping;\n gtm?: GTMMapping;\n}\n\n// GA4-specific mapping\nexport interface GA4Mapping {\n include?: Include;\n}\n\n// GTM specific mapping\nexport interface GTMMapping {}\n\nexport type Rule = WalkerOSMapping.Rule<Mapping>;\nexport type Rules = WalkerOSMapping.Rules<Rule>;\nexport type Param = WalkerOSMapping.Value;\n\nexport type Include = Array<\n | 'all'\n | 'context'\n | 'data'\n | 'event'\n | 'globals'\n | 'source'\n | 'user'\n | 'version'\n>;\n\nexport type Parameters = Gtag.ControlParams &\n Gtag.EventParams &\n Gtag.CustomParams;\n"],"mappings":";AAEA,IAAM,gBAAgB,oBAAI,IAAY;AAO/B,SAAS,UACd,IACA,MAAM,gDACNA,YAAqB,WAAW,UAC1B;AAEN,MAAI,cAAc,IAAI,EAAE,EAAG;AAE3B,QAAM,SAASA,UAAS,cAAc,QAAQ;AAC9C,SAAO,MAAM,MAAM;AACnB,EAAAA,UAAS,KAAK,YAAY,MAAM;AAChC,gBAAc,IAAI,EAAE;AACtB;AAEO,SAAS,eAAe,QAAuC;AACpE,QAAM,IAAI;AAGV,IAAE,YAAY,EAAE,aAAa,CAAC;AAG9B,MAAI,CAAC,EAAE,MAAM;AACX,MAAE,OAAO,WAAY;AACnB,MAAC,EAAE,UAAwB,KAAK,SAAS;AAAA,IAC3C;AAAA,EACF;AAEA,SAAO,EAAE;AACX;;;ACjCA,SAAS,cAAc;AAEhB,SAAS,QACd,UACA,YACA,KACA,QACM;AACN,QAAM,EAAE,QAAQ,UAAAC,UAAS,IAAI,OAAO,GAAG;AACvC,QAAM,EAAE,eAAe,eAAe,sBAAsB,SAAS,IACnE;AAEF,MAAI,CAAC,cAAe,QAAO,MAAM,2CAA2C;AAG5E,MAAI,WAAY,WAAU,eAAe,QAAWA,SAAoB;AAGxE,iBAAe,MAAgB;AAE/B,QAAM,eAAmC,CAAC;AAG1C,MAAI,cAAe,cAAa,gBAAgB;AAGhD,MAAI;AACF,iBAAa,uBAAuB;AAGtC,MAAI,aAAa,MAAO,cAAa,iBAAiB;AAEtD,QAAM,OAAO,OAAO;AACpB,OAAK,MAAM,oBAAI,KAAK,CAAC;AAGrB,OAAK,UAAU,eAAe,YAAY;AAC5C;A;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACtCO,SAAS,iBACd,OACA,SACY;AACZ,QAAM,SAAqB,CAAC;AAG5B,MAAI,QAAQ,SAAS,KAAK;AACxB,cAAU;AAAA,MACR;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAEF,UAAQ,QAAQ,CAAC,cAAc;AAC7B,QAAI,QAAQ,MAAM,SAA8C,KAAK,CAAC;AAGtE,QAAI,aAAa;AACf,cAAQ;AAAA,QACN,IAAI,MAAM;AAAA,QACV,QAAQ,MAAM;AAAA,QACd,SAAS,MAAM;AAAA,QACf,QAAQ,MAAM;AAAA,QACd,QAAQ,MAAM;AAAA,QACd,OAAO,MAAM;AAAA,QACb,OAAO,MAAM;AAAA,MACf;AAEF,WAAO,QAAQ,KAAK,EAAE,QAAQ,CAAC,CAAC,KAAK,GAAG,MAAM;AAE5C,UAAI,aAAa,UAAW,OAAO,IAAmC,CAAC;AAEvE,aAAO,GAAG,SAAS,IAAI,GAAG,EAAE,IAAI;AAAA,IAClC,CAAC;AAAA,EACH,CAAC;AAED,SAAO;AACT;;;ACxCO,SAAS,mBACd,WACA,YAAY,MACJ;AACR,MAAI,CAAC,UAAW,QAAO;AAEvB,SAAO,UAAU,QAAQ,QAAQ,GAAG,EAAE,YAAY;AACpD;AAEA,eAAsB,QACpB,OACA,MACA,QACA,cACA,WAC6B;AAC7B,QAAM,WAAW,EAAS,IAAI,IAAI,OAAO,CAAC;AAG1C,QAAM,mBAAmB,OAAO,OAC5B,MAAM,GAAgB,OAAO,OAAO,MAAM,EAAE,UAAU,CAAC,IACvD,CAAC;AAGL,QAAM,kBAAiB,6CAAc,QACjC,MAAM,GAAgB,OAAO,aAAa,MAAM,EAAE,UAAU,CAAC,IAC7D,CAAC;AAGL,QAAM,aAAa,EAAS,gBAAgB,IAAI,mBAAmB,CAAC;AACpE,QAAM,WAAW,EAAS,cAAc,IAAI,iBAAiB,CAAC;AAG9D,SAAO,EAAO,EAAO,UAAU,UAAU,GAAG,QAAQ;AACtD;;;ACjCA,SAAS,UAAAC,eAAc;AAEhB,SAAS,aACd,OACA,UACA,UAAsB,CAAC,GACvB,MACA,KACA,QACM;AACN,QAAM,EAAE,OAAO,IAAIA,QAAO,GAAG;AAE7B,MAAI,CAAC,SAAS;AACZ,WAAO,MAAM,2CAA2C;AAE1D,QAAM,YAAY,EAAS,IAAI,IAAI,OAAO,CAAC;AAE3C,QAAM,gBAAgB;AAAA,IACpB;AAAA;AAAA,IAEA,QAAQ,WAAW,SAAS,WAAW,CAAC,MAAM;AAAA,EAChD;AAEA,QAAM,cAA0B;AAAA,IAC9B,GAAG;AAAA,IACH,GAAG;AAAA,EACL;AAGA,MAAI,YAAY,MAAM;AACtB,MAAI,SAAS,cAAc,OAAO;AAEhC,gBAAY,mBAAmB,SAAS;AAAA,EAC1C;AAGA,cAAY,UAAU,SAAS;AAG/B,MAAI,SAAS,MAAO,aAAY,aAAa;AAE7C,QAAM,OAAO,OAAO;AACpB,OAAK,SAAS,WAAW,WAAW;AACtC;;;AC7CA,SAAS,UAAAC,eAAc;AAEhB,SAAS,QACd,UACA,YACA,KACA,QACM;AACN,QAAM,EAAE,QAAQ,UAAAC,UAAS,IAAID,QAAO,GAAG;AACvC,QAAM,EAAE,aAAa,IAAI;AAEzB,MAAI,CAAC,aAAc,QAAO,MAAM,0CAA0C;AAG1E,MAAI,CAAC,SAAS,SAAU,UAAS,WAAW;AAE5C,MAAI,WAAY,WAAU,cAAc,QAAWC,SAAoB;AAGvE,iBAAe,MAAgB;AAE/B,QAAM,OAAO,OAAO;AACpB,OAAK,MAAM,oBAAI,KAAK,CAAC;AAGrB,OAAK,UAAU,YAAY;AAC7B;;;AC1BA,SAAS,UAAAC,eAAc;AAEhB,SAAS,aACd,OACA,UACA,UAAsB,CAAC,GACvB,MACA,aACA,KACA,QACM;AACN,QAAM,EAAE,cAAc,SAAS,IAAI;AACnC,QAAM,YAAY,EAAS,IAAI,IAAI,OAAO,CAAC;AAG3C,QAAM,kBAAkB,QAAQ,SAAS;AACzC,MAAI,CAAC,gBAAiB,QAAO,MAAM,kCAAkC;AAErE,QAAM,SAA4B;AAAA,IAChC,SAAS,GAAG,YAAY,IAAI,eAAe;AAAA,IAC3C,UAAU,YAAY;AAAA,IACtB,GAAG;AAAA,EACL;AAEA,QAAM,EAAE,OAAO,IAAIA,QAAO,GAAG;AAC7B,QAAM,OAAO,OAAO;AACpB,OAAK,SAAS,cAAc,MAAM;AACpC;;;AC5BA,SAAS,UAAAC,eAAc;AAEvB,IAAM,mBAAmB;AACzB,IAAM,gBAAgB;AAEf,SAAS,QACd,UACA,YACA,KACA,SACM;AACN,QAAM,EAAE,QAAQ,UAAAC,UAAS,IAAID,QAAO,GAAG;AACvC,QAAM,EAAE,aAAa,WAAW,OAAO,IAAI;AAC3C,QAAM,gBAAgB,aAAa;AAGnC,MAAI,kBAAkB,kBAAkB;AACtC,WAAO,YAAY,OAAO,aAAa,CAAC;AAAA,EAC1C,OAAO;AACL,WAAO,aAAa,IAAK,OAAO,aAAa,KAAmB,CAAC;AAAA,EACnE;AAGA,QAAM,iBAAiB,OAAO,aAAa;AAE3C,iBAAe,KAAK;AAAA,IAClB,cAAa,oBAAI,KAAK,GAAE,QAAQ;AAAA,IAChC,OAAO;AAAA,EACT,CAAC;AAGD,MAAI,cAAc;AAChB,IAAAE;AAAA,MACE;AAAA,MACA,UAAU;AAAA,MACV;AAAA,MACAD;AAAA,IACF;AACJ;AAEA,SAASC,WACP,aACA,KACA,eACAD,YAAqB,WAAW,UAChC;AACA,QAAM,KAAK,iBAAiB,mBAAmB,QAAQ,gBAAgB;AACvE,QAAM,SAASA,UAAS,cAAc,QAAQ;AAC9C,SAAO,MAAM,MAAM,cAAc;AACjC,EAAAA,UAAS,KAAK,YAAY,MAAM;AAClC;;;ACjDA,SAAS,UAAAE,eAAc;AAEhB,SAAS,aACd,OACA,UACA,UAAsB,CAAC,GACvB,MACA,KACA,SACM;AACN,QAAM,EAAE,OAAO,IAAIA,QAAO,GAAG;AAC7B,QAAM,MAAM,EAAE,OAAO,MAAM,KAAK;AAEhC,EAAC,OAAO,UAAwB,KAAK;AAAA,IACnC,GAAG;AAAA,IACH,GAAI,EAAS,IAAI,IAAI,OAAO;AAAA,EAC9B,CAAC;AACH;;;ACdA,SAAS,UAAAC,eAAc;;;ACPvB;;;ADaA,IAAI,oBAAoB;AAGjB,SAAS,oBAA0B;AACxC,sBAAoB;AACtB;AAGA,IAAM,0BAA0C;AAAA,EAC9C,WAAW,CAAC,cAAc,gBAAgB,oBAAoB;AAAA,EAC9D,YAAY,CAAC,mBAAmB;AAClC;AAEO,IAAM,kBAA+B;AAAA,EAC1C,MAAM;AAAA,EAEN,QAAQ,EAAE,UAAU,CAAC,EAAE;AAAA,EAEvB,KAAK,EAAE,QAAQ,KAAK,OAAO,GAAG;AAC5B,UAAM,EAAE,WAAW,CAAC,GAAwB,WAAW,IAAI;AAC3D,UAAM,EAAE,KAAK,KAAK,IAAI,IAAI;AAG1B,QAAI,EAAC,2BAAK,kBAAiB,EAAC,2BAAK,iBAAgB,EAAC,2BAAK,cAAa;AAClE,aAAO;AAAA,QACL;AAAA,MACF;AAAA,IACF;AAGA,QAAI,2BAAK,eAAe;AACtB,cAAQ,KAAK,YAAY,KAAK,MAAM;AAAA,IACtC;AAGA,QAAI,2BAAK,cAAc;AACrB,cAAQ,KAAK,YAAY,KAAK,MAAM;AAAA,IACtC;AAGA,QAAI,2BAAK,aAAa;AACpB,cAAQ,KAAK,YAAY,KAAK,MAAM;AAAA,IACtC;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,KAAK,OAAO,EAAE,QAAQ,UAAU,CAAC,GAAG,MAAM,KAAK,WAAW,OAAO,GAAG;AACxE,UAAM,EAAE,WAAW,CAAC,EAAE,IAAI;AAC1B,UAAM,EAAE,KAAK,KAAK,IAAI,IAAI;AAC1B,UAAM,eAAe,QAAQ,YAAY,CAAC;AAG1C,UAAM,UAAU,MAAM;AAAA,MACpB;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AACA,UAAM,UAAU,MAAM;AAAA,MACpB;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AACA,UAAM,UAAU,MAAM;AAAA,MACpB;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAGA,QAAI,2BAAK,eAAe;AACtB,mBAAa,OAAO,KAAK,aAAa,KAAK,SAAS,KAAK,MAAM;AAAA,IACjE;AAIA,SAAI,2BAAK,iBAAgB,QAAQ,MAAM;AACrC;AAAA,QACE;AAAA,QACA;AAAA,QACA,aAAa;AAAA,QACb;AAAA,QACA,QAAQ;AAAA,QACR;AAAA,QACA;AAAA,MACF;AAAA,IACF;AAGA,QAAI,2BAAK,aAAa;AACpB,mBAAa,OAAO,KAAK,aAAa,KAAK,SAAS,KAAK,MAAM;AAAA,IACjE;AAAA,EACF;AAAA,EAEA,GAAG,MAAM,SAAS;AAjHpB;AAmHI,QAAI,SAAS,aAAa,CAAC,QAAQ,KAAM;AAEzC,UAAM,UAAU,QAAQ;AACxB,UAAM,aAAY,aAAQ,WAAR,mBAAgB,aAAY,CAAC;AAC/C,UAAM,EAAE,OAAO,KAAK,IAAI;AAGxB,QAAI,CAAC,KAAM;AAGX,UAAM,EAAE,OAAO,IAAIC,QAAO,QAAQ,GAAG;AACrC,UAAM,OAAO,eAAe,MAAgB;AAC5C,QAAI,CAAC,KAAM;AAGX,UAAM,iBACJ,SAAS,OAAO,0BAA0B;AAG5C,QAAI,CAAC,mBAAmB;AAEtB,YAAM,gBAAgB,oBAAI,IAAY;AACtC,aAAO,OAAO,cAAc,EAAE,QAAQ,CAAC,WAAW;AAChD,cAAM,aAAa,MAAM,QAAQ,MAAM,IAAI,SAAS,CAAC,MAAM;AAC3D,mBAAW,QAAQ,CAAC,UAAU,cAAc,IAAI,KAAK,CAAC;AAAA,MACxD,CAAC;AAGD,UAAI,cAAc,OAAO,GAAG;AAC1B,cAAM,iBAA2C,CAAC;AAClD,sBAAc,QAAQ,CAAC,UAAU;AAC/B,yBAAe,KAAK,IAAI;AAAA,QAC1B,CAAC;AAGD,aAAK,WAAW,WAAW,cAAc;AAAA,MAC3C;AAEA,0BAAoB;AAAA,IACtB;AAGA,UAAM,cAAoD,CAAC;AAG3D,WAAO,QAAQ,OAAO,EAAE,QAAQ,CAAC,CAAC,eAAe,OAAO,MAAM;AAC5D,YAAM,aAAa,eAAe,aAAa;AAC/C,UAAI,CAAC,WAAY;AAEjB,YAAM,SAAS,MAAM,QAAQ,UAAU,IAAI,aAAa,CAAC,UAAU;AACnE,YAAM,eAAe,UAAU,YAAY;AAE3C,aAAO,QAAQ,CAAC,UAAU;AACxB,oBAAY,KAAK,IAAI;AAAA,MACvB,CAAC;AAAA,IACH,CAAC;AAGD,QAAI,OAAO,KAAK,WAAW,EAAE,WAAW,EAAG;AAG3C,SAAK,WAAW,UAAU,WAAW;AAAA,EACvC;AACF;AAEA,IAAO,gBAAQ;","names":["document","document","getEnv","getEnv","document","getEnv","getEnv","document","addScript","getEnv","getEnv","getEnv"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@walkeros/web-destination-gtag",
3
3
  "description": "Unified Google destination for walkerOS (GA4, Ads, GTM)",
4
- "version": "0.4.2",
4
+ "version": "0.5.0",
5
5
  "license": "MIT",
6
6
  "main": "./dist/index.js",
7
7
  "module": "./dist/index.mjs",
@@ -35,7 +35,7 @@
35
35
  "update": "npx npm-check-updates -u && npm update"
36
36
  },
37
37
  "dependencies": {
38
- "@walkeros/web-core": "0.4.2"
38
+ "@walkeros/web-core": "0.5.0"
39
39
  },
40
40
  "repository": {
41
41
  "url": "git+https://github.com/elbwalker/walkerOS.git",