diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.getalltrackers.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.getalltrackers.md
new file mode 100644
index 000000000..dd12938e2
--- /dev/null
+++ b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.getalltrackers.md
@@ -0,0 +1,19 @@
+
+
+[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [getAllTrackers](./react-native-tracker.getalltrackers.md)
+
+## getAllTrackers() function
+
+Retrieves all initialized trackers
+
+Signature:
+
+```typescript
+export declare function getAllTrackers(): ReactNativeTracker[];
+```
+Returns:
+
+[ReactNativeTracker](./react-native-tracker.reactnativetracker.md)\[\]
+
+All initialized trackers
+
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.getwebviewcallback.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.getwebviewcallback.md
new file mode 100644
index 000000000..1fb83961f
--- /dev/null
+++ b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.getwebviewcallback.md
@@ -0,0 +1,25 @@
+
+
+[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [getWebViewCallback](./react-native-tracker.getwebviewcallback.md)
+
+## getWebViewCallback() function
+
+Enables tracking events from apps rendered in react-native-webview components. The apps need to use the Snowplow WebView tracker to track the events.
+
+To subscribe for the events, set the `onMessage` attribute: ``
+
+Signature:
+
+```typescript
+export declare function getWebViewCallback(): (message: {
+ nativeEvent: {
+ data: string;
+ };
+}) => void;
+```
+Returns:
+
+(message: { nativeEvent: { data: string; }; }) => void
+
+Callback to subscribe for events from Web views tracked using the Snowplow WebView tracker.
+
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.md
index a4a06599b..ceca4c8d4 100644
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.md
+++ b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.md
@@ -15,7 +15,9 @@
| Function | Description |
| --- | --- |
+| [getAllTrackers()](./react-native-tracker.getalltrackers.md) | Retrieves all initialized trackers |
| [getTracker(trackerNamespace)](./react-native-tracker.gettracker.md) | Retrieves an initialized tracker given its namespace |
+| [getWebViewCallback()](./react-native-tracker.getwebviewcallback.md) | Enables tracking events from apps rendered in react-native-webview components. The apps need to use the Snowplow WebView tracker to track the events.To subscribe for the events, set the onMessage
attribute: <WebView onMessage={getWebViewCallback()} ... />
|
| [newTracker(configuration)](./react-native-tracker.newtracker.md) | Creates a new tracker instance with the given configuration |
| [removeAllTrackers()](./react-native-tracker.removealltrackers.md) | Removes all initialized trackers |
| [removeTracker(trackerNamespace)](./react-native-tracker.removetracker.md) | Removes a tracker given its namespace |
@@ -47,7 +49,6 @@
| [RuleSet](./react-native-tracker.ruleset.md) | A ruleset has accept or reject properties that contain rules for matching Iglu schema URIs |
| [SessionConfiguration](./react-native-tracker.sessionconfiguration.md) | Configuration for session tracking |
| [SessionState](./react-native-tracker.sessionstate.md) | Current session state that is tracked in events. |
-| [StructuredEvent](./react-native-tracker.structuredevent.md) | A Structured Event A classic style of event tracking, allows for easier movement between analytics systems. A loosely typed event, creating a Self Describing event is preferred, but useful for interoperability. |
| [SubjectConfiguration](./react-native-tracker.subjectconfiguration.md) | Configuration of subject properties tracked with events |
| [TrackerConfiguration](./react-native-tracker.trackerconfiguration.md) | The configuration object for initialising the tracker |
| [TrackerCore](./react-native-tracker.trackercore.md) | Export interface containing all Core functions |
@@ -85,7 +86,8 @@
| [ScreenSize](./react-native-tracker.screensize.md) | Screen size in pixels |
| [ScreenViewProps](./react-native-tracker.screenviewprops.md) | ScreenView event properties schema: iglu:com.snowplowanalytics.mobile/screen\_view/jsonschema/1-0-0 |
| [ScrollChangedProps](./react-native-tracker.scrollchangedprops.md) | Event tracked when a scroll view's scroll position changes. If screen engagement tracking is enabled, the scroll changed events will be aggregated into a screen_summary
entity.Schema: iglu:com.snowplowanalytics.mobile/scroll_changed/jsonschema/1-0-0
|
-| [SelfDescribingJson](./react-native-tracker.selfdescribingjson.md) | Export interface for any Self-Describing JSON such as context or Self Describing events |
+| [SelfDescribing](./react-native-tracker.selfdescribing.md) | Interface for any self-describing JSON such as context entities or self-describing events |
+| [StructuredProps](./react-native-tracker.structuredprops.md) | Properties for a structured event. A classic style of event tracking, allows for easier movement between analytics systems. Self-describing events are preferred for their schema validation. |
| [Timestamp](./react-native-tracker.timestamp.md) | Algebraic datatype representing possible timestamp type choice |
| [TimingProps](./react-native-tracker.timingprops.md) | Timing event properties |
| [Trigger](./react-native-tracker.trigger.md) | Trigger for MessageNotification event |
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.reactnativetracker.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.reactnativetracker.md
index 720dc9ee8..b0068bf18 100644
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.reactnativetracker.md
+++ b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.reactnativetracker.md
@@ -11,11 +11,11 @@ The ReactNativeTracker type
```typescript
export declare type ReactNativeTracker = {
namespace: string;
- readonly trackSelfDescribingEvent: = Record>(argmap: SelfDescribingJson, contexts?: EventContext[]) => void;
+ readonly trackSelfDescribingEvent: = Record>(argmap: SelfDescribing, contexts?: EventContext[]) => void;
readonly trackScreenViewEvent: (argmap: ScreenViewProps, contexts?: EventContext[]) => void;
readonly trackScrollChangedEvent: (argmap: ScrollChangedProps, contexts?: EventContext[]) => void;
readonly trackListItemViewEvent: (argmap: ListItemViewProps, contexts?: EventContext[]) => void;
- readonly trackStructuredEvent: (argmap: StructuredEvent, contexts?: EventContext[]) => void;
+ readonly trackStructuredEvent: (argmap: StructuredProps, contexts?: EventContext[]) => void;
readonly trackPageViewEvent: (argmap: PageViewEvent, contexts?: EventContext[]) => void;
readonly trackTimingEvent: (argmap: TimingProps, contexts?: EventContext[]) => void;
readonly trackDeepLinkReceivedEvent: (argmap: DeepLinkReceivedProps, contexts?: EventContext[]) => void;
@@ -50,5 +50,5 @@ export declare type ReactNativeTracker = {
readonly refreshPlatformContext: () => Promise;
};
```
-References: [EventContext](./react-native-tracker.eventcontext.md), [ScreenViewProps](./react-native-tracker.screenviewprops.md), [ScrollChangedProps](./react-native-tracker.scrollchangedprops.md), [ListItemViewProps](./react-native-tracker.listitemviewprops.md), [TimingProps](./react-native-tracker.timingprops.md), [DeepLinkReceivedProps](./react-native-tracker.deeplinkreceivedprops.md), [MessageNotificationProps](./react-native-tracker.messagenotificationprops.md), [ScreenSize](./react-native-tracker.screensize.md), [SubjectConfiguration](./react-native-tracker.subjectconfiguration.md), [SessionState](./react-native-tracker.sessionstate.md)
+References: [SelfDescribing](./react-native-tracker.selfdescribing.md), [EventContext](./react-native-tracker.eventcontext.md), [ScreenViewProps](./react-native-tracker.screenviewprops.md), [ScrollChangedProps](./react-native-tracker.scrollchangedprops.md), [ListItemViewProps](./react-native-tracker.listitemviewprops.md), [StructuredProps](./react-native-tracker.structuredprops.md), [TimingProps](./react-native-tracker.timingprops.md), [DeepLinkReceivedProps](./react-native-tracker.deeplinkreceivedprops.md), [MessageNotificationProps](./react-native-tracker.messagenotificationprops.md), [ScreenSize](./react-native-tracker.screensize.md), [SubjectConfiguration](./react-native-tracker.subjectconfiguration.md), [SessionState](./react-native-tracker.sessionstate.md)
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.selfdescribing.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.selfdescribing.md
new file mode 100644
index 000000000..1efbed825
--- /dev/null
+++ b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.selfdescribing.md
@@ -0,0 +1,13 @@
+
+
+[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [SelfDescribing](./react-native-tracker.selfdescribing.md)
+
+## SelfDescribing type
+
+Interface for any self-describing JSON such as context entities or self-describing events
+
+Signature:
+
+```typescript
+export declare type SelfDescribing> = SelfDescribingJson;
+```
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.selfdescribingjson.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.selfdescribingjson.md
deleted file mode 100644
index 0a32034ac..000000000
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.selfdescribingjson.md
+++ /dev/null
@@ -1,16 +0,0 @@
-
-
-[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [SelfDescribingJson](./react-native-tracker.selfdescribingjson.md)
-
-## SelfDescribingJson type
-
-Export interface for any Self-Describing JSON such as context or Self Describing events
-
-Signature:
-
-```typescript
-type SelfDescribingJson> = {
- schema: string;
- data: T extends any[] ? never : T extends {} ? T : never;
-};
-```
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.action.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.action.md
deleted file mode 100644
index a1e9ec061..000000000
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.action.md
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [StructuredEvent](./react-native-tracker.structuredevent.md) > [action](./react-native-tracker.structuredevent.action.md)
-
-## StructuredEvent.action property
-
-Signature:
-
-```typescript
-action: string;
-```
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.category.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.category.md
deleted file mode 100644
index 0e1a2cbb6..000000000
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.category.md
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [StructuredEvent](./react-native-tracker.structuredevent.md) > [category](./react-native-tracker.structuredevent.category.md)
-
-## StructuredEvent.category property
-
-Signature:
-
-```typescript
-category: string;
-```
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.label.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.label.md
deleted file mode 100644
index 2d8731a8d..000000000
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.label.md
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [StructuredEvent](./react-native-tracker.structuredevent.md) > [label](./react-native-tracker.structuredevent.label.md)
-
-## StructuredEvent.label property
-
-Signature:
-
-```typescript
-label?: string;
-```
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.md
deleted file mode 100644
index 0ce7bdde8..000000000
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.md
+++ /dev/null
@@ -1,24 +0,0 @@
-
-
-[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [StructuredEvent](./react-native-tracker.structuredevent.md)
-
-## StructuredEvent interface
-
-A Structured Event A classic style of event tracking, allows for easier movement between analytics systems. A loosely typed event, creating a Self Describing event is preferred, but useful for interoperability.
-
-Signature:
-
-```typescript
-interface StructuredEvent
-```
-
-## Properties
-
-| Property | Type | Description |
-| --- | --- | --- |
-| [action](./react-native-tracker.structuredevent.action.md) | string | |
-| [category](./react-native-tracker.structuredevent.category.md) | string | |
-| [label?](./react-native-tracker.structuredevent.label.md) | string | (Optional) |
-| [property?](./react-native-tracker.structuredevent.property.md) | string | (Optional) |
-| [value?](./react-native-tracker.structuredevent.value.md) | number | (Optional) |
-
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.property.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.property.md
deleted file mode 100644
index 1015c9a8e..000000000
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.property.md
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [StructuredEvent](./react-native-tracker.structuredevent.md) > [property](./react-native-tracker.structuredevent.property.md)
-
-## StructuredEvent.property property
-
-Signature:
-
-```typescript
-property?: string;
-```
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.value.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.value.md
deleted file mode 100644
index e7673c4ca..000000000
--- a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredevent.value.md
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [StructuredEvent](./react-native-tracker.structuredevent.md) > [value](./react-native-tracker.structuredevent.value.md)
-
-## StructuredEvent.value property
-
-Signature:
-
-```typescript
-value?: number;
-```
diff --git a/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredprops.md b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredprops.md
new file mode 100644
index 000000000..0eded0b1e
--- /dev/null
+++ b/api-docs/docs/react-native-tracker/markdown/react-native-tracker.structuredprops.md
@@ -0,0 +1,13 @@
+
+
+[Home](./index.md) > [@snowplow/react-native-tracker](./react-native-tracker.md) > [StructuredProps](./react-native-tracker.structuredprops.md)
+
+## StructuredProps type
+
+Properties for a structured event. A classic style of event tracking, allows for easier movement between analytics systems. Self-describing events are preferred for their schema validation.
+
+Signature:
+
+```typescript
+export declare type StructuredProps = StructuredEvent;
+```
diff --git a/api-docs/docs/react-native-tracker/react-native-tracker.api.md b/api-docs/docs/react-native-tracker/react-native-tracker.api.md
index 452ad94c4..0914ed86a 100644
--- a/api-docs/docs/react-native-tracker/react-native-tracker.api.md
+++ b/api-docs/docs/react-native-tracker/react-native-tracker.api.md
@@ -24,6 +24,8 @@ export type ConditionalContextProvider = FilterProvider | RuleSetProvider;
// @public
export type ContextFilter = (args?: ContextEvent) => boolean;
+// Warning: (ae-forgotten-export) The symbol "SelfDescribingJson" needs to be exported by the entry point index.d.ts
+//
// @public
export type ContextGenerator = (args?: ContextEvent) => SelfDescribingJson | SelfDescribingJson[] | undefined;
@@ -197,9 +199,19 @@ export interface FormFocusOrChangeEvent {
value: string | null;
}
+// @public
+export function getAllTrackers(): ReactNativeTracker[];
+
// @public
export function getTracker(trackerNamespace: string): ReactNativeTracker | undefined;
+// @public
+export function getWebViewCallback(): (message: {
+ nativeEvent: {
+ data: string;
+ };
+}) => void;
+
// @public
export type JsonProcessor = (payloadBuilder: PayloadBuilder, jsonForProcessing: EventJson, contextEntitiesForProcessing: SelfDescribingJson[]) => void;
@@ -354,11 +366,11 @@ export interface PlatformContextRetriever {
// @public
export type ReactNativeTracker = {
namespace: string;
- readonly trackSelfDescribingEvent: = Record>(argmap: SelfDescribingJson, contexts?: EventContext[]) => void;
+ readonly trackSelfDescribingEvent: = Record>(argmap: SelfDescribing, contexts?: EventContext[]) => void;
readonly trackScreenViewEvent: (argmap: ScreenViewProps, contexts?: EventContext[]) => void;
readonly trackScrollChangedEvent: (argmap: ScrollChangedProps, contexts?: EventContext[]) => void;
readonly trackListItemViewEvent: (argmap: ListItemViewProps, contexts?: EventContext[]) => void;
- readonly trackStructuredEvent: (argmap: StructuredEvent, contexts?: EventContext[]) => void;
+ readonly trackStructuredEvent: (argmap: StructuredProps, contexts?: EventContext[]) => void;
readonly trackPageViewEvent: (argmap: PageViewEvent, contexts?: EventContext[]) => void;
readonly trackTimingEvent: (argmap: TimingProps, contexts?: EventContext[]) => void;
readonly trackDeepLinkReceivedEvent: (argmap: DeepLinkReceivedProps, contexts?: EventContext[]) => void;
@@ -446,10 +458,7 @@ export type ScrollChangedProps = {
};
// @public
-export type SelfDescribingJson> = {
- schema: string;
- data: T extends any[] ? never : T extends {} ? T : never;
-};
+export type SelfDescribing> = SelfDescribingJson;
// @public
export interface SessionConfiguration {
@@ -469,19 +478,10 @@ export interface SessionState {
userId: string;
}
+// Warning: (ae-forgotten-export) The symbol "StructuredEvent" needs to be exported by the entry point index.d.ts
+//
// @public
-export interface StructuredEvent {
- // (undocumented)
- action: string;
- // (undocumented)
- category: string;
- // (undocumented)
- label?: string;
- // (undocumented)
- property?: string;
- // (undocumented)
- value?: number;
-}
+export type StructuredProps = StructuredEvent;
// @public
export interface SubjectConfiguration {
diff --git a/common/changes/@snowplow/react-native-tracker/issue-web_view_rn_2024-12-10-15-26.json b/common/changes/@snowplow/react-native-tracker/issue-web_view_rn_2024-12-10-15-26.json
new file mode 100644
index 000000000..3dfabca89
--- /dev/null
+++ b/common/changes/@snowplow/react-native-tracker/issue-web_view_rn_2024-12-10-15-26.json
@@ -0,0 +1,10 @@
+{
+ "changes": [
+ {
+ "packageName": "@snowplow/react-native-tracker",
+ "comment": "Add WebView tracker integration to the React Native tracker (#1399)",
+ "type": "none"
+ }
+ ],
+ "packageName": "@snowplow/react-native-tracker"
+}
\ No newline at end of file
diff --git a/trackers/react-native-tracker/src/events.ts b/trackers/react-native-tracker/src/events.ts
index 1195b83f2..02e1ef523 100644
--- a/trackers/react-native-tracker/src/events.ts
+++ b/trackers/react-native-tracker/src/events.ts
@@ -4,10 +4,9 @@ import {
buildStructEvent,
PageViewEvent,
SelfDescribingJson,
- StructuredEvent,
TrackerCore,
} from '@snowplow/tracker-core';
-import { EventContext, MessageNotificationProps, TimingProps } from './types';
+import { EventContext, MessageNotificationProps, StructuredProps, TimingProps } from './types';
export function newTrackEventFunctions(core: TrackerCore) {
const trackSelfDescribingEvent = = Record>(
@@ -17,7 +16,7 @@ export function newTrackEventFunctions(core: TrackerCore) {
core.track(buildSelfDescribingEvent({ event: argmap }), contexts);
};
- const trackStructuredEvent = (argmap: StructuredEvent, contexts?: EventContext[]) => {
+ const trackStructuredEvent = (argmap: StructuredProps, contexts?: EventContext[]) => {
return core.track(buildStructEvent(argmap), contexts)?.eid;
};
diff --git a/trackers/react-native-tracker/src/index.ts b/trackers/react-native-tracker/src/index.ts
index 6c29689cb..a4cdb5590 100644
--- a/trackers/react-native-tracker/src/index.ts
+++ b/trackers/react-native-tracker/src/index.ts
@@ -2,4 +2,5 @@
import 'react-native-get-random-values';
export * from './types';
-export * from './tracker';
+export { newTracker, getTracker, getAllTrackers, removeTracker, removeAllTrackers } from './tracker';
+export * from './web_view_interface';
diff --git a/trackers/react-native-tracker/src/tracker.ts b/trackers/react-native-tracker/src/tracker.ts
index 86d375e38..cc9d9e958 100644
--- a/trackers/react-native-tracker/src/tracker.ts
+++ b/trackers/react-native-tracker/src/tracker.ts
@@ -160,6 +160,31 @@ export function getTracker(trackerNamespace: string): ReactNativeTracker | undef
return initializedTrackers[trackerNamespace]?.tracker;
}
+/**
+ * Retrieves all initialized trackers
+ * @returns All initialized trackers
+ */
+export function getAllTrackers(): ReactNativeTracker[] {
+ return Object.values(initializedTrackers).map(({ tracker }) => tracker);
+}
+
+/**
+ * Internal function to retrieve the tracker core given its namespace
+ * @param trackerNamespace - Tracker namespace
+ * @returns Tracker core if exists
+ */
+export function getTrackerCore(trackerNamespace: string): TrackerCore | undefined {
+ return initializedTrackers[trackerNamespace]?.core;
+}
+
+/**
+ * Internal function to retrieve all initialized tracker cores
+ * @returns All initialized tracker cores
+ */
+export function getAllTrackerCores(): TrackerCore[] {
+ return Object.values(initializedTrackers).map(({ core }) => core);
+}
+
/**
* Removes a tracker given its namespace
*
diff --git a/trackers/react-native-tracker/src/types.ts b/trackers/react-native-tracker/src/types.ts
index e078d3047..e2d256bc8 100755
--- a/trackers/react-native-tracker/src/types.ts
+++ b/trackers/react-native-tracker/src/types.ts
@@ -622,6 +622,19 @@ export interface SessionState {
firstEventTimestamp?: string;
}
+/**
+ * Properties for a structured event.
+ * A classic style of event tracking, allows for easier movement between analytics systems.
+ * Self-describing events are preferred for their schema validation.
+ */
+export type StructuredProps = StructuredEvent;
+
+/**
+ * Interface for any self-describing JSON such as context entities or self-describing events
+ * @typeParam T - The type of the data object within a SelfDescribingJson
+ */
+export type SelfDescribing> = SelfDescribingJson;
+
/**
* The ReactNativeTracker type
*/
@@ -638,7 +651,7 @@ export type ReactNativeTracker = {
* @typeParam TData - The type of the data object within the SelfDescribing object
*/
readonly trackSelfDescribingEvent: = Record>(
- argmap: SelfDescribingJson,
+ argmap: SelfDescribing,
contexts?: EventContext[]
) => void;
@@ -672,7 +685,7 @@ export type ReactNativeTracker = {
* @param argmap - The structured event properties
* @param contexts - The array of event contexts
*/
- readonly trackStructuredEvent: (argmap: StructuredEvent, contexts?: EventContext[]) => void;
+ readonly trackStructuredEvent: (argmap: StructuredProps, contexts?: EventContext[]) => void;
/**
* Tracks a page-view event
@@ -886,9 +899,7 @@ export type ReactNativeTracker = {
export {
version,
PageViewEvent,
- StructuredEvent,
FormFocusOrChangeEvent,
- SelfDescribingJson,
Timestamp,
PayloadBuilder,
Payload,
diff --git a/trackers/react-native-tracker/src/web_view_interface.ts b/trackers/react-native-tracker/src/web_view_interface.ts
new file mode 100644
index 000000000..de6d9d1c5
--- /dev/null
+++ b/trackers/react-native-tracker/src/web_view_interface.ts
@@ -0,0 +1,215 @@
+'use strict';
+
+import { buildSelfDescribingEvent, payloadBuilder } from '@snowplow/tracker-core';
+import { getAllTrackerCores, getAllTrackers, getTracker, getTrackerCore } from './tracker';
+import type {
+ ScreenViewProps,
+ SelfDescribing,
+ StructuredProps,
+ ReactNativeTracker,
+ TrackerCore,
+ PayloadBuilder,
+ Payload,
+} from './types';
+
+/**
+ * Internal event type for events with payload properties tracked using the WebView tracker.
+ */
+interface WebViewEvent {
+ selfDescribingEventData?: SelfDescribing;
+ eventName?: string;
+ trackerVersion?: string;
+ useragent?: string;
+ pageUrl?: string;
+ pageTitle?: string;
+ referrer?: string;
+ category?: string;
+ action?: string;
+ label?: string;
+ property?: string;
+ value?: number;
+ pingXOffsetMin?: number;
+ pingXOffsetMax?: number;
+ pingYOffsetMin?: number;
+ pingYOffsetMax?: number;
+}
+
+/**
+ * Internal event type for page views tracked using the WebView tracker.
+ */
+interface WebViewPageViewEvent {
+ title?: string | null;
+ url?: string;
+ referrer?: string;
+}
+
+/**
+ * Internal type exchanged in messages received from the WebView tracker in Web views through the web view callback.
+ */
+type WebViewMessage = {
+ command: 'trackSelfDescribingEvent' | 'trackStructEvent' | 'trackPageView' | 'trackScreenView' | 'trackWebViewEvent';
+ event: StructuredProps | SelfDescribing | ScreenViewProps | WebViewPageViewEvent | WebViewEvent;
+ context?: Array | null;
+ trackers?: Array;
+};
+
+function forEachTracker(trackers: Array | undefined, iterator: (tracker: ReactNativeTracker) => void): void {
+ if (trackers && trackers.length > 0) {
+ trackers
+ .map(getTracker)
+ .filter((t) => t !== undefined)
+ .map((t) => t!)
+ .forEach(iterator);
+ } else {
+ getAllTrackers().forEach(iterator);
+ }
+}
+
+function forEachTrackerCore(trackers: Array | undefined, iterator: (tracker: TrackerCore) => void): void {
+ if (trackers && trackers.length > 0) {
+ trackers
+ .map(getTrackerCore)
+ .filter((t) => t !== undefined)
+ .map((t) => t!)
+ .forEach(iterator);
+ } else {
+ getAllTrackerCores().forEach(iterator);
+ }
+}
+
+/**
+ * Wrapper around the PayloadBuilder that disables overriding property values.
+ * This is to prevent the tracker overriding values like tracker version set in the WebView.
+ */
+function webViewPayloadBuilder(pb: PayloadBuilder): PayloadBuilder {
+ const addedKeys = new Set();
+
+ const add = (key: string, value: unknown): void => {
+ if (!addedKeys.has(key)) {
+ addedKeys.add(key);
+ pb.add(key, value);
+ }
+ };
+
+ const addDict = (dict: Payload): void => {
+ for (const key in dict) {
+ if (Object.prototype.hasOwnProperty.call(dict, key)) {
+ add(key, dict[key]);
+ }
+ }
+ };
+
+ return {
+ ...pb,
+ add,
+ addDict,
+ };
+}
+
+/**
+ * Enables tracking events from apps rendered in react-native-webview components.
+ * The apps need to use the Snowplow WebView tracker to track the events.
+ *
+ * To subscribe for the events, set the `onMessage` attribute:
+ * ``
+ *
+ * @returns Callback to subscribe for events from Web views tracked using the Snowplow WebView tracker.
+ */
+export function getWebViewCallback() {
+ return (message: { nativeEvent: { data: string } }): void => {
+ const data = JSON.parse(message.nativeEvent.data) as WebViewMessage;
+ switch (data.command) {
+ case 'trackSelfDescribingEvent':
+ forEachTracker(data.trackers, (tracker) => {
+ tracker.trackSelfDescribingEvent(data.event as SelfDescribing, data.context || []);
+ });
+ break;
+
+ case 'trackStructEvent':
+ forEachTracker(data.trackers, (tracker) => {
+ tracker.trackStructuredEvent(data.event as StructuredProps, data.context || []);
+ });
+ break;
+
+ case 'trackPageView':
+ forEachTracker(data.trackers, (tracker) => {
+ const event = data.event as WebViewPageViewEvent;
+ tracker.trackPageViewEvent({
+ pageTitle: event.title,
+ pageUrl: event.url,
+ referrer: event.referrer,
+ });
+ });
+ break;
+
+ case 'trackScreenView':
+ forEachTracker(data.trackers, (tracker) => {
+ tracker.trackScreenViewEvent(data.event as ScreenViewProps, data.context || []);
+ });
+ break;
+
+ case 'trackWebViewEvent':
+ forEachTrackerCore(data.trackers, (tracker) => {
+ const event = data.event as WebViewEvent;
+ let pb: PayloadBuilder;
+ if (event.selfDescribingEventData) {
+ pb = buildSelfDescribingEvent({ event: event.selfDescribingEventData });
+ } else {
+ pb = payloadBuilder();
+ }
+ pb = webViewPayloadBuilder(pb);
+
+ if (event.eventName !== undefined) {
+ pb.add('e', event.eventName);
+ }
+ if (event.action !== undefined) {
+ pb.add('se_ac', event.action);
+ }
+ if (event.category !== undefined) {
+ pb.add('se_ca', event.category);
+ }
+ if (event.label !== undefined) {
+ pb.add('se_la', event.label);
+ }
+ if (event.property !== undefined) {
+ pb.add('se_pr', event.property);
+ }
+ if (event.value !== undefined) {
+ pb.add('se_va', event.value.toString());
+ }
+ if (event.pageUrl !== undefined) {
+ pb.add('url', event.pageUrl);
+ }
+ if (event.pageTitle !== undefined) {
+ pb.add('page', event.pageTitle);
+ }
+ if (event.referrer !== undefined) {
+ pb.add('refr', event.referrer);
+ }
+ if (event.pingXOffsetMin !== undefined) {
+ pb.add('pp_mix', event.pingXOffsetMin.toString());
+ }
+ if (event.pingXOffsetMax !== undefined) {
+ pb.add('pp_max', event.pingXOffsetMax.toString());
+ }
+ if (event.pingYOffsetMin !== undefined) {
+ pb.add('pp_miy', event.pingYOffsetMin.toString());
+ }
+ if (event.pingYOffsetMax !== undefined) {
+ pb.add('pp_may', event.pingYOffsetMax.toString());
+ }
+ if (event.trackerVersion !== undefined) {
+ pb.add('tv', event.trackerVersion);
+ }
+ if (event.useragent !== undefined) {
+ pb.add('ua', event.useragent);
+ }
+ tracker.track(pb, data.context || []);
+ });
+ break;
+
+ default:
+ console.warn(`Unknown command from WebView: ${data.command}`);
+ }
+ };
+}
diff --git a/trackers/react-native-tracker/test/web_view_interface.test.ts b/trackers/react-native-tracker/test/web_view_interface.test.ts
new file mode 100644
index 000000000..da33775af
--- /dev/null
+++ b/trackers/react-native-tracker/test/web_view_interface.test.ts
@@ -0,0 +1,386 @@
+import { getWebViewCallback, newTracker, removeTracker } from '../src';
+
+function createMockFetch(status: number, requests: Request[]) {
+ return async (input: Request) => {
+ requests.push(input);
+ let response = new Response(null, { status });
+ return response;
+ };
+}
+
+describe('WebView interface', () => {
+ let requests: Request[];
+ let mockFetch: ReturnType;
+
+ beforeEach(async () => {
+ requests = [];
+ mockFetch = createMockFetch(200, requests);
+ });
+
+ afterEach(() => {
+ removeTracker('test');
+ });
+
+ it('tracks a page view event', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackPageView',
+ event: {
+ title: 'Home',
+ url: 'http://localhost:9090',
+ referrer: 'http://refr.com',
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ expect(event.e).toBe('pv');
+ expect(event.url).toBe('http://localhost:9090');
+ expect(event.refr).toBe('http://refr.com');
+ expect(event.page).toBe('Home');
+ });
+
+ it('tracks a self-describing event', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ encodeBase64: false,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackSelfDescribingEvent',
+ event: {
+ schema: 'iglu:com.snowplowanalytics.snowplow/event/jsonschema/1-0-0',
+ data: {
+ key: 'value',
+ },
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ const { e, ue_pr } = event;
+ expect(e).toBe('ue');
+ expect(ue_pr).toBeDefined();
+ const { data } = JSON.parse(ue_pr);
+ expect(data.schema).toBe('iglu:com.snowplowanalytics.snowplow/event/jsonschema/1-0-0');
+ expect(data.data.key).toBe('value');
+ });
+
+ it('tracks a structured event', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackStructEvent',
+ event: {
+ category: 'category',
+ action: 'action',
+ label: 'label',
+ property: 'property',
+ value: 1,
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ const { e, se_ca, se_ac, se_la, se_pr, se_va } = event;
+ expect(e).toBe('se');
+ expect(se_ca).toBe('category');
+ expect(se_ac).toBe('action');
+ expect(se_la).toBe('label');
+ expect(se_pr).toBe('property');
+ expect(se_va).toBe('1');
+ });
+
+ it('tracks a screen view event', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ encodeBase64: false,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackScreenView',
+ event: {
+ name: 'Home',
+ id: 'home',
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ expect(event.e).toBe('ue');
+ expect(event.ue_pr).toBeDefined();
+ const { data } = JSON.parse(event.ue_pr);
+ expect(data.schema).toBe('iglu:com.snowplowanalytics.mobile/screen_view/jsonschema/1-0-0');
+ expect(data.data.name).toBe('Home');
+ expect(data.data.id).toBe('home');
+ });
+
+ describe('WebView event tracking', () => {
+
+ it('tracks a page view event', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackWebViewEvent',
+ event: {
+ eventName: 'pv',
+ pageTitle: 'Home',
+ pageUrl: 'http://localhost:9090',
+ referrer: 'http://refr.com',
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ expect(event.e).toBe('pv');
+ expect(event.url).toBe('http://localhost:9090');
+ expect(event.refr).toBe('http://refr.com');
+ expect(event.page).toBe('Home');
+ });
+
+ it('tracks a self-describing event', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ encodeBase64: false,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackWebViewEvent',
+ event: {
+ selfDescribingEventData: {
+ schema: 'iglu:com.snowplowanalytics.snowplow/event/jsonschema/1-0-0',
+ data: {
+ key: 'value',
+ },
+ },
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ const { e, ue_pr } = event;
+ expect(e).toBe('ue');
+ expect(ue_pr).toBeDefined();
+ const { data } = JSON.parse(ue_pr);
+ expect(data.schema).toBe('iglu:com.snowplowanalytics.snowplow/event/jsonschema/1-0-0');
+ expect(data.data.key).toBe('value');
+ });
+
+ it('tracks a structured event', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackWebViewEvent',
+ event: {
+ eventName: 'se',
+ category: 'category',
+ action: 'action',
+ label: 'label',
+ property: 'property',
+ value: 1,
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ const { e, se_ca, se_ac, se_la, se_pr, se_va } = event;
+ expect(e).toBe('se');
+ expect(se_ca).toBe('category');
+ expect(se_ac).toBe('action');
+ expect(se_la).toBe('label');
+ expect(se_pr).toBe('property');
+ expect(se_va).toBe('1');
+ });
+
+ it('tracks a page ping event', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackWebViewEvent',
+ event: {
+ eventName: 'pp',
+ pageTitle: 'Home',
+ pageUrl: 'http://localhost:9090',
+ referrer: 'http://refr.com',
+ pingXOffsetMin: 1,
+ pingXOffsetMax: 2,
+ pingYOffsetMin: 3,
+ pingYOffsetMax: 4,
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ expect(event.e).toBe('pp');
+ expect(event.url).toBe('http://localhost:9090');
+ expect(event.refr).toBe('http://refr.com');
+ expect(event.page).toBe('Home');
+ expect(event.pp_mix).toBe('1');
+ expect(event.pp_max).toBe('2');
+ expect(event.pp_miy).toBe('3');
+ expect(event.pp_may).toBe('4');
+ });
+ });
+
+ it('tracks tracker version and useragent', async () => {
+ const tracker = await newTracker({
+ namespace: 'test',
+ appId: 'my-app',
+ endpoint: 'http://localhost:9090',
+ customFetch: mockFetch,
+ });
+
+ const webViewInterface = getWebViewCallback();
+ webViewInterface({
+ nativeEvent: {
+ data: JSON.stringify({
+ command: 'trackWebViewEvent',
+ event: {
+ eventName: 'pv',
+ pageTitle: 'Home',
+ pageUrl: 'http://localhost:9090',
+ trackerVersion: 'wv-1.0.0',
+ useragent: 'Mozilla/5.0',
+ },
+ }),
+ },
+ });
+
+ await tracker.flush();
+ expect(requests.length).toBe(1);
+
+ const [request] = requests;
+ const payload = await request?.json();
+ expect(payload.data.length).toBe(1);
+
+ const [event] = payload.data;
+ expect(event.e).toBe('pv');
+ expect(event.url).toBe('http://localhost:9090');
+ expect(event.page).toBe('Home');
+ expect(event.tv).toBe('wv-1.0.0');
+ expect(event.ua).toBe('Mozilla/5.0');
+ });
+});