journey-mapper-tracker 1.0.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.
- package/README.md +133 -0
- package/dist/api.d.ts +25 -0
- package/dist/api.d.ts.map +1 -0
- package/dist/api.js +61 -0
- package/dist/index.d.ts +43 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +75 -0
- package/dist/screenshot.d.ts +12 -0
- package/dist/screenshot.d.ts.map +1 -0
- package/dist/screenshot.js +42 -0
- package/dist/tracker.d.ts +33 -0
- package/dist/tracker.d.ts.map +1 -0
- package/dist/tracker.js +164 -0
- package/dist/types.d.ts +21 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +3 -0
- package/package.json +64 -0
package/README.md
ADDED
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
# Journey Mapper Tracker SDK
|
|
2
|
+
|
|
3
|
+
Track user page flows and capture screenshots automatically for UX research and analysis.
|
|
4
|
+
|
|
5
|
+
## Installation
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
npm install journey-mapper-tracker
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Quick Start
|
|
12
|
+
|
|
13
|
+
```typescript
|
|
14
|
+
import { JourneyTracker } from 'journey-mapper-tracker';
|
|
15
|
+
|
|
16
|
+
// Initialize the tracker
|
|
17
|
+
JourneyTracker.init({
|
|
18
|
+
apiKey: 'jm_your_api_key_here',
|
|
19
|
+
endpoint: 'https://your-journey-mapper-instance.com', // optional
|
|
20
|
+
captureDelay: 1000, // optional, ms to wait after navigation
|
|
21
|
+
excludedRoutes: ['/admin/*', '/login'], // optional
|
|
22
|
+
debug: false, // optional
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
// Manually capture a page (optional)
|
|
26
|
+
await JourneyTracker.capture();
|
|
27
|
+
|
|
28
|
+
// Stop tracking when done
|
|
29
|
+
JourneyTracker.stop();
|
|
30
|
+
```
|
|
31
|
+
|
|
32
|
+
## React/Next.js Integration
|
|
33
|
+
|
|
34
|
+
### Option 1: Use the Provider Component
|
|
35
|
+
|
|
36
|
+
```tsx
|
|
37
|
+
// components/JourneyTrackerProvider.tsx
|
|
38
|
+
"use client";
|
|
39
|
+
|
|
40
|
+
import { useEffect } from "react";
|
|
41
|
+
import { JourneyTracker } from "journey-mapper-tracker";
|
|
42
|
+
|
|
43
|
+
export function JourneyTrackerProvider({
|
|
44
|
+
apiKey,
|
|
45
|
+
children
|
|
46
|
+
}: {
|
|
47
|
+
apiKey: string;
|
|
48
|
+
children: React.ReactNode;
|
|
49
|
+
}) {
|
|
50
|
+
useEffect(() => {
|
|
51
|
+
JourneyTracker.init({
|
|
52
|
+
apiKey,
|
|
53
|
+
endpoint: window.location.origin,
|
|
54
|
+
debug: process.env.NODE_ENV === 'development',
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
return () => {
|
|
58
|
+
JourneyTracker.stop();
|
|
59
|
+
};
|
|
60
|
+
}, [apiKey]);
|
|
61
|
+
|
|
62
|
+
return <>{children}</>;
|
|
63
|
+
}
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
### Option 2: Initialize in Layout
|
|
67
|
+
|
|
68
|
+
```tsx
|
|
69
|
+
// app/layout.tsx (Next.js 13+)
|
|
70
|
+
"use client";
|
|
71
|
+
|
|
72
|
+
import { useEffect } from 'react';
|
|
73
|
+
import { JourneyTracker } from 'journey-mapper-tracker';
|
|
74
|
+
|
|
75
|
+
export default function RootLayout({ children }) {
|
|
76
|
+
useEffect(() => {
|
|
77
|
+
JourneyTracker.init({
|
|
78
|
+
apiKey: process.env.NEXT_PUBLIC_JOURNEY_MAPPER_KEY!,
|
|
79
|
+
debug: true,
|
|
80
|
+
});
|
|
81
|
+
}, []);
|
|
82
|
+
|
|
83
|
+
return (
|
|
84
|
+
<html>
|
|
85
|
+
<body>{children}</body>
|
|
86
|
+
</html>
|
|
87
|
+
);
|
|
88
|
+
}
|
|
89
|
+
```
|
|
90
|
+
|
|
91
|
+
## Configuration Options
|
|
92
|
+
|
|
93
|
+
| Option | Type | Default | Description |
|
|
94
|
+
|--------|------|---------|-------------|
|
|
95
|
+
| `apiKey` | `string` | **required** | Your project API key from Journey Mapper |
|
|
96
|
+
| `endpoint` | `string` | `window.location.origin` | Your Journey Mapper server URL |
|
|
97
|
+
| `captureDelay` | `number` | `1000` | Milliseconds to wait after navigation before capturing |
|
|
98
|
+
| `excludedRoutes` | `string[]` | `[]` | Routes to skip (supports wildcards like `/admin/*`) |
|
|
99
|
+
| `debug` | `boolean` | `false` | Enable console logging for debugging |
|
|
100
|
+
|
|
101
|
+
## API Reference
|
|
102
|
+
|
|
103
|
+
### `JourneyTracker.init(config)`
|
|
104
|
+
|
|
105
|
+
Initialize and start tracking. Must be called before any other methods.
|
|
106
|
+
|
|
107
|
+
### `JourneyTracker.stop()`
|
|
108
|
+
|
|
109
|
+
Stop tracking and complete the current session. Call this when unmounting or when you want to end the session.
|
|
110
|
+
|
|
111
|
+
### `JourneyTracker.capture()`
|
|
112
|
+
|
|
113
|
+
Manually capture the current page. Useful for capturing state after user interactions that don't trigger navigation.
|
|
114
|
+
|
|
115
|
+
### `JourneyTracker.isInitialized()`
|
|
116
|
+
|
|
117
|
+
Returns `true` if the tracker is initialized and running.
|
|
118
|
+
|
|
119
|
+
## How It Works
|
|
120
|
+
|
|
121
|
+
1. **Automatic Detection**: The SDK detects page navigations (including SPA navigation via `pushState`/`replaceState`)
|
|
122
|
+
2. **Screenshot Capture**: Uses `html2canvas` to capture full-page screenshots
|
|
123
|
+
3. **Data Upload**: Sends captured data to your Journey Mapper instance
|
|
124
|
+
4. **Session Management**: Groups all captured pages into a session for easy analysis
|
|
125
|
+
|
|
126
|
+
## Requirements
|
|
127
|
+
|
|
128
|
+
- Browser environment (not compatible with SSR/Node.js)
|
|
129
|
+
- React 17+ (optional - works with any framework)
|
|
130
|
+
|
|
131
|
+
## License
|
|
132
|
+
|
|
133
|
+
MIT
|
package/dist/api.d.ts
ADDED
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { TrackerConfig, PageData, SessionData } from './types';
|
|
2
|
+
/**
|
|
3
|
+
* API client for communicating with Journey Mapper backend
|
|
4
|
+
*/
|
|
5
|
+
export declare class ApiClient {
|
|
6
|
+
private apiKey;
|
|
7
|
+
private endpoint;
|
|
8
|
+
private debug;
|
|
9
|
+
constructor(config: TrackerConfig);
|
|
10
|
+
private log;
|
|
11
|
+
private request;
|
|
12
|
+
/**
|
|
13
|
+
* Start a new tracking session
|
|
14
|
+
*/
|
|
15
|
+
startSession(sessionData: SessionData): Promise<void>;
|
|
16
|
+
/**
|
|
17
|
+
* Send a page capture
|
|
18
|
+
*/
|
|
19
|
+
sendPage(sessionId: string, page: PageData): Promise<void>;
|
|
20
|
+
/**
|
|
21
|
+
* Complete the tracking session
|
|
22
|
+
*/
|
|
23
|
+
completeSession(sessionId: string): Promise<void>;
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=api.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"api.d.ts","sourceRoot":"","sources":["../src/api.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAI/D;;GAEG;AACH,qBAAa,SAAS;IAClB,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,KAAK,CAAU;gBAEX,MAAM,EAAE,aAAa;IAMjC,OAAO,CAAC,GAAG;YAMG,OAAO;IA2BrB;;OAEG;IACG,YAAY,CAAC,WAAW,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;IAI3D;;OAEG;IACG,QAAQ,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC;IAIhE;;OAEG;IACG,eAAe,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;CAG1D"}
|
package/dist/api.js
ADDED
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ApiClient = void 0;
|
|
4
|
+
const DEFAULT_ENDPOINT = 'https://journey-mapper.app';
|
|
5
|
+
/**
|
|
6
|
+
* API client for communicating with Journey Mapper backend
|
|
7
|
+
*/
|
|
8
|
+
class ApiClient {
|
|
9
|
+
constructor(config) {
|
|
10
|
+
this.apiKey = config.apiKey;
|
|
11
|
+
this.endpoint = config.endpoint || DEFAULT_ENDPOINT;
|
|
12
|
+
this.debug = config.debug || false;
|
|
13
|
+
}
|
|
14
|
+
log(...args) {
|
|
15
|
+
if (this.debug) {
|
|
16
|
+
console.log('[JourneyTracker]', ...args);
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
async request(action, data) {
|
|
20
|
+
const url = `${this.endpoint}/api/sdk/pages`;
|
|
21
|
+
this.log(`Sending ${action} request to ${url}`);
|
|
22
|
+
const response = await fetch(url, {
|
|
23
|
+
method: 'POST',
|
|
24
|
+
headers: {
|
|
25
|
+
'Content-Type': 'application/json',
|
|
26
|
+
'x-api-key': this.apiKey,
|
|
27
|
+
},
|
|
28
|
+
body: JSON.stringify({
|
|
29
|
+
action,
|
|
30
|
+
...data,
|
|
31
|
+
}),
|
|
32
|
+
});
|
|
33
|
+
if (!response.ok) {
|
|
34
|
+
const error = await response.json();
|
|
35
|
+
throw new Error(error.error || `Request failed: ${response.status}`);
|
|
36
|
+
}
|
|
37
|
+
const result = await response.json();
|
|
38
|
+
this.log(`${action} response:`, result);
|
|
39
|
+
return result;
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Start a new tracking session
|
|
43
|
+
*/
|
|
44
|
+
async startSession(sessionData) {
|
|
45
|
+
await this.request('start', sessionData);
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Send a page capture
|
|
49
|
+
*/
|
|
50
|
+
async sendPage(sessionId, page) {
|
|
51
|
+
await this.request('page', { sessionId, page });
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Complete the tracking session
|
|
55
|
+
*/
|
|
56
|
+
async completeSession(sessionId) {
|
|
57
|
+
await this.request('complete', { sessionId });
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
exports.ApiClient = ApiClient;
|
|
61
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBpLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2FwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFFQSxNQUFNLGdCQUFnQixHQUFHLDRCQUE0QixDQUFDO0FBRXREOztHQUVHO0FBQ0gsTUFBYSxTQUFTO0lBS2xCLFlBQVksTUFBcUI7UUFDN0IsSUFBSSxDQUFDLE1BQU0sR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDO1FBQzVCLElBQUksQ0FBQyxRQUFRLEdBQUcsTUFBTSxDQUFDLFFBQVEsSUFBSSxnQkFBZ0IsQ0FBQztRQUNwRCxJQUFJLENBQUMsS0FBSyxHQUFHLE1BQU0sQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDO0lBQ3ZDLENBQUM7SUFFTyxHQUFHLENBQUMsR0FBRyxJQUFXO1FBQ3RCLElBQUksSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ2IsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0IsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDO1FBQzdDLENBQUM7SUFDTCxDQUFDO0lBRU8sS0FBSyxDQUFDLE9BQU8sQ0FBQyxNQUFjLEVBQUUsSUFBUztRQUMzQyxNQUFNLEdBQUcsR0FBRyxHQUFHLElBQUksQ0FBQyxRQUFRLGdCQUFnQixDQUFDO1FBRTdDLElBQUksQ0FBQyxHQUFHLENBQUMsV0FBVyxNQUFNLGVBQWUsR0FBRyxFQUFFLENBQUMsQ0FBQztRQUVoRCxNQUFNLFFBQVEsR0FBRyxNQUFNLEtBQUssQ0FBQyxHQUFHLEVBQUU7WUFDOUIsTUFBTSxFQUFFLE1BQU07WUFDZCxPQUFPLEVBQUU7Z0JBQ0wsY0FBYyxFQUFFLGtCQUFrQjtnQkFDbEMsV0FBVyxFQUFFLElBQUksQ0FBQyxNQUFNO2FBQzNCO1lBQ0QsSUFBSSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUM7Z0JBQ2pCLE1BQU07Z0JBQ04sR0FBRyxJQUFJO2FBQ1YsQ0FBQztTQUNMLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxRQUFRLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDZixNQUFNLEtBQUssR0FBRyxNQUFNLFFBQVEsQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNwQyxNQUFNLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxLQUFLLElBQUksbUJBQW1CLFFBQVEsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDO1FBQ3pFLENBQUM7UUFFRCxNQUFNLE1BQU0sR0FBRyxNQUFNLFFBQVEsQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNyQyxJQUFJLENBQUMsR0FBRyxDQUFDLEdBQUcsTUFBTSxZQUFZLEVBQUUsTUFBTSxDQUFDLENBQUM7UUFDeEMsT0FBTyxNQUFNLENBQUM7SUFDbEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsS0FBSyxDQUFDLFlBQVksQ0FBQyxXQUF3QjtRQUN2QyxNQUFNLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLFdBQVcsQ0FBQyxDQUFDO0lBQzdDLENBQUM7SUFFRDs7T0FFRztJQUNILEtBQUssQ0FBQyxRQUFRLENBQUMsU0FBaUIsRUFBRSxJQUFjO1FBQzVDLE1BQU0sSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEVBQUUsRUFBRSxTQUFTLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUNwRCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxLQUFLLENBQUMsZUFBZSxDQUFDLFNBQWlCO1FBQ25DLE1BQU0sSUFBSSxDQUFDLE9BQU8sQ0FBQyxVQUFVLEVBQUUsRUFBRSxTQUFTLEVBQUUsQ0FBQyxDQUFDO0lBQ2xELENBQUM7Q0FDSjtBQWhFRCw4QkFnRUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBUcmFja2VyQ29uZmlnLCBQYWdlRGF0YSwgU2Vzc2lvbkRhdGEgfSBmcm9tICcuL3R5cGVzJztcblxuY29uc3QgREVGQVVMVF9FTkRQT0lOVCA9ICdodHRwczovL2pvdXJuZXktbWFwcGVyLmFwcCc7XG5cbi8qKlxuICogQVBJIGNsaWVudCBmb3IgY29tbXVuaWNhdGluZyB3aXRoIEpvdXJuZXkgTWFwcGVyIGJhY2tlbmRcbiAqL1xuZXhwb3J0IGNsYXNzIEFwaUNsaWVudCB7XG4gICAgcHJpdmF0ZSBhcGlLZXk6IHN0cmluZztcbiAgICBwcml2YXRlIGVuZHBvaW50OiBzdHJpbmc7XG4gICAgcHJpdmF0ZSBkZWJ1ZzogYm9vbGVhbjtcblxuICAgIGNvbnN0cnVjdG9yKGNvbmZpZzogVHJhY2tlckNvbmZpZykge1xuICAgICAgICB0aGlzLmFwaUtleSA9IGNvbmZpZy5hcGlLZXk7XG4gICAgICAgIHRoaXMuZW5kcG9pbnQgPSBjb25maWcuZW5kcG9pbnQgfHwgREVGQVVMVF9FTkRQT0lOVDtcbiAgICAgICAgdGhpcy5kZWJ1ZyA9IGNvbmZpZy5kZWJ1ZyB8fCBmYWxzZTtcbiAgICB9XG5cbiAgICBwcml2YXRlIGxvZyguLi5hcmdzOiBhbnlbXSkge1xuICAgICAgICBpZiAodGhpcy5kZWJ1Zykge1xuICAgICAgICAgICAgY29uc29sZS5sb2coJ1tKb3VybmV5VHJhY2tlcl0nLCAuLi5hcmdzKTtcbiAgICAgICAgfVxuICAgIH1cblxuICAgIHByaXZhdGUgYXN5bmMgcmVxdWVzdChhY3Rpb246IHN0cmluZywgZGF0YTogYW55KTogUHJvbWlzZTxhbnk+IHtcbiAgICAgICAgY29uc3QgdXJsID0gYCR7dGhpcy5lbmRwb2ludH0vYXBpL3Nkay9wYWdlc2A7XG5cbiAgICAgICAgdGhpcy5sb2coYFNlbmRpbmcgJHthY3Rpb259IHJlcXVlc3QgdG8gJHt1cmx9YCk7XG5cbiAgICAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBmZXRjaCh1cmwsIHtcbiAgICAgICAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgICAgICAgaGVhZGVyczoge1xuICAgICAgICAgICAgICAgICdDb250ZW50LVR5cGUnOiAnYXBwbGljYXRpb24vanNvbicsXG4gICAgICAgICAgICAgICAgJ3gtYXBpLWtleSc6IHRoaXMuYXBpS2V5LFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIGJvZHk6IEpTT04uc3RyaW5naWZ5KHtcbiAgICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgICAgLi4uZGF0YSxcbiAgICAgICAgICAgIH0pLFxuICAgICAgICB9KTtcblxuICAgICAgICBpZiAoIXJlc3BvbnNlLm9rKSB7XG4gICAgICAgICAgICBjb25zdCBlcnJvciA9IGF3YWl0IHJlc3BvbnNlLmpzb24oKTtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihlcnJvci5lcnJvciB8fCBgUmVxdWVzdCBmYWlsZWQ6ICR7cmVzcG9uc2Uuc3RhdHVzfWApO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgcmVzcG9uc2UuanNvbigpO1xuICAgICAgICB0aGlzLmxvZyhgJHthY3Rpb259IHJlc3BvbnNlOmAsIHJlc3VsdCk7XG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogU3RhcnQgYSBuZXcgdHJhY2tpbmcgc2Vzc2lvblxuICAgICAqL1xuICAgIGFzeW5jIHN0YXJ0U2Vzc2lvbihzZXNzaW9uRGF0YTogU2Vzc2lvbkRhdGEpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAgICAgYXdhaXQgdGhpcy5yZXF1ZXN0KCdzdGFydCcsIHNlc3Npb25EYXRhKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBTZW5kIGEgcGFnZSBjYXB0dXJlXG4gICAgICovXG4gICAgYXN5bmMgc2VuZFBhZ2Uoc2Vzc2lvbklkOiBzdHJpbmcsIHBhZ2U6IFBhZ2VEYXRhKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgICAgIGF3YWl0IHRoaXMucmVxdWVzdCgncGFnZScsIHsgc2Vzc2lvbklkLCBwYWdlIH0pO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENvbXBsZXRlIHRoZSB0cmFja2luZyBzZXNzaW9uXG4gICAgICovXG4gICAgYXN5bmMgY29tcGxldGVTZXNzaW9uKHNlc3Npb25JZDogc3RyaW5nKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgICAgIGF3YWl0IHRoaXMucmVxdWVzdCgnY29tcGxldGUnLCB7IHNlc3Npb25JZCB9KTtcbiAgICB9XG59XG4iXX0=
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { TrackerConfig } from './types';
|
|
2
|
+
export { Tracker } from './tracker';
|
|
3
|
+
export { TrackerConfig, PageData, SessionData } from './types';
|
|
4
|
+
/**
|
|
5
|
+
* JourneyTracker - Main export for easy initialization
|
|
6
|
+
*
|
|
7
|
+
* Usage:
|
|
8
|
+
* ```typescript
|
|
9
|
+
* import { JourneyTracker } from '@journey-mapper/tracker';
|
|
10
|
+
*
|
|
11
|
+
* // Initialize the tracker
|
|
12
|
+
* JourneyTracker.init({
|
|
13
|
+
* apiKey: 'your-api-key',
|
|
14
|
+
* endpoint: 'https://your-journey-mapper-instance.com', // optional
|
|
15
|
+
* captureDelay: 1000, // optional, ms to wait after navigation
|
|
16
|
+
* excludedRoutes: ['/admin/*'], // optional
|
|
17
|
+
* debug: false, // optional
|
|
18
|
+
* });
|
|
19
|
+
*
|
|
20
|
+
* // Later, stop tracking
|
|
21
|
+
* JourneyTracker.stop();
|
|
22
|
+
* ```
|
|
23
|
+
*/
|
|
24
|
+
export declare const JourneyTracker: {
|
|
25
|
+
/**
|
|
26
|
+
* Initialize and start the tracker
|
|
27
|
+
*/
|
|
28
|
+
init(config: TrackerConfig): Promise<void>;
|
|
29
|
+
/**
|
|
30
|
+
* Stop tracking and complete the session
|
|
31
|
+
*/
|
|
32
|
+
stop(): Promise<void>;
|
|
33
|
+
/**
|
|
34
|
+
* Manually capture the current page
|
|
35
|
+
*/
|
|
36
|
+
capture(): Promise<void>;
|
|
37
|
+
/**
|
|
38
|
+
* Check if tracker is initialized
|
|
39
|
+
*/
|
|
40
|
+
isInitialized(): boolean;
|
|
41
|
+
};
|
|
42
|
+
export default JourneyTracker;
|
|
43
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAExC,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,aAAa,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAI/D;;;;;;;;;;;;;;;;;;;GAmBG;AACH,eAAO,MAAM,cAAc;IACvB;;OAEG;iBACgB,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC;IAmBhD;;OAEG;YACW,OAAO,CAAC,IAAI,CAAC;IAU3B;;OAEG;eACc,OAAO,CAAC,IAAI,CAAC;IAQ9B;;OAEG;qBACc,OAAO;CAG3B,CAAC;AAEF,eAAe,cAAc,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.JourneyTracker = exports.Tracker = void 0;
|
|
4
|
+
const tracker_1 = require("./tracker");
|
|
5
|
+
var tracker_2 = require("./tracker");
|
|
6
|
+
Object.defineProperty(exports, "Tracker", { enumerable: true, get: function () { return tracker_2.Tracker; } });
|
|
7
|
+
let globalTracker = null;
|
|
8
|
+
/**
|
|
9
|
+
* JourneyTracker - Main export for easy initialization
|
|
10
|
+
*
|
|
11
|
+
* Usage:
|
|
12
|
+
* ```typescript
|
|
13
|
+
* import { JourneyTracker } from '@journey-mapper/tracker';
|
|
14
|
+
*
|
|
15
|
+
* // Initialize the tracker
|
|
16
|
+
* JourneyTracker.init({
|
|
17
|
+
* apiKey: 'your-api-key',
|
|
18
|
+
* endpoint: 'https://your-journey-mapper-instance.com', // optional
|
|
19
|
+
* captureDelay: 1000, // optional, ms to wait after navigation
|
|
20
|
+
* excludedRoutes: ['/admin/*'], // optional
|
|
21
|
+
* debug: false, // optional
|
|
22
|
+
* });
|
|
23
|
+
*
|
|
24
|
+
* // Later, stop tracking
|
|
25
|
+
* JourneyTracker.stop();
|
|
26
|
+
* ```
|
|
27
|
+
*/
|
|
28
|
+
exports.JourneyTracker = {
|
|
29
|
+
/**
|
|
30
|
+
* Initialize and start the tracker
|
|
31
|
+
*/
|
|
32
|
+
async init(config) {
|
|
33
|
+
if (typeof window === 'undefined') {
|
|
34
|
+
console.warn('[JourneyTracker] Cannot initialize in non-browser environment');
|
|
35
|
+
return;
|
|
36
|
+
}
|
|
37
|
+
if (globalTracker) {
|
|
38
|
+
console.warn('[JourneyTracker] Already initialized');
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
if (!config.apiKey) {
|
|
42
|
+
throw new Error('[JourneyTracker] API key is required');
|
|
43
|
+
}
|
|
44
|
+
globalTracker = new tracker_1.Tracker(config);
|
|
45
|
+
await globalTracker.start();
|
|
46
|
+
},
|
|
47
|
+
/**
|
|
48
|
+
* Stop tracking and complete the session
|
|
49
|
+
*/
|
|
50
|
+
async stop() {
|
|
51
|
+
if (!globalTracker) {
|
|
52
|
+
console.warn('[JourneyTracker] Not initialized');
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
55
|
+
await globalTracker.stop();
|
|
56
|
+
globalTracker = null;
|
|
57
|
+
},
|
|
58
|
+
/**
|
|
59
|
+
* Manually capture the current page
|
|
60
|
+
*/
|
|
61
|
+
async capture() {
|
|
62
|
+
if (!globalTracker) {
|
|
63
|
+
throw new Error('[JourneyTracker] Not initialized. Call init() first.');
|
|
64
|
+
}
|
|
65
|
+
await globalTracker.capture();
|
|
66
|
+
},
|
|
67
|
+
/**
|
|
68
|
+
* Check if tracker is initialized
|
|
69
|
+
*/
|
|
70
|
+
isInitialized() {
|
|
71
|
+
return globalTracker !== null;
|
|
72
|
+
},
|
|
73
|
+
};
|
|
74
|
+
exports.default = exports.JourneyTracker;
|
|
75
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsdUNBQW9DO0FBR3BDLHFDQUFvQztBQUEzQixrR0FBQSxPQUFPLE9BQUE7QUFHaEIsSUFBSSxhQUFhLEdBQW1CLElBQUksQ0FBQztBQUV6Qzs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW1CRztBQUNVLFFBQUEsY0FBYyxHQUFHO0lBQzFCOztPQUVHO0lBQ0gsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFxQjtRQUM1QixJQUFJLE9BQU8sTUFBTSxLQUFLLFdBQVcsRUFBRSxDQUFDO1lBQ2hDLE9BQU8sQ0FBQyxJQUFJLENBQUMsK0RBQStELENBQUMsQ0FBQztZQUM5RSxPQUFPO1FBQ1gsQ0FBQztRQUVELElBQUksYUFBYSxFQUFFLENBQUM7WUFDaEIsT0FBTyxDQUFDLElBQUksQ0FBQyxzQ0FBc0MsQ0FBQyxDQUFDO1lBQ3JELE9BQU87UUFDWCxDQUFDO1FBRUQsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNqQixNQUFNLElBQUksS0FBSyxDQUFDLHNDQUFzQyxDQUFDLENBQUM7UUFDNUQsQ0FBQztRQUVELGFBQWEsR0FBRyxJQUFJLGlCQUFPLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDcEMsTUFBTSxhQUFhLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDaEMsQ0FBQztJQUVEOztPQUVHO0lBQ0gsS0FBSyxDQUFDLElBQUk7UUFDTixJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDakIsT0FBTyxDQUFDLElBQUksQ0FBQyxrQ0FBa0MsQ0FBQyxDQUFDO1lBQ2pELE9BQU87UUFDWCxDQUFDO1FBRUQsTUFBTSxhQUFhLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDM0IsYUFBYSxHQUFHLElBQUksQ0FBQztJQUN6QixDQUFDO0lBRUQ7O09BRUc7SUFDSCxLQUFLLENBQUMsT0FBTztRQUNULElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztZQUNqQixNQUFNLElBQUksS0FBSyxDQUFDLHNEQUFzRCxDQUFDLENBQUM7UUFDNUUsQ0FBQztRQUVELE1BQU0sYUFBYSxDQUFDLE9BQU8sRUFBRSxDQUFDO0lBQ2xDLENBQUM7SUFFRDs7T0FFRztJQUNILGFBQWE7UUFDVCxPQUFPLGFBQWEsS0FBSyxJQUFJLENBQUM7SUFDbEMsQ0FBQztDQUNKLENBQUM7QUFFRixrQkFBZSxzQkFBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVHJhY2tlciB9IGZyb20gJy4vdHJhY2tlcic7XG5pbXBvcnQgeyBUcmFja2VyQ29uZmlnIH0gZnJvbSAnLi90eXBlcyc7XG5cbmV4cG9ydCB7IFRyYWNrZXIgfSBmcm9tICcuL3RyYWNrZXInO1xuZXhwb3J0IHsgVHJhY2tlckNvbmZpZywgUGFnZURhdGEsIFNlc3Npb25EYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5cbmxldCBnbG9iYWxUcmFja2VyOiBUcmFja2VyIHwgbnVsbCA9IG51bGw7XG5cbi8qKlxuICogSm91cm5leVRyYWNrZXIgLSBNYWluIGV4cG9ydCBmb3IgZWFzeSBpbml0aWFsaXphdGlvblxuICogXG4gKiBVc2FnZTpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IEpvdXJuZXlUcmFja2VyIH0gZnJvbSAnQGpvdXJuZXktbWFwcGVyL3RyYWNrZXInO1xuICogXG4gKiAvLyBJbml0aWFsaXplIHRoZSB0cmFja2VyXG4gKiBKb3VybmV5VHJhY2tlci5pbml0KHtcbiAqICAgYXBpS2V5OiAneW91ci1hcGkta2V5JyxcbiAqICAgZW5kcG9pbnQ6ICdodHRwczovL3lvdXItam91cm5leS1tYXBwZXItaW5zdGFuY2UuY29tJywgLy8gb3B0aW9uYWxcbiAqICAgY2FwdHVyZURlbGF5OiAxMDAwLCAvLyBvcHRpb25hbCwgbXMgdG8gd2FpdCBhZnRlciBuYXZpZ2F0aW9uXG4gKiAgIGV4Y2x1ZGVkUm91dGVzOiBbJy9hZG1pbi8qJ10sIC8vIG9wdGlvbmFsXG4gKiAgIGRlYnVnOiBmYWxzZSwgLy8gb3B0aW9uYWxcbiAqIH0pO1xuICogXG4gKiAvLyBMYXRlciwgc3RvcCB0cmFja2luZ1xuICogSm91cm5leVRyYWNrZXIuc3RvcCgpO1xuICogYGBgXG4gKi9cbmV4cG9ydCBjb25zdCBKb3VybmV5VHJhY2tlciA9IHtcbiAgICAvKipcbiAgICAgKiBJbml0aWFsaXplIGFuZCBzdGFydCB0aGUgdHJhY2tlclxuICAgICAqL1xuICAgIGFzeW5jIGluaXQoY29uZmlnOiBUcmFja2VyQ29uZmlnKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgICAgIGlmICh0eXBlb2Ygd2luZG93ID09PSAndW5kZWZpbmVkJykge1xuICAgICAgICAgICAgY29uc29sZS53YXJuKCdbSm91cm5leVRyYWNrZXJdIENhbm5vdCBpbml0aWFsaXplIGluIG5vbi1icm93c2VyIGVudmlyb25tZW50Jyk7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBpZiAoZ2xvYmFsVHJhY2tlcikge1xuICAgICAgICAgICAgY29uc29sZS53YXJuKCdbSm91cm5leVRyYWNrZXJdIEFscmVhZHkgaW5pdGlhbGl6ZWQnKTtcbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICghY29uZmlnLmFwaUtleSkge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdbSm91cm5leVRyYWNrZXJdIEFQSSBrZXkgaXMgcmVxdWlyZWQnKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGdsb2JhbFRyYWNrZXIgPSBuZXcgVHJhY2tlcihjb25maWcpO1xuICAgICAgICBhd2FpdCBnbG9iYWxUcmFja2VyLnN0YXJ0KCk7XG4gICAgfSxcblxuICAgIC8qKlxuICAgICAqIFN0b3AgdHJhY2tpbmcgYW5kIGNvbXBsZXRlIHRoZSBzZXNzaW9uXG4gICAgICovXG4gICAgYXN5bmMgc3RvcCgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAgICAgaWYgKCFnbG9iYWxUcmFja2VyKSB7XG4gICAgICAgICAgICBjb25zb2xlLndhcm4oJ1tKb3VybmV5VHJhY2tlcl0gTm90IGluaXRpYWxpemVkJyk7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBhd2FpdCBnbG9iYWxUcmFja2VyLnN0b3AoKTtcbiAgICAgICAgZ2xvYmFsVHJhY2tlciA9IG51bGw7XG4gICAgfSxcblxuICAgIC8qKlxuICAgICAqIE1hbnVhbGx5IGNhcHR1cmUgdGhlIGN1cnJlbnQgcGFnZVxuICAgICAqL1xuICAgIGFzeW5jIGNhcHR1cmUoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgICAgIGlmICghZ2xvYmFsVHJhY2tlcikge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdbSm91cm5leVRyYWNrZXJdIE5vdCBpbml0aWFsaXplZC4gQ2FsbCBpbml0KCkgZmlyc3QuJyk7XG4gICAgICAgIH1cblxuICAgICAgICBhd2FpdCBnbG9iYWxUcmFja2VyLmNhcHR1cmUoKTtcbiAgICB9LFxuXG4gICAgLyoqXG4gICAgICogQ2hlY2sgaWYgdHJhY2tlciBpcyBpbml0aWFsaXplZFxuICAgICAqL1xuICAgIGlzSW5pdGlhbGl6ZWQoKTogYm9vbGVhbiB7XG4gICAgICAgIHJldHVybiBnbG9iYWxUcmFja2VyICE9PSBudWxsO1xuICAgIH0sXG59O1xuXG5leHBvcnQgZGVmYXVsdCBKb3VybmV5VHJhY2tlcjtcbiJdfQ==
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Capture a screenshot of the current page using html2canvas
|
|
3
|
+
*/
|
|
4
|
+
export declare function captureScreenshot(): Promise<string>;
|
|
5
|
+
/**
|
|
6
|
+
* Get current viewport dimensions
|
|
7
|
+
*/
|
|
8
|
+
export declare function getViewport(): {
|
|
9
|
+
width: number;
|
|
10
|
+
height: number;
|
|
11
|
+
};
|
|
12
|
+
//# sourceMappingURL=screenshot.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"screenshot.d.ts","sourceRoot":"","sources":["../src/screenshot.ts"],"names":[],"mappings":"AAEA;;GAEG;AACH,wBAAsB,iBAAiB,IAAI,OAAO,CAAC,MAAM,CAAC,CAoBzD;AAED;;GAEG;AACH,wBAAgB,WAAW,IAAI;IAAE,KAAK,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,CAK/D"}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.captureScreenshot = captureScreenshot;
|
|
7
|
+
exports.getViewport = getViewport;
|
|
8
|
+
const html2canvas_1 = __importDefault(require("html2canvas"));
|
|
9
|
+
/**
|
|
10
|
+
* Capture a screenshot of the current page using html2canvas
|
|
11
|
+
*/
|
|
12
|
+
async function captureScreenshot() {
|
|
13
|
+
try {
|
|
14
|
+
const canvas = await (0, html2canvas_1.default)(document.body, {
|
|
15
|
+
useCORS: true,
|
|
16
|
+
allowTaint: true,
|
|
17
|
+
logging: false,
|
|
18
|
+
scale: 1, // Use 1x scale to reduce image size
|
|
19
|
+
windowWidth: window.innerWidth,
|
|
20
|
+
windowHeight: window.innerHeight,
|
|
21
|
+
scrollX: 0,
|
|
22
|
+
scrollY: 0,
|
|
23
|
+
});
|
|
24
|
+
// Convert to base64 JPEG for smaller file size
|
|
25
|
+
const dataUrl = canvas.toDataURL('image/jpeg', 0.7);
|
|
26
|
+
return dataUrl;
|
|
27
|
+
}
|
|
28
|
+
catch (error) {
|
|
29
|
+
console.error('[JourneyTracker] Screenshot capture failed:', error);
|
|
30
|
+
throw error;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Get current viewport dimensions
|
|
35
|
+
*/
|
|
36
|
+
function getViewport() {
|
|
37
|
+
return {
|
|
38
|
+
width: window.innerWidth,
|
|
39
|
+
height: window.innerHeight,
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NyZWVuc2hvdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9zY3JlZW5zaG90LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBS0EsOENBb0JDO0FBS0Qsa0NBS0M7QUFuQ0QsOERBQXNDO0FBRXRDOztHQUVHO0FBQ0ksS0FBSyxVQUFVLGlCQUFpQjtJQUNuQyxJQUFJLENBQUM7UUFDRCxNQUFNLE1BQU0sR0FBRyxNQUFNLElBQUEscUJBQVcsRUFBQyxRQUFRLENBQUMsSUFBSSxFQUFFO1lBQzVDLE9BQU8sRUFBRSxJQUFJO1lBQ2IsVUFBVSxFQUFFLElBQUk7WUFDaEIsT0FBTyxFQUFFLEtBQUs7WUFDZCxLQUFLLEVBQUUsQ0FBQyxFQUFFLG9DQUFvQztZQUM5QyxXQUFXLEVBQUUsTUFBTSxDQUFDLFVBQVU7WUFDOUIsWUFBWSxFQUFFLE1BQU0sQ0FBQyxXQUFXO1lBQ2hDLE9BQU8sRUFBRSxDQUFDO1lBQ1YsT0FBTyxFQUFFLENBQUM7U0FDYixDQUFDLENBQUM7UUFFSCwrQ0FBK0M7UUFDL0MsTUFBTSxPQUFPLEdBQUcsTUFBTSxDQUFDLFNBQVMsQ0FBQyxZQUFZLEVBQUUsR0FBRyxDQUFDLENBQUM7UUFDcEQsT0FBTyxPQUFPLENBQUM7SUFDbkIsQ0FBQztJQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7UUFDYixPQUFPLENBQUMsS0FBSyxDQUFDLDZDQUE2QyxFQUFFLEtBQUssQ0FBQyxDQUFDO1FBQ3BFLE1BQU0sS0FBSyxDQUFDO0lBQ2hCLENBQUM7QUFDTCxDQUFDO0FBRUQ7O0dBRUc7QUFDSCxTQUFnQixXQUFXO0lBQ3ZCLE9BQU87UUFDSCxLQUFLLEVBQUUsTUFBTSxDQUFDLFVBQVU7UUFDeEIsTUFBTSxFQUFFLE1BQU0sQ0FBQyxXQUFXO0tBQzdCLENBQUM7QUFDTixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGh0bWwyY2FudmFzIGZyb20gJ2h0bWwyY2FudmFzJztcblxuLyoqXG4gKiBDYXB0dXJlIGEgc2NyZWVuc2hvdCBvZiB0aGUgY3VycmVudCBwYWdlIHVzaW5nIGh0bWwyY2FudmFzXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjYXB0dXJlU2NyZWVuc2hvdCgpOiBQcm9taXNlPHN0cmluZz4ge1xuICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IGNhbnZhcyA9IGF3YWl0IGh0bWwyY2FudmFzKGRvY3VtZW50LmJvZHksIHtcbiAgICAgICAgICAgIHVzZUNPUlM6IHRydWUsXG4gICAgICAgICAgICBhbGxvd1RhaW50OiB0cnVlLFxuICAgICAgICAgICAgbG9nZ2luZzogZmFsc2UsXG4gICAgICAgICAgICBzY2FsZTogMSwgLy8gVXNlIDF4IHNjYWxlIHRvIHJlZHVjZSBpbWFnZSBzaXplXG4gICAgICAgICAgICB3aW5kb3dXaWR0aDogd2luZG93LmlubmVyV2lkdGgsXG4gICAgICAgICAgICB3aW5kb3dIZWlnaHQ6IHdpbmRvdy5pbm5lckhlaWdodCxcbiAgICAgICAgICAgIHNjcm9sbFg6IDAsXG4gICAgICAgICAgICBzY3JvbGxZOiAwLFxuICAgICAgICB9KTtcblxuICAgICAgICAvLyBDb252ZXJ0IHRvIGJhc2U2NCBKUEVHIGZvciBzbWFsbGVyIGZpbGUgc2l6ZVxuICAgICAgICBjb25zdCBkYXRhVXJsID0gY2FudmFzLnRvRGF0YVVSTCgnaW1hZ2UvanBlZycsIDAuNyk7XG4gICAgICAgIHJldHVybiBkYXRhVXJsO1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoJ1tKb3VybmV5VHJhY2tlcl0gU2NyZWVuc2hvdCBjYXB0dXJlIGZhaWxlZDonLCBlcnJvcik7XG4gICAgICAgIHRocm93IGVycm9yO1xuICAgIH1cbn1cblxuLyoqXG4gKiBHZXQgY3VycmVudCB2aWV3cG9ydCBkaW1lbnNpb25zXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRWaWV3cG9ydCgpOiB7IHdpZHRoOiBudW1iZXI7IGhlaWdodDogbnVtYmVyIH0ge1xuICAgIHJldHVybiB7XG4gICAgICAgIHdpZHRoOiB3aW5kb3cuaW5uZXJXaWR0aCxcbiAgICAgICAgaGVpZ2h0OiB3aW5kb3cuaW5uZXJIZWlnaHQsXG4gICAgfTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { TrackerConfig } from './types';
|
|
2
|
+
/**
|
|
3
|
+
* Core tracker that handles page navigation detection and screenshot capture
|
|
4
|
+
*/
|
|
5
|
+
export declare class Tracker {
|
|
6
|
+
private config;
|
|
7
|
+
private apiClient;
|
|
8
|
+
private sessionId;
|
|
9
|
+
private isInitialized;
|
|
10
|
+
private currentUrl;
|
|
11
|
+
private captureTimeout;
|
|
12
|
+
constructor(config: TrackerConfig);
|
|
13
|
+
private log;
|
|
14
|
+
private generateSessionId;
|
|
15
|
+
private isExcluded;
|
|
16
|
+
/**
|
|
17
|
+
* Start tracking
|
|
18
|
+
*/
|
|
19
|
+
start(): Promise<void>;
|
|
20
|
+
/**
|
|
21
|
+
* Stop tracking and complete the session
|
|
22
|
+
*/
|
|
23
|
+
stop(): Promise<void>;
|
|
24
|
+
private setupNavigationListeners;
|
|
25
|
+
private removeNavigationListeners;
|
|
26
|
+
private handleNavigation;
|
|
27
|
+
private capturePage;
|
|
28
|
+
/**
|
|
29
|
+
* Manually trigger a page capture
|
|
30
|
+
*/
|
|
31
|
+
capture(): Promise<void>;
|
|
32
|
+
}
|
|
33
|
+
//# sourceMappingURL=tracker.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tracker.d.ts","sourceRoot":"","sources":["../src/tracker.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAIxC;;GAEG;AACH,qBAAa,OAAO;IAChB,OAAO,CAAC,MAAM,CAAgB;IAC9B,OAAO,CAAC,SAAS,CAAY;IAC7B,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,aAAa,CAAkB;IACvC,OAAO,CAAC,UAAU,CAAc;IAChC,OAAO,CAAC,cAAc,CAA+B;gBAEzC,MAAM,EAAE,aAAa;IAWjC,OAAO,CAAC,GAAG;IAMX,OAAO,CAAC,iBAAiB;IAIzB,OAAO,CAAC,UAAU;IAWlB;;OAEG;IACG,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IA8B5B;;OAEG;IACG,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAiB3B,OAAO,CAAC,wBAAwB;IAuBhC,OAAO,CAAC,yBAAyB;IAYjC,OAAO,CAAC,gBAAgB,CAmBtB;YAEY,WAAW;IA2BzB;;OAEG;IACG,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;CAMjC"}
|
package/dist/tracker.js
ADDED
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.Tracker = void 0;
|
|
4
|
+
const api_1 = require("./api");
|
|
5
|
+
const screenshot_1 = require("./screenshot");
|
|
6
|
+
/**
|
|
7
|
+
* Core tracker that handles page navigation detection and screenshot capture
|
|
8
|
+
*/
|
|
9
|
+
class Tracker {
|
|
10
|
+
constructor(config) {
|
|
11
|
+
this.isInitialized = false;
|
|
12
|
+
this.currentUrl = '';
|
|
13
|
+
this.captureTimeout = null;
|
|
14
|
+
this.handleNavigation = () => {
|
|
15
|
+
const newUrl = window.location.href;
|
|
16
|
+
if (newUrl === this.currentUrl) {
|
|
17
|
+
return;
|
|
18
|
+
}
|
|
19
|
+
this.log('Navigation detected:', this.currentUrl, '->', newUrl);
|
|
20
|
+
this.currentUrl = newUrl;
|
|
21
|
+
// Clear existing timeout
|
|
22
|
+
if (this.captureTimeout) {
|
|
23
|
+
clearTimeout(this.captureTimeout);
|
|
24
|
+
}
|
|
25
|
+
// Delay capture to allow page to render
|
|
26
|
+
this.captureTimeout = setTimeout(() => {
|
|
27
|
+
this.capturePage();
|
|
28
|
+
}, this.config.captureDelay);
|
|
29
|
+
};
|
|
30
|
+
this.config = {
|
|
31
|
+
captureDelay: 1000, // Default 1s delay after navigation
|
|
32
|
+
excludedRoutes: [],
|
|
33
|
+
debug: false,
|
|
34
|
+
...config,
|
|
35
|
+
};
|
|
36
|
+
this.apiClient = new api_1.ApiClient(config);
|
|
37
|
+
this.sessionId = this.generateSessionId();
|
|
38
|
+
}
|
|
39
|
+
log(...args) {
|
|
40
|
+
if (this.config.debug) {
|
|
41
|
+
console.log('[JourneyTracker]', ...args);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
generateSessionId() {
|
|
45
|
+
return `jm_${Date.now()}_${Math.random().toString(36).substring(2, 11)}`;
|
|
46
|
+
}
|
|
47
|
+
isExcluded(url) {
|
|
48
|
+
const path = new URL(url).pathname;
|
|
49
|
+
return (this.config.excludedRoutes || []).some(route => {
|
|
50
|
+
if (route.includes('*')) {
|
|
51
|
+
const regex = new RegExp(route.replace(/\*/g, '.*'));
|
|
52
|
+
return regex.test(path);
|
|
53
|
+
}
|
|
54
|
+
return path === route || path.startsWith(route);
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Start tracking
|
|
59
|
+
*/
|
|
60
|
+
async start() {
|
|
61
|
+
if (this.isInitialized) {
|
|
62
|
+
this.log('Already initialized');
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
this.log('Starting tracker with session:', this.sessionId);
|
|
66
|
+
try {
|
|
67
|
+
// Start session on backend
|
|
68
|
+
await this.apiClient.startSession({
|
|
69
|
+
sessionId: this.sessionId,
|
|
70
|
+
userAgent: navigator.userAgent,
|
|
71
|
+
});
|
|
72
|
+
// Set up navigation listeners
|
|
73
|
+
this.setupNavigationListeners();
|
|
74
|
+
// Capture initial page
|
|
75
|
+
this.currentUrl = window.location.href;
|
|
76
|
+
await this.capturePage();
|
|
77
|
+
this.isInitialized = true;
|
|
78
|
+
this.log('Tracker started successfully');
|
|
79
|
+
}
|
|
80
|
+
catch (error) {
|
|
81
|
+
console.error('[JourneyTracker] Failed to start:', error);
|
|
82
|
+
throw error;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Stop tracking and complete the session
|
|
87
|
+
*/
|
|
88
|
+
async stop() {
|
|
89
|
+
if (!this.isInitialized) {
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
this.log('Stopping tracker');
|
|
93
|
+
try {
|
|
94
|
+
await this.apiClient.completeSession(this.sessionId);
|
|
95
|
+
this.removeNavigationListeners();
|
|
96
|
+
this.isInitialized = false;
|
|
97
|
+
this.log('Tracker stopped successfully');
|
|
98
|
+
}
|
|
99
|
+
catch (error) {
|
|
100
|
+
console.error('[JourneyTracker] Failed to stop:', error);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
setupNavigationListeners() {
|
|
104
|
+
// Listen for popstate (browser back/forward)
|
|
105
|
+
window.addEventListener('popstate', this.handleNavigation);
|
|
106
|
+
// Monkey-patch pushState and replaceState for SPA navigation
|
|
107
|
+
const originalPushState = history.pushState;
|
|
108
|
+
const originalReplaceState = history.replaceState;
|
|
109
|
+
history.pushState = (...args) => {
|
|
110
|
+
originalPushState.apply(history, args);
|
|
111
|
+
this.handleNavigation();
|
|
112
|
+
};
|
|
113
|
+
history.replaceState = (...args) => {
|
|
114
|
+
originalReplaceState.apply(history, args);
|
|
115
|
+
this.handleNavigation();
|
|
116
|
+
};
|
|
117
|
+
// Store originals for cleanup
|
|
118
|
+
window.__jm_originalPushState = originalPushState;
|
|
119
|
+
window.__jm_originalReplaceState = originalReplaceState;
|
|
120
|
+
}
|
|
121
|
+
removeNavigationListeners() {
|
|
122
|
+
window.removeEventListener('popstate', this.handleNavigation);
|
|
123
|
+
// Restore original methods
|
|
124
|
+
if (window.__jm_originalPushState) {
|
|
125
|
+
history.pushState = window.__jm_originalPushState;
|
|
126
|
+
}
|
|
127
|
+
if (window.__jm_originalReplaceState) {
|
|
128
|
+
history.replaceState = window.__jm_originalReplaceState;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
async capturePage() {
|
|
132
|
+
const url = window.location.href;
|
|
133
|
+
if (this.isExcluded(url)) {
|
|
134
|
+
this.log('Skipping excluded route:', url);
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
this.log('Capturing page:', url);
|
|
138
|
+
try {
|
|
139
|
+
const screenshot = await (0, screenshot_1.captureScreenshot)();
|
|
140
|
+
const viewport = (0, screenshot_1.getViewport)();
|
|
141
|
+
await this.apiClient.sendPage(this.sessionId, {
|
|
142
|
+
url,
|
|
143
|
+
title: document.title,
|
|
144
|
+
screenshot,
|
|
145
|
+
viewport,
|
|
146
|
+
});
|
|
147
|
+
this.log('Page captured successfully');
|
|
148
|
+
}
|
|
149
|
+
catch (error) {
|
|
150
|
+
console.error('[JourneyTracker] Failed to capture page:', error);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Manually trigger a page capture
|
|
155
|
+
*/
|
|
156
|
+
async capture() {
|
|
157
|
+
if (!this.isInitialized) {
|
|
158
|
+
throw new Error('Tracker not initialized. Call start() first.');
|
|
159
|
+
}
|
|
160
|
+
await this.capturePage();
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
exports.Tracker = Tracker;
|
|
164
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tracker.js","sourceRoot":"","sources":["../src/tracker.ts"],"names":[],"mappings":"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","sourcesContent":["import { TrackerConfig } from './types';\nimport { ApiClient } from './api';\nimport { captureScreenshot, getViewport } from './screenshot';\n\n/**\n * Core tracker that handles page navigation detection and screenshot capture\n */\nexport class Tracker {\n    private config: TrackerConfig;\n    private apiClient: ApiClient;\n    private sessionId: string;\n    private isInitialized: boolean = false;\n    private currentUrl: string = '';\n    private captureTimeout: NodeJS.Timeout | null = null;\n\n    constructor(config: TrackerConfig) {\n        this.config = {\n            captureDelay: 1000, // Default 1s delay after navigation\n            excludedRoutes: [],\n            debug: false,\n            ...config,\n        };\n        this.apiClient = new ApiClient(config);\n        this.sessionId = this.generateSessionId();\n    }\n\n    private log(...args: any[]) {\n        if (this.config.debug) {\n            console.log('[JourneyTracker]', ...args);\n        }\n    }\n\n    private generateSessionId(): string {\n        return `jm_${Date.now()}_${Math.random().toString(36).substring(2, 11)}`;\n    }\n\n    private isExcluded(url: string): boolean {\n        const path = new URL(url).pathname;\n        return (this.config.excludedRoutes || []).some(route => {\n            if (route.includes('*')) {\n                const regex = new RegExp(route.replace(/\\*/g, '.*'));\n                return regex.test(path);\n            }\n            return path === route || path.startsWith(route);\n        });\n    }\n\n    /**\n     * Start tracking\n     */\n    async start(): Promise<void> {\n        if (this.isInitialized) {\n            this.log('Already initialized');\n            return;\n        }\n\n        this.log('Starting tracker with session:', this.sessionId);\n\n        try {\n            // Start session on backend\n            await this.apiClient.startSession({\n                sessionId: this.sessionId,\n                userAgent: navigator.userAgent,\n            });\n\n            // Set up navigation listeners\n            this.setupNavigationListeners();\n\n            // Capture initial page\n            this.currentUrl = window.location.href;\n            await this.capturePage();\n\n            this.isInitialized = true;\n            this.log('Tracker started successfully');\n        } catch (error) {\n            console.error('[JourneyTracker] Failed to start:', error);\n            throw error;\n        }\n    }\n\n    /**\n     * Stop tracking and complete the session\n     */\n    async stop(): Promise<void> {\n        if (!this.isInitialized) {\n            return;\n        }\n\n        this.log('Stopping tracker');\n\n        try {\n            await this.apiClient.completeSession(this.sessionId);\n            this.removeNavigationListeners();\n            this.isInitialized = false;\n            this.log('Tracker stopped successfully');\n        } catch (error) {\n            console.error('[JourneyTracker] Failed to stop:', error);\n        }\n    }\n\n    private setupNavigationListeners(): void {\n        // Listen for popstate (browser back/forward)\n        window.addEventListener('popstate', this.handleNavigation);\n\n        // Monkey-patch pushState and replaceState for SPA navigation\n        const originalPushState = history.pushState;\n        const originalReplaceState = history.replaceState;\n\n        history.pushState = (...args) => {\n            originalPushState.apply(history, args);\n            this.handleNavigation();\n        };\n\n        history.replaceState = (...args) => {\n            originalReplaceState.apply(history, args);\n            this.handleNavigation();\n        };\n\n        // Store originals for cleanup\n        (window as any).__jm_originalPushState = originalPushState;\n        (window as any).__jm_originalReplaceState = originalReplaceState;\n    }\n\n    private removeNavigationListeners(): void {\n        window.removeEventListener('popstate', this.handleNavigation);\n\n        // Restore original methods\n        if ((window as any).__jm_originalPushState) {\n            history.pushState = (window as any).__jm_originalPushState;\n        }\n        if ((window as any).__jm_originalReplaceState) {\n            history.replaceState = (window as any).__jm_originalReplaceState;\n        }\n    }\n\n    private handleNavigation = (): void => {\n        const newUrl = window.location.href;\n\n        if (newUrl === this.currentUrl) {\n            return;\n        }\n\n        this.log('Navigation detected:', this.currentUrl, '->', newUrl);\n        this.currentUrl = newUrl;\n\n        // Clear existing timeout\n        if (this.captureTimeout) {\n            clearTimeout(this.captureTimeout);\n        }\n\n        // Delay capture to allow page to render\n        this.captureTimeout = setTimeout(() => {\n            this.capturePage();\n        }, this.config.captureDelay);\n    };\n\n    private async capturePage(): Promise<void> {\n        const url = window.location.href;\n\n        if (this.isExcluded(url)) {\n            this.log('Skipping excluded route:', url);\n            return;\n        }\n\n        this.log('Capturing page:', url);\n\n        try {\n            const screenshot = await captureScreenshot();\n            const viewport = getViewport();\n\n            await this.apiClient.sendPage(this.sessionId, {\n                url,\n                title: document.title,\n                screenshot,\n                viewport,\n            });\n\n            this.log('Page captured successfully');\n        } catch (error) {\n            console.error('[JourneyTracker] Failed to capture page:', error);\n        }\n    }\n\n    /**\n     * Manually trigger a page capture\n     */\n    async capture(): Promise<void> {\n        if (!this.isInitialized) {\n            throw new Error('Tracker not initialized. Call start() first.');\n        }\n        await this.capturePage();\n    }\n}\n"]}
|
package/dist/types.d.ts
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
export interface TrackerConfig {
|
|
2
|
+
apiKey: string;
|
|
3
|
+
endpoint?: string;
|
|
4
|
+
captureDelay?: number;
|
|
5
|
+
excludedRoutes?: string[];
|
|
6
|
+
debug?: boolean;
|
|
7
|
+
}
|
|
8
|
+
export interface PageData {
|
|
9
|
+
url: string;
|
|
10
|
+
title: string;
|
|
11
|
+
screenshot: string;
|
|
12
|
+
viewport: {
|
|
13
|
+
width: number;
|
|
14
|
+
height: number;
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
export interface SessionData {
|
|
18
|
+
sessionId: string;
|
|
19
|
+
userAgent: string;
|
|
20
|
+
}
|
|
21
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,aAAa;IAC1B,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;IAC1B,KAAK,CAAC,EAAE,OAAO,CAAC;CACnB;AAED,MAAM,WAAW,QAAQ;IACrB,GAAG,EAAE,MAAM,CAAC;IACZ,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE;QACN,KAAK,EAAE,MAAM,CAAC;QACd,MAAM,EAAE,MAAM,CAAC;KAClB,CAAC;CACL;AAED,MAAM,WAAW,WAAW;IACxB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;CACrB"}
|
package/dist/types.js
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgVHJhY2tlckNvbmZpZyB7XG4gICAgYXBpS2V5OiBzdHJpbmc7XG4gICAgZW5kcG9pbnQ/OiBzdHJpbmc7XG4gICAgY2FwdHVyZURlbGF5PzogbnVtYmVyO1xuICAgIGV4Y2x1ZGVkUm91dGVzPzogc3RyaW5nW107XG4gICAgZGVidWc/OiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBhZ2VEYXRhIHtcbiAgICB1cmw6IHN0cmluZztcbiAgICB0aXRsZTogc3RyaW5nO1xuICAgIHNjcmVlbnNob3Q6IHN0cmluZztcbiAgICB2aWV3cG9ydDoge1xuICAgICAgICB3aWR0aDogbnVtYmVyO1xuICAgICAgICBoZWlnaHQ6IG51bWJlcjtcbiAgICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNlc3Npb25EYXRhIHtcbiAgICBzZXNzaW9uSWQ6IHN0cmluZztcbiAgICB1c2VyQWdlbnQ6IHN0cmluZztcbn1cbiJdfQ==
|
package/package.json
ADDED
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "journey-mapper-tracker",
|
|
3
|
+
"version": "1.0.0",
|
|
4
|
+
"description": "Track user page flows and capture screenshots for UX research - Journey Mapper SDK",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"module": "dist/index.esm.js",
|
|
7
|
+
"types": "dist/index.d.ts",
|
|
8
|
+
"exports": {
|
|
9
|
+
".": {
|
|
10
|
+
"import": "./dist/index.js",
|
|
11
|
+
"require": "./dist/index.js",
|
|
12
|
+
"types": "./dist/index.d.ts"
|
|
13
|
+
}
|
|
14
|
+
},
|
|
15
|
+
"scripts": {
|
|
16
|
+
"build": "tsc",
|
|
17
|
+
"watch": "tsc --watch",
|
|
18
|
+
"prepublishOnly": "npm run build"
|
|
19
|
+
},
|
|
20
|
+
"files": [
|
|
21
|
+
"dist",
|
|
22
|
+
"README.md"
|
|
23
|
+
],
|
|
24
|
+
"keywords": [
|
|
25
|
+
"journey-mapper",
|
|
26
|
+
"user-flow",
|
|
27
|
+
"tracking",
|
|
28
|
+
"screenshots",
|
|
29
|
+
"ux-research",
|
|
30
|
+
"user-journey",
|
|
31
|
+
"heatmap",
|
|
32
|
+
"analytics",
|
|
33
|
+
"session-replay",
|
|
34
|
+
"page-flow"
|
|
35
|
+
],
|
|
36
|
+
"author": "Journey Mapper",
|
|
37
|
+
"license": "MIT",
|
|
38
|
+
"repository": {
|
|
39
|
+
"type": "git",
|
|
40
|
+
"url": "https://github.com/journey-mapper/tracker"
|
|
41
|
+
},
|
|
42
|
+
"homepage": "https://journeymapper.com",
|
|
43
|
+
"bugs": {
|
|
44
|
+
"url": "https://github.com/journey-mapper/tracker/issues"
|
|
45
|
+
},
|
|
46
|
+
"peerDependencies": {
|
|
47
|
+
"react": ">=17.0.0"
|
|
48
|
+
},
|
|
49
|
+
"peerDependenciesMeta": {
|
|
50
|
+
"react": {
|
|
51
|
+
"optional": true
|
|
52
|
+
}
|
|
53
|
+
},
|
|
54
|
+
"dependencies": {
|
|
55
|
+
"html2canvas": "^1.4.1"
|
|
56
|
+
},
|
|
57
|
+
"devDependencies": {
|
|
58
|
+
"@types/node": "^20.0.0",
|
|
59
|
+
"typescript": "^5.0.0"
|
|
60
|
+
},
|
|
61
|
+
"engines": {
|
|
62
|
+
"node": ">=16.0.0"
|
|
63
|
+
}
|
|
64
|
+
}
|