@growth-rail/react-native 2.0.1 → 2.0.2
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/CHANGELOG.md +6 -0
- package/README.md +28 -1
- package/dist/index.d.mts +4 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.js +1 -1
- package/dist/index.mjs +1 -1
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
package/README.md
CHANGED
|
@@ -34,6 +34,8 @@ const App = () => {
|
|
|
34
34
|
<GrowthRailProvider
|
|
35
35
|
projectSecretKey="YOUR_PROJECT_SECRET_KEY"
|
|
36
36
|
userId="USER_UNIQUE_ID" // optional at start
|
|
37
|
+
appearance="dark" // "light", "dark", or omit for system default
|
|
38
|
+
debug={true} // Enable verbose console logging
|
|
37
39
|
>
|
|
38
40
|
<YourAppContent />
|
|
39
41
|
</GrowthRailProvider>
|
|
@@ -132,7 +134,10 @@ const MyProfile = () => {
|
|
|
132
134
|
const { showReferralDashboard } = useGrowthRail();
|
|
133
135
|
|
|
134
136
|
return (
|
|
135
|
-
<Button
|
|
137
|
+
<Button
|
|
138
|
+
title="Refer & Earn"
|
|
139
|
+
onPress={() => showReferralDashboard({ appearance: 'dark' })}
|
|
140
|
+
/>
|
|
136
141
|
);
|
|
137
142
|
};
|
|
138
143
|
```
|
|
@@ -150,6 +155,19 @@ const getLink = async () => {
|
|
|
150
155
|
};
|
|
151
156
|
```
|
|
152
157
|
|
|
158
|
+
## Theme & Appearance
|
|
159
|
+
|
|
160
|
+
The SDK supports light and dark modes out of the box.
|
|
161
|
+
|
|
162
|
+
- **System Default**: If no appearance is specified, the SDK follows the user's system setting using `useColorScheme()`.
|
|
163
|
+
- **Global Setting**: Set the `appearance` prop on `GrowthRailProvider`.
|
|
164
|
+
- **Manual Override**: Pass `appearance` to `showReferralDashboard({ appearance: 'dark' })`.
|
|
165
|
+
|
|
166
|
+
### Standard Colors
|
|
167
|
+
The SDK uses a premium, neutral color palette designed to fit seamlessly into any application:
|
|
168
|
+
- **Light**: Clean white background with high-contrast Gray 900 text.
|
|
169
|
+
- **Dark**: Deep Gray 900 background with soft Gray 50 text for reduced eye strain and optimal readability.
|
|
170
|
+
|
|
153
171
|
## API Reference
|
|
154
172
|
|
|
155
173
|
### `GrowthRailProvider` Props
|
|
@@ -157,8 +175,17 @@ const getLink = async () => {
|
|
|
157
175
|
| :--- | :--- | :--- |
|
|
158
176
|
| `projectSecretKey` | `string` | Your GrowthRail Project Secret Key. |
|
|
159
177
|
| `userId` | `string` | (Optional) The current user's unique ID. |
|
|
178
|
+
| `appearance` | `'light' \| 'dark'` | (Optional) Force a specific theme. Defaults to system. |
|
|
179
|
+
| `debug` | `boolean` | (Optional) Enable verbose console logging. |
|
|
160
180
|
| `apiUrl` | `string` | (Optional) Custom API endpoint. |
|
|
161
181
|
|
|
182
|
+
### `ReferrerModalOptions`
|
|
183
|
+
Options passed to `showReferralDashboard(options)`:
|
|
184
|
+
- `title`: `string` - Override the dashboard title.
|
|
185
|
+
- `description`: `string` - Override the dashboard description.
|
|
186
|
+
- `appearance`: `'light' \| 'dark'` - Override the appearance for this call.
|
|
187
|
+
- `theme.primaryColor`: `string` - Your brand color (e.g., `#2563eb`).
|
|
188
|
+
|
|
162
189
|
### `useGrowthRail()`
|
|
163
190
|
Provides access to SDK methods and state:
|
|
164
191
|
- `initAppUser(userId)`: Initializes a user in the GrowthRail system.
|
package/dist/index.d.mts
CHANGED
|
@@ -58,6 +58,8 @@ interface NewUserExperience {
|
|
|
58
58
|
interface GrowthRailProviderProps extends GrowthRailOptions {
|
|
59
59
|
/** The unique ID for the current application user. */
|
|
60
60
|
userId?: string;
|
|
61
|
+
/** Enable the debug overlay for inspecting SDK state, events, and API calls. */
|
|
62
|
+
debug?: boolean;
|
|
61
63
|
/** Your application components. */
|
|
62
64
|
children: React.ReactNode;
|
|
63
65
|
}
|
|
@@ -107,6 +109,8 @@ interface UseGrowthRailReturn {
|
|
|
107
109
|
isUserReady: boolean;
|
|
108
110
|
/** The last error that occurred during an operation. */
|
|
109
111
|
error: Error | null;
|
|
112
|
+
/** Whether the debug overlay is enabled. */
|
|
113
|
+
isDebugEnabled: boolean;
|
|
110
114
|
}
|
|
111
115
|
/**
|
|
112
116
|
* A hook that provides access to GrowthRail SDK methods and state in React Native.
|
package/dist/index.d.ts
CHANGED
|
@@ -58,6 +58,8 @@ interface NewUserExperience {
|
|
|
58
58
|
interface GrowthRailProviderProps extends GrowthRailOptions {
|
|
59
59
|
/** The unique ID for the current application user. */
|
|
60
60
|
userId?: string;
|
|
61
|
+
/** Enable the debug overlay for inspecting SDK state, events, and API calls. */
|
|
62
|
+
debug?: boolean;
|
|
61
63
|
/** Your application components. */
|
|
62
64
|
children: React.ReactNode;
|
|
63
65
|
}
|
|
@@ -107,6 +109,8 @@ interface UseGrowthRailReturn {
|
|
|
107
109
|
isUserReady: boolean;
|
|
108
110
|
/** The last error that occurred during an operation. */
|
|
109
111
|
error: Error | null;
|
|
112
|
+
/** Whether the debug overlay is enabled. */
|
|
113
|
+
isDebugEnabled: boolean;
|
|
110
114
|
}
|
|
111
115
|
/**
|
|
112
116
|
* A hook that provides access to GrowthRail SDK methods and state in React Native.
|
package/dist/index.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";var Ce=Object.create;var _=Object.defineProperty;var ke=Object.getOwnPropertyDescriptor;var Te=Object.getOwnPropertyNames;var xe=Object.getPrototypeOf,ve=Object.prototype.hasOwnProperty;var Le=(o,e,a)=>e in o?_(o,e,{enumerable:!0,configurable:!0,writable:!0,value:a}):o[e]=a;var Ie=(o,e)=>{for(var a in e)_(o,a,{get:e[a],enumerable:!0})},ue=(o,e,a,s)=>{if(e&&typeof e=="object"||typeof e=="function")for(let l of Te(e))!ve.call(o,l)&&l!==a&&_(o,l,{get:()=>e[l],enumerable:!(s=ke(e,l))||s.enumerable});return o};var W=(o,e,a)=>(a=o!=null?Ce(xe(o)):{},ue(e||!o||!o.__esModule?_(a,"default",{value:o,enumerable:!0}):a,o)),Se=o=>ue(_({},"__esModule",{value:!0}),o);var ce=(o,e,a)=>Le(o,typeof e!="symbol"?e+"":e,a);var Be={};Ie(Be,{GrowthRailProvider:()=>be,ReferralDashboard:()=>te,TriggerButton:()=>oe,useGrowthRail:()=>ye});module.exports=Se(Be);var n=W(require("react")),C=require("react-native"),we=W(require("react-native-play-install-referrer"));var Z=class{constructor(e){ce(this,"baseUrl");ce(this,"projectSecretKey");let a=typeof process<"u"&&process.env?.ENV_BASE_URL;this.baseUrl=a||"https://api.growthrail.dev",this.projectSecretKey=e.projectSecretKey}async request(e,a,s){let l={"Content-Type":"application/json","X-GrowthRail-ProjectSecret":this.projectSecretKey,Origin:this.baseUrl};try{let f=await fetch(`${this.baseUrl}${e}`,{method:a,headers:l,body:s?JSON.stringify(s):void 0});if(!f.ok){let k="";try{k=await f.text()}catch{}throw new Error(`GrowthRail API Error ${f.status} (${this.baseUrl}${e}): ${f.statusText} - ${k}`)}return f.json()}catch(f){throw console.error("GrowthRail SDK Error:",f),f}}async getNewUserExperience(){return this.request("/api/v1/sdk/new-user-experience","POST")}async initAppUser(e){return this.request("/api/v1/sdk/init-app-user","POST",{clientProvidedId:e})}async trackReferral(e,a){return this.request("/api/v1/sdk/track-referral","POST",{referralCode:e,rewardEventName:a})}async trackRewardEvent(e,a,s){return this.request("/api/v1/sdk/track-reward-event","POST",{refereeId:e,referralTrackingId:a,eventName:s})}async checkDeferredDeepLink(e){return this.request("/api/v1/sdk/match-link","POST",{fingerprint:e})}};var K=W(require("@react-native-async-storage/async-storage")),ee={TRACKED_REFERRAL:"growth-rail-tracked-referral",IS_FIRST_LAUNCH:"growth-rail-is-first-launch"},E=class{static async setTrackedReferral(e){await K.default.setItem(ee.TRACKED_REFERRAL,e)}static async getTrackedReferral(){return await K.default.getItem(ee.TRACKED_REFERRAL)}static async isFirstLaunch(){return await K.default.getItem(ee.IS_FIRST_LAUNCH)===null?(await K.default.setItem(ee.IS_FIRST_LAUNCH,"false"),!0):!1}};var r=W(require("react")),t=require("react-native");var{width:pe,height:de}=t.Dimensions.get("window"),te=({visible:o,options:e,onClose:a})=>{let s=re(),[l,f]=(0,r.useState)(null),[k,d]=(0,r.useState)(!1),[p,m]=(0,r.useState)(!1),g=(0,r.useRef)(new t.Animated.Value(0)).current,T=(0,r.useRef)(new t.Animated.Value(0)).current,z=e?.title||"Invite Friends & Earn",B=e?.theme?.primaryColor||"#2563eb",M=e?.theme?.backgroundColor||"#ffffff",ie=e?.theme?.tintColor||"#000000",ne=e?.theme?.tintAlpha??.5,v=(e?.componentType||"modal")==="drawer",U=e?.position||"bottom-right",Y="#111827",j="#6b7280";(0,r.useEffect)(()=>{o?(l||ae(),t.Animated.parallel([t.Animated.timing(T,{toValue:1,duration:300,useNativeDriver:!0}),t.Animated.spring(g,{toValue:1,useNativeDriver:!0,tension:50,friction:8})]).start()):(g.setValue(0),T.setValue(0))},[o]);let $=()=>{t.Animated.parallel([t.Animated.timing(T,{toValue:0,duration:250,useNativeDriver:!0}),t.Animated.timing(g,{toValue:0,duration:250,useNativeDriver:!0})]).start(()=>{a()})},ae=async()=>{d(!0);try{if(s.isUserReady){let G=await s.getReferralLink();f(G||null)}}catch(G){console.error("GrowthRail: Failed to load referral link",G)}finally{d(!1)}},J=()=>{l&&(t.Clipboard.setString(l),m(!0),setTimeout(()=>m(!1),2e3))},V=G=>{if(!l)return;let D="",I=encodeURIComponent(l),H=encodeURIComponent("Check this out!");switch(G){case"twitter":D=`https://twitter.com/intent/tweet?url=${I}&text=${H}`;break;case"facebook":D=`https://www.facebook.com/sharer/sharer.php?u=${I}`;break;case"linkedin":D=`https://www.linkedin.com/sharing/share-offsite/?url=${I}`;break;case"whatsapp":D=`whatsapp://send?text=${H}%20${I}`;break;case"email":D=`mailto:?subject=Check this out&body=${H}%20${I}`;break}D&&t.Linking.openURL(D).catch(()=>{})},Q=()=>ie||"#000000",O=[];v?U.includes("right")?O=[{translateX:g.interpolate({inputRange:[0,1],outputRange:[pe,0]})}]:U.includes("left")?O=[{translateX:g.interpolate({inputRange:[0,1],outputRange:[-pe,0]})}]:O=[{translateY:g.interpolate({inputRange:[0,1],outputRange:[de,0]})}]:O=[{scale:g.interpolate({inputRange:[0,1],outputRange:[.9,1]})},{translateY:g.interpolate({inputRange:[0,1],outputRange:[20,0]})}];let ge=U.includes("top"),L=U.includes("right"),A=U.includes("left"),se=v?{position:"absolute",...ge?{top:100}:{bottom:100},...L?{right:0}:A?{left:0}:{left:0,right:0},width:L||A?"85%":"100%",maxHeight:L||A?de-200:"80%"}:{};return r.default.createElement(t.Modal,{visible:o,transparent:!0,animationType:"none",onRequestClose:$},r.default.createElement(t.View,{style:i.fullScreen},r.default.createElement(t.Animated.View,{style:[i.overlay,{backgroundColor:Q(),opacity:T.interpolate({inputRange:[0,1],outputRange:[0,ne]})}]},r.default.createElement(t.TouchableOpacity,{style:i.fullScreen,activeOpacity:1,onPress:$})),r.default.createElement(t.Animated.View,{style:[v?se:i.centeredModalContainer,{backgroundColor:M,transform:O},v&&L&&{borderTopLeftRadius:24,borderBottomLeftRadius:24},v&&A&&{borderTopRightRadius:24,borderBottomRightRadius:24},v&&!L&&!A&&U.includes("bottom")&&{borderTopLeftRadius:24,borderTopRightRadius:24},v&&!L&&!A&&U.includes("top")&&{borderBottomLeftRadius:24,borderBottomRightRadius:24},!v&&{borderRadius:24,opacity:g}]},r.default.createElement(t.View,{style:i.contentWrapper},v&&!L&&!A&&U.includes("bottom")&&r.default.createElement(t.View,{style:i.drawerHandle}),r.default.createElement(t.TouchableOpacity,{style:i.closeBtn,onPress:$},r.default.createElement(t.Text,{style:i.closeIcon},"\u2715")),r.default.createElement(t.ScrollView,{contentContainerStyle:i.scrollContent,showsVerticalScrollIndicator:!1},r.default.createElement(t.View,{style:i.header},r.default.createElement(t.View,{style:i.iconLgContainer},r.default.createElement(t.View,{style:i.bowContainerLg},r.default.createElement(t.View,{style:[i.bowLoopLg,i.bowLeftLg,{borderColor:B}]}),r.default.createElement(t.View,{style:[i.bowLoopLg,i.bowRightLg,{borderColor:B}]})),r.default.createElement(t.View,{style:[i.giftLidLg,{backgroundColor:B}]}),r.default.createElement(t.View,{style:[i.giftBodyLg,{borderColor:B}]},r.default.createElement(t.View,{style:[i.ribbonVerticalLg,{backgroundColor:"#fff",opacity:.3}]}))),r.default.createElement(t.Text,{style:[i.title,{color:Y}]},z),r.default.createElement(t.Text,{style:[i.subtitle,{color:j}]},e?.description||"Share your unique link and earn rewards for every friend who signs up.")),s.isUserReady?k?r.default.createElement(t.ActivityIndicator,{size:"large",color:B,style:i.loader}):r.default.createElement(t.View,{style:i.mainContent},r.default.createElement(t.Text,{style:i.label},"YOUR UNIQUE LINK"),r.default.createElement(t.View,{style:i.inputGroup},r.default.createElement(t.Text,{numberOfLines:1,style:i.input},l||"Initializing..."),r.default.createElement(t.TouchableOpacity,{style:[i.copyBtn,{backgroundColor:p?"#10b981":B}],onPress:J},r.default.createElement(t.Text,{style:i.copyBtnText},p?"Copied!":"Copy"))),r.default.createElement(t.View,{style:i.shareLabelContainer},r.default.createElement(t.View,{style:i.divider}),r.default.createElement(t.Text,{style:i.shareLabel},"Share via"),r.default.createElement(t.View,{style:i.divider})),r.default.createElement(t.View,{style:i.socialContainer},r.default.createElement(q,{platform:"twitter",color:"#000000",onPress:()=>V("twitter"),icon:"\u{1D54F}"}),r.default.createElement(q,{platform:"facebook",color:"#1877F2",onPress:()=>V("facebook"),icon:"f"}),r.default.createElement(q,{platform:"linkedin",color:"#0A66C2",onPress:()=>V("linkedin"),icon:"in"}),r.default.createElement(q,{platform:"whatsapp",color:"#25D366",onPress:()=>V("whatsapp"),icon:"W"}),r.default.createElement(q,{platform:"email",color:"#888888",onPress:()=>V("email"),icon:"\u2709"}))):r.default.createElement(t.View,{style:i.loadingBox},r.default.createElement(t.Text,{style:{color:j,textAlign:"center"}},"Please log in to view your referral dashboard.")),r.default.createElement(t.View,{style:i.footer},r.default.createElement(t.Text,{style:i.poweredBy},"Powered by ",r.default.createElement(t.Text,{style:i.brand,onPress:()=>t.Linking.openURL("https://growthrail.com")},"Growth Rail"))))))))},q=({color:o,onPress:e,icon:a})=>r.default.createElement(t.TouchableOpacity,{style:[i.socialBtn,{borderColor:"#e5e7eb"}],onPress:e},r.default.createElement(t.Text,{style:[i.socialIcon,{color:o}]},a)),i=t.StyleSheet.create({fullScreen:{flex:1},overlay:{...t.StyleSheet.absoluteFillObject},centeredModalContainer:{width:"90%",maxWidth:400,alignSelf:"center",marginTop:de*.15,overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:20},shadowOpacity:.25,shadowRadius:30,elevation:20},contentWrapper:{paddingTop:t.Platform.OS==="ios"?20:0},drawerHandle:{width:40,height:4,backgroundColor:"#e5e7eb",borderRadius:2,alignSelf:"center",marginTop:12},scrollContent:{padding:32,alignItems:"center"},closeBtn:{position:"absolute",top:20,right:20,zIndex:10,width:32,height:32,borderRadius:16,backgroundColor:"#f3f4f6",justifyContent:"center",alignItems:"center"},closeIcon:{fontSize:14,color:"#9ca3af"},header:{alignItems:"center",marginBottom:28,width:"100%"},socialIcon:{fontSize:18,fontWeight:"bold"},title:{fontSize:22,fontWeight:"700",letterSpacing:-.5,marginBottom:8,textAlign:"center"},subtitle:{fontSize:14,lineHeight:20,textAlign:"center",paddingHorizontal:10},loadingBox:{padding:16,backgroundColor:"#f9fafb",borderRadius:12,width:"100%",alignItems:"center",borderWidth:1,borderColor:"#e5e7eb",marginTop:20},loader:{marginVertical:20},mainContent:{width:"100%"},label:{fontSize:12,fontWeight:"600",color:"#6b7280",marginBottom:8,letterSpacing:.5},inputGroup:{flexDirection:"row",alignItems:"center",backgroundColor:"#f9fafb",borderWidth:1,borderColor:"#e5e7eb",borderRadius:12,padding:6,marginBottom:24},input:{flex:1,paddingHorizontal:12,fontSize:14,color:"#111827",fontFamily:t.Platform.OS==="ios"?"Menlo":"monospace"},copyBtn:{paddingVertical:8,paddingHorizontal:16,borderRadius:8,justifyContent:"center",alignItems:"center"},copyBtnText:{color:"#fff",fontSize:13,fontWeight:"600"},shareLabelContainer:{flexDirection:"row",alignItems:"center",marginBottom:16,marginTop:8},divider:{flex:1,height:1,backgroundColor:"#e5e7eb"},shareLabel:{marginHorizontal:12,fontSize:13,fontWeight:"500",color:"#6b7280"},socialContainer:{flexDirection:"row",justifyContent:"center",gap:12},socialBtn:{width:44,height:44,borderRadius:22,backgroundColor:"#fff",borderWidth:1,justifyContent:"center",alignItems:"center"},iconLgContainer:{width:60,height:60,justifyContent:"center",alignItems:"center",marginBottom:16},bowContainerLg:{flexDirection:"row",position:"absolute",top:2,zIndex:2},bowLoopLg:{width:20,height:16,borderWidth:3,borderColor:"#e5e7eb",borderRadius:10},bowLeftLg:{marginRight:-2,transform:[{rotate:"-15deg"}]},bowRightLg:{marginLeft:-2,transform:[{rotate:"15deg"}]},giftLidLg:{width:44,height:10,backgroundColor:"#f3f4f6",borderRadius:2,marginTop:12,zIndex:3},giftBodyLg:{width:36,height:24,borderWidth:3,borderTopWidth:0,borderBottomLeftRadius:4,borderBottomRightRadius:4,justifyContent:"center",alignItems:"center",marginTop:-2},ribbonVerticalLg:{width:4,height:"100%",opacity:.1},footer:{marginTop:32,alignItems:"center"},poweredBy:{fontSize:11,color:"#9ca3af"},brand:{fontWeight:"600",textDecorationLine:"underline"}});var R=W(require("react")),h=require("react-native"),oe=({onPress:o,options:e})=>{let a=e?.modal?.theme?.primaryColor||"#2563eb",s=e?.trigger?.position||"bottom-right",f=(e?.trigger?.displayMode||"floating")==="edge",k=s.includes("right"),d=s.includes("top"),p=(0,R.useRef)(new h.Animated.Value(-1)).current;(0,R.useEffect)(()=>{let z=()=>{p.setValue(-1),h.Animated.timing(p,{toValue:2,duration:3e3,easing:h.Easing.linear,useNativeDriver:!0}).start(()=>{setTimeout(z,1e3)})};z()},[]);let m=p.interpolate({inputRange:[-1,2],outputRange:f?[-50,100]:[-60,120]}),g={"bottom-right":{bottom:100,right:0},"bottom-left":{bottom:100,left:0},"top-right":{top:100,right:0},"top-left":{top:100,left:0}};f||(g["bottom-right"]={bottom:30,right:20},g["bottom-left"]={bottom:30,left:20},g["top-right"]={top:60,right:20},g["top-left"]={top:60,left:20});let T=()=>R.default.createElement(h.View,{style:y.iconContainer},R.default.createElement(h.View,{style:y.bowContainer},R.default.createElement(h.View,{style:[y.bowLoop,y.bowLeft]}),R.default.createElement(h.View,{style:[y.bowLoop,y.bowRight]})),R.default.createElement(h.View,{style:y.giftLid}),R.default.createElement(h.View,{style:y.giftBody},R.default.createElement(h.View,{style:y.ribbonVertical})));return R.default.createElement(h.TouchableOpacity,{style:[f?y.edgeButton:y.floatingButton,{backgroundColor:a},g[s],f&&(k?y.edgeRight:y.edgeLeft)],onPress:o,activeOpacity:.9},R.default.createElement(h.View,{style:y.content},T()),R.default.createElement(h.Animated.View,{style:[y.shimmer,{transform:[{translateX:m},{skewX:"-15deg"}]}]}))},y=h.StyleSheet.create({floatingButton:{position:"absolute",width:60,height:60,borderRadius:30,justifyContent:"center",alignItems:"center",overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:8},shadowOpacity:.2,shadowRadius:12,elevation:8,zIndex:2147483646},edgeButton:{position:"absolute",width:50,height:50,justifyContent:"center",alignItems:"center",overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:4},shadowOpacity:.1,shadowRadius:8,elevation:4,zIndex:2147483646},edgeRight:{borderTopLeftRadius:8,borderBottomLeftRadius:8,paddingLeft:4},edgeLeft:{borderTopRightRadius:8,borderBottomRightRadius:8,paddingRight:4},content:{justifyContent:"center",alignItems:"center"},iconContainer:{width:28,height:28,justifyContent:"center",alignItems:"center"},bowContainer:{flexDirection:"row",position:"absolute",top:0,zIndex:2},bowLoop:{width:10,height:8,borderWidth:2,borderColor:"#fff",borderRadius:5},bowLeft:{marginRight:-1,transform:[{rotate:"-15deg"}]},bowRight:{marginLeft:-1,transform:[{rotate:"15deg"}]},giftLid:{width:22,height:5,backgroundColor:"#fff",borderRadius:1,marginTop:6,zIndex:3},giftBody:{width:18,height:12,borderWidth:2,borderColor:"#fff",borderTopWidth:0,borderBottomLeftRadius:2,borderBottomRightRadius:2,justifyContent:"center",alignItems:"center",marginTop:-1},ribbonVertical:{width:2,height:"100%",backgroundColor:"#fff",opacity:.8},shimmer:{position:"absolute",top:0,left:0,width:50,height:"100%",backgroundColor:"rgba(255, 255, 255, 0.25)"}});var x=W(require("react")),w=require("react-native"),{width:Pe}=w.Dimensions.get("window"),he=({visible:o,options:e,onClose:a})=>{let s=(0,x.useRef)(new w.Animated.Value(0)).current,[l,f]=(0,x.useState)(o),k=e.position.includes("bottom")?"bottom":"top",d=e.position.includes("left")?"left":e.position.includes("right")?"right":"center";if((0,x.useEffect)(()=>{o?(f(!0),w.Animated.spring(s,{toValue:1,useNativeDriver:!0,tension:50,friction:8}).start()):w.Animated.timing(s,{toValue:0,duration:250,useNativeDriver:!0}).start(()=>{f(!1)})},[o]),!l)return null;let p=s.interpolate({inputRange:[0,1],outputRange:[k==="top"?-100:100,0]}),m=s.interpolate({inputRange:[0,1],outputRange:[0,1]}),g={position:"absolute",[k]:w.Platform.OS==="ios"?60:40,zIndex:9999};return d==="center"?(g.left=20,g.right=20,g.alignItems="center"):g[d]=20,x.default.createElement(w.Animated.View,{style:[g,{transform:[{translateY:p}],opacity:m}]},x.default.createElement(w.View,{style:[X.banner,{backgroundColor:e.themeColor||"#2563eb"}]},x.default.createElement(w.Text,{style:X.icon},"\u{1F381}"),x.default.createElement(w.Text,{style:X.text},e.text),x.default.createElement(w.TouchableOpacity,{onPress:a,style:X.closeBtn},x.default.createElement(w.Text,{style:X.closeIcon},"\u2715"))))},X=w.StyleSheet.create({banner:{flexDirection:"row",alignItems:"center",padding:12,paddingHorizontal:16,borderRadius:12,shadowColor:"#000",shadowOffset:{width:0,height:4},shadowOpacity:.15,shadowRadius:10,elevation:5,maxWidth:Pe-40},icon:{fontSize:18,marginRight:10},text:{color:"#fff",fontSize:14,fontWeight:"600",flex:1},closeBtn:{marginLeft:10,padding:4},closeIcon:{color:"rgba(255, 255, 255, 0.8)",fontSize:14,fontWeight:"bold"}});var me=(0,n.createContext)(void 0),be=({children:o,userId:e,...a})=>{let[s,l]=(0,n.useState)(!1),[f,k]=(0,n.useState)(!1),[d,p]=(0,n.useState)(null),m=(0,n.useRef)(new Z(a)),g=(0,n.useRef)(!1),T=(0,n.useRef)(null),[z,B]=(0,n.useState)(!1),[M,ie]=(0,n.useState)(),[ne,fe]=(0,n.useState)(!1),[v,U]=(0,n.useState)(),[Y,j]=(0,n.useState)(!1),[$,ae]=(0,n.useState)(),J=(0,n.useCallback)(()=>{let{width:u,height:c}=C.Dimensions.get("window"),b=Intl.DateTimeFormat().resolvedOptions().locale,S=C.PixelRatio.get(),P,F;if(C.Platform.OS==="android"){let le=C.Platform.constants;P=le.Model,F=le.Brand||le.Manufacturer}return{os:C.Platform.OS,platform:C.Platform.OS,osVersion:String(C.Platform.Version),screenWidth:u,screenHeight:c,timezoneOffsetInMinutes:String(new Date().getTimezoneOffset()),locale:b,language:b,deviceModel:P,deviceBrand:F,pixelRatio:S}},[]),V=(0,n.useCallback)(async(u,c)=>{let b=await m.current.trackReferral(u,c);return await E.setTrackedReferral(b.referralTrackingId),b},[]),Q=(0,n.useCallback)(async u=>{try{let c=u.match(/[?&]referralCode=([^&]+)/);c&&(console.log("GrowthRail: Tracking referral from deep link",c[1]),await V(c[1]))}catch(c){console.warn("GrowthRail: Error handling deep link",c)}},[V]),O=(0,n.useCallback)(async()=>{try{let u=J(),{referralTrackingId:c}=await m.current.checkDeferredDeepLink(u);c&&(console.log("GrowthRail: Found deferred referral via fingerprinting",c),await E.setTrackedReferral(c))}catch(u){console.warn("GrowthRail: Fingerprint check failed",u)}},[J]),ge=(0,n.useCallback)(()=>new Promise(u=>{we.default.getInstallReferrerInfo(async(c,b)=>{if(!b&&c&&c.installReferrer)try{let P=decodeURIComponent(c.installReferrer).match(/[?&]referralCode=([^&]+)/);if(P){let{referralTrackingId:F}=await V(P[1]);await E.setTrackedReferral(F),console.log("GrowthRail: Tracked referral via Install Referrer")}}catch(S){console.warn("GrowthRail: Failed to process install referrer",S)}else await O();u()})}),[V,O]),L=(0,n.useCallback)(async u=>{if(d?.id===u&&f)return d;let c=(async()=>{try{let b=await m.current.initAppUser(u),S={...b,id:u};return p(S),k(!0),b.referrerExperience?.trigger?.show&&fe(!0),S}catch(b){throw T.current=null,b}})();return T.current=c,c},[d,f]),A=(0,n.useCallback)(async u=>{let c=d;!c&&T.current&&(c=await T.current);let b=c?.id||e;if(!b)throw new Error("GrowthRail: User must be initialized before tracking a reward event.");let S=await E.getTrackedReferral();if(!S){console.warn("GrowthRail: No referral tracking ID found. Reward event not attributed.");return}return m.current.trackRewardEvent(b,S,u)},[d,e]),se=(0,n.useCallback)(async()=>await E.getTrackedReferral(),[]),G=(0,n.useCallback)(async()=>{if(d)return d.referralLink;if(T.current)return(await T.current).referralLink},[d]);(0,n.useEffect)(()=>{if(g.current)return;g.current=!0,(async()=>{let c=await C.Linking.getInitialURL();c&&await Q(c);let b=C.Linking.addEventListener("url",({url:P})=>Q(P));await E.isFirstLaunch()&&(C.Platform.OS==="android"?await O():await O()),e&&await L(e);try{let P=await m.current.getNewUserExperience(),F=await E.getTrackedReferral();P.banner?.show&&F&&(ae(P.banner),j(!0))}catch{}return l(!0),()=>b.remove()})()},[]),(0,n.useEffect)(()=>{s&&e&&d?.id!==e&&L(e)},[e,s]);let D={isInitialized:s,isUserReady:f,currentUser:d,referralCode:d?.referralCode,referralLink:d?.referralLink,showDashboard:u=>{ie(u),B(!0)},hideDashboard:()=>B(!1),initUser:L,trackReward:A,getReferralTrackingId:se,getReferralLink:G};if(!s)return null;let I=d?.referrerExperience,H={...I?.modal,...M,position:I?.trigger?.position||M?.position||"bottom-right",theme:{...I?.modal?.theme||{},...M?.theme||{}}},Re={...I,...v,trigger:{...I?.trigger||{},...v?.trigger||{}}};return n.default.createElement(me.Provider,{value:D},o,n.default.createElement(te,{visible:z,options:H,onClose:()=>B(!1)}),ne&&n.default.createElement(oe,{options:Re,onPress:()=>D.showDashboard()}),Y&&n.default.createElement(he,{visible:Y,options:$,onClose:()=>j(!1)}))},re=()=>{let o=(0,n.useContext)(me);if(!o)throw new Error("useGrowthRail must be used within a GrowthRailProvider");return o};var N=require("react");var ye=()=>{let[o,e]=(0,N.useState)(!1),[a,s]=(0,N.useState)(null),l=re(),f=(0,N.useCallback)(async d=>{e(!0),s(null);try{return await l.initUser(d)}catch(p){let m=p instanceof Error?p:new Error(String(p));throw s(m),m}finally{e(!1)}},[l]),k=(0,N.useCallback)(async d=>{e(!0),s(null);try{await l.trackReward(d)}catch(p){let m=p instanceof Error?p:new Error(String(p));throw s(m),m}finally{e(!1)}},[l]);return{initAppUser:f,trackRewardEvent:k,showReferralDashboard:l.showDashboard,hideReferralDashboard:l.hideDashboard,getReferralTrackingId:l.getReferralTrackingId,getReferralLink:l.getReferralLink,isLoading:o,isInitialized:l.isInitialized,isUserReady:l.isUserReady,error:a}};0&&(module.exports={GrowthRailProvider,ReferralDashboard,TriggerButton,useGrowthRail});
|
|
1
|
+
"use strict";var Te=Object.create;var Y=Object.defineProperty;var xe=Object.getOwnPropertyDescriptor;var Le=Object.getOwnPropertyNames;var ve=Object.getPrototypeOf,Ie=Object.prototype.hasOwnProperty;var Se=(i,e,t)=>e in i?Y(i,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):i[e]=t;var De=(i,e)=>{for(var t in e)Y(i,t,{get:e[t],enumerable:!0})},he=(i,e,t,c)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of Le(e))!Ie.call(i,a)&&a!==t&&Y(i,a,{get:()=>e[a],enumerable:!(c=xe(e,a))||c.enumerable});return i};var _=(i,e,t)=>(t=i!=null?Te(ve(i)):{},he(e||!i||!i.__esModule?Y(t,"default",{value:i,enumerable:!0}):t,i)),Be=i=>he(Y({},"__esModule",{value:!0}),i);var ee=(i,e,t)=>Se(i,typeof e!="symbol"?e+"":e,t);var Ue={};De(Ue,{GrowthRailProvider:()=>ye,ReferralDashboard:()=>oe,TriggerButton:()=>ne,useGrowthRail:()=>Re});module.exports=Be(Ue);var s=_(require("react")),T=require("react-native"),me=_(require("react-native-play-install-referrer"));var Oe="[GrowthRail]",f={log(i,e,t,c){let a=`${Oe} [${e}] ${t}`;switch(i){case"error":console.error(a,c||"");break;case"warn":console.warn(a,c||"");break;default:console.log(a,c||"");break}}};var te=class{constructor(e){ee(this,"baseUrl");ee(this,"projectSecretKey");ee(this,"_debug",!1);let t=typeof process<"u"&&process.env?.ENV_BASE_URL;this.baseUrl=t||"https://api.growthrail.dev",this.projectSecretKey=e.projectSecretKey}setDebug(e){this._debug=e}async request(e,t,c){let a={"Content-Type":"application/json","X-GrowthRail-ProjectSecret":this.projectSecretKey,Origin:this.baseUrl},y=Date.now();this._debug&&f.log("api","APIClient",`${t} ${e}`,{method:t,endpoint:e,body:c??null});try{let h=await fetch(`${this.baseUrl}${e}`,{method:t,headers:a,body:c?JSON.stringify(c):void 0}),b=Date.now()-y;if(!h.ok){let l="";try{l=await h.text()}catch{}let R=`GrowthRail API Error ${h.status} (${this.baseUrl}${e}): ${h.statusText} - ${l}`;throw this._debug&&f.log("api","APIClient",`\u2717 ${t} ${e}`,{method:t,endpoint:e,status:h.status,duration:b,error:l}),new Error(R)}return this._debug&&f.log("api","APIClient",`\u2713 ${t} ${e}`,{method:t,endpoint:e,status:h.status,duration:b}),h.json()}catch(h){let b=Date.now()-y;throw this._debug&&f.log("api","APIClient",`\u2717 ${t} ${e} \u2014 Network error`,{method:t,endpoint:e,duration:b,error:String(h)}),console.error("GrowthRail SDK Error:",h),h}}async getNewUserExperience(){return this.request("/api/v1/sdk/new-user-experience","POST")}async initAppUser(e){return this.request("/api/v1/sdk/init-app-user","POST",{clientProvidedId:e})}async trackReferral(e,t){return this.request("/api/v1/sdk/track-referral","POST",{referralCode:e,rewardEventName:t})}async trackRewardEvent(e,t,c){return this.request("/api/v1/sdk/track-reward-event","POST",{refereeId:e,referralTrackingId:t,eventName:c})}async checkDeferredDeepLink(e){return this.request("/api/v1/sdk/match-link","POST",{fingerprint:e})}};var J=_(require("@react-native-async-storage/async-storage")),re={TRACKED_REFERRAL:"growth-rail-tracked-referral",IS_FIRST_LAUNCH:"growth-rail-is-first-launch"},O=class{static async setTrackedReferral(e){await J.default.setItem(re.TRACKED_REFERRAL,e)}static async getTrackedReferral(){return await J.default.getItem(re.TRACKED_REFERRAL)}static async isFirstLaunch(){return await J.default.getItem(re.IS_FIRST_LAUNCH)===null?(await J.default.setItem(re.IS_FIRST_LAUNCH,"false"),!0):!1}};var o=_(require("react")),r=require("react-native");var{width:pe,height:fe}=r.Dimensions.get("window"),oe=({visible:i,options:e,onClose:t})=>{let c=ie(),[a,y]=(0,o.useState)(null),[h,b]=(0,o.useState)(!1),[l,R]=(0,o.useState)(!1),u=(0,o.useRef)(new r.Animated.Value(0)).current,V=(0,o.useRef)(new r.Animated.Value(0)).current,I=e?.title||"Invite Friends & Earn",P=e?.theme?.primaryColor||"#2563eb",K=e?.theme?.backgroundColor||"#ffffff",q=e?.theme?.tintColor||"#000000",ae=e?.theme?.tintAlpha??.5,S=(e?.componentType||"modal")==="drawer",D=e?.position||"bottom-right",ue="#111827",X="#6b7280";(0,o.useEffect)(()=>{i?(a||se(),r.Animated.parallel([r.Animated.timing(V,{toValue:1,duration:300,useNativeDriver:!0}),r.Animated.spring(u,{toValue:1,useNativeDriver:!0,tension:50,friction:8})]).start()):(u.setValue(0),V.setValue(0))},[i]);let W=()=>{r.Animated.parallel([r.Animated.timing(V,{toValue:0,duration:250,useNativeDriver:!0}),r.Animated.timing(u,{toValue:0,duration:250,useNativeDriver:!0})]).start(()=>{t()})},se=async()=>{b(!0);try{if(c.isUserReady){let F=await c.getReferralLink();y(F||null)}}catch(F){console.error("GrowthRail: Failed to load referral link",F)}finally{b(!1)}},le=()=>{a&&(r.Clipboard.setString(a),R(!0),setTimeout(()=>R(!1),2e3))},A=F=>{if(!a)return;let U="",$=encodeURIComponent(a),E=encodeURIComponent("Check this out!");switch(F){case"twitter":U=`https://twitter.com/intent/tweet?url=${$}&text=${E}`;break;case"facebook":U=`https://www.facebook.com/sharer/sharer.php?u=${$}`;break;case"linkedin":U=`https://www.linkedin.com/sharing/share-offsite/?url=${$}`;break;case"whatsapp":U=`whatsapp://send?text=${E}%20${$}`;break;case"email":U=`mailto:?subject=Check this out&body=${E}%20${$}`;break}U&&r.Linking.openURL(U).catch(()=>{})},N=()=>q||"#000000",z=[];S?D.includes("right")?z=[{translateX:u.interpolate({inputRange:[0,1],outputRange:[pe,0]})}]:D.includes("left")?z=[{translateX:u.interpolate({inputRange:[0,1],outputRange:[-pe,0]})}]:z=[{translateY:u.interpolate({inputRange:[0,1],outputRange:[fe,0]})}]:z=[{scale:u.interpolate({inputRange:[0,1],outputRange:[.9,1]})},{translateY:u.interpolate({inputRange:[0,1],outputRange:[20,0]})}];let M=D.includes("top"),G=D.includes("right"),B=D.includes("left"),ce=S?{position:"absolute",...M?{top:100}:{bottom:100},...G?{right:0}:B?{left:0}:{left:0,right:0},width:G||B?"85%":"100%",maxHeight:G||B?fe-200:"80%"}:{};return o.default.createElement(r.Modal,{visible:i,transparent:!0,animationType:"none",onRequestClose:W},o.default.createElement(r.View,{style:n.fullScreen},o.default.createElement(r.Animated.View,{style:[n.overlay,{backgroundColor:N(),opacity:V.interpolate({inputRange:[0,1],outputRange:[0,ae]})}]},o.default.createElement(r.TouchableOpacity,{style:n.fullScreen,activeOpacity:1,onPress:W})),o.default.createElement(r.Animated.View,{style:[S?ce:n.centeredModalContainer,{backgroundColor:K,transform:z},S&&G&&{borderTopLeftRadius:24,borderBottomLeftRadius:24},S&&B&&{borderTopRightRadius:24,borderBottomRightRadius:24},S&&!G&&!B&&D.includes("bottom")&&{borderTopLeftRadius:24,borderTopRightRadius:24},S&&!G&&!B&&D.includes("top")&&{borderBottomLeftRadius:24,borderBottomRightRadius:24},!S&&{borderRadius:24,opacity:u}]},o.default.createElement(r.View,{style:n.contentWrapper},S&&!G&&!B&&D.includes("bottom")&&o.default.createElement(r.View,{style:n.drawerHandle}),o.default.createElement(r.TouchableOpacity,{style:n.closeBtn,onPress:W},o.default.createElement(r.Text,{style:n.closeIcon},"\u2715")),o.default.createElement(r.ScrollView,{contentContainerStyle:n.scrollContent,showsVerticalScrollIndicator:!1},o.default.createElement(r.View,{style:n.header},o.default.createElement(r.View,{style:n.iconLgContainer},o.default.createElement(r.View,{style:n.bowContainerLg},o.default.createElement(r.View,{style:[n.bowLoopLg,n.bowLeftLg,{borderColor:P}]}),o.default.createElement(r.View,{style:[n.bowLoopLg,n.bowRightLg,{borderColor:P}]})),o.default.createElement(r.View,{style:[n.giftLidLg,{backgroundColor:P}]}),o.default.createElement(r.View,{style:[n.giftBodyLg,{borderColor:P}]},o.default.createElement(r.View,{style:[n.ribbonVerticalLg,{backgroundColor:"#fff",opacity:.3}]}))),o.default.createElement(r.Text,{style:[n.title,{color:ue}]},I),o.default.createElement(r.Text,{style:[n.subtitle,{color:X}]},e?.description||"Share your unique link and earn rewards for every friend who signs up.")),c.isUserReady?h?o.default.createElement(r.ActivityIndicator,{size:"large",color:P,style:n.loader}):o.default.createElement(r.View,{style:n.mainContent},o.default.createElement(r.Text,{style:n.label},"YOUR UNIQUE LINK"),o.default.createElement(r.View,{style:n.inputGroup},o.default.createElement(r.Text,{numberOfLines:1,style:n.input},a||"Initializing..."),o.default.createElement(r.TouchableOpacity,{style:[n.copyBtn,{backgroundColor:l?"#10b981":P}],onPress:le},o.default.createElement(r.Text,{style:n.copyBtnText},l?"Copied!":"Copy"))),o.default.createElement(r.View,{style:n.shareLabelContainer},o.default.createElement(r.View,{style:n.divider}),o.default.createElement(r.Text,{style:n.shareLabel},"Share via"),o.default.createElement(r.View,{style:n.divider})),o.default.createElement(r.View,{style:n.socialContainer},o.default.createElement(Q,{platform:"twitter",color:"#000000",onPress:()=>A("twitter"),icon:"\u{1D54F}"}),o.default.createElement(Q,{platform:"facebook",color:"#1877F2",onPress:()=>A("facebook"),icon:"f"}),o.default.createElement(Q,{platform:"linkedin",color:"#0A66C2",onPress:()=>A("linkedin"),icon:"in"}),o.default.createElement(Q,{platform:"whatsapp",color:"#25D366",onPress:()=>A("whatsapp"),icon:"W"}),o.default.createElement(Q,{platform:"email",color:"#888888",onPress:()=>A("email"),icon:"\u2709"}))):o.default.createElement(r.View,{style:n.loadingBox},o.default.createElement(r.Text,{style:{color:X,textAlign:"center"}},"Please log in to view your referral dashboard.")),o.default.createElement(r.View,{style:n.footer},o.default.createElement(r.Text,{style:n.poweredBy},"Powered by ",o.default.createElement(r.Text,{style:n.brand,onPress:()=>r.Linking.openURL("https://growthrail.com")},"Growth Rail"))))))))},Q=({color:i,onPress:e,icon:t})=>o.default.createElement(r.TouchableOpacity,{style:[n.socialBtn,{borderColor:"#e5e7eb"}],onPress:e},o.default.createElement(r.Text,{style:[n.socialIcon,{color:i}]},t)),n=r.StyleSheet.create({fullScreen:{flex:1},overlay:{...r.StyleSheet.absoluteFillObject},centeredModalContainer:{width:"90%",maxWidth:400,alignSelf:"center",marginTop:fe*.15,overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:20},shadowOpacity:.25,shadowRadius:30,elevation:20},contentWrapper:{paddingTop:r.Platform.OS==="ios"?20:0},drawerHandle:{width:40,height:4,backgroundColor:"#e5e7eb",borderRadius:2,alignSelf:"center",marginTop:12},scrollContent:{padding:32,alignItems:"center"},closeBtn:{position:"absolute",top:20,right:20,zIndex:10,width:32,height:32,borderRadius:16,backgroundColor:"#f3f4f6",justifyContent:"center",alignItems:"center"},closeIcon:{fontSize:14,color:"#9ca3af"},header:{alignItems:"center",marginBottom:28,width:"100%"},socialIcon:{fontSize:18,fontWeight:"bold"},title:{fontSize:22,fontWeight:"700",letterSpacing:-.5,marginBottom:8,textAlign:"center"},subtitle:{fontSize:14,lineHeight:20,textAlign:"center",paddingHorizontal:10},loadingBox:{padding:16,backgroundColor:"#f9fafb",borderRadius:12,width:"100%",alignItems:"center",borderWidth:1,borderColor:"#e5e7eb",marginTop:20},loader:{marginVertical:20},mainContent:{width:"100%"},label:{fontSize:12,fontWeight:"600",color:"#6b7280",marginBottom:8,letterSpacing:.5},inputGroup:{flexDirection:"row",alignItems:"center",backgroundColor:"#f9fafb",borderWidth:1,borderColor:"#e5e7eb",borderRadius:12,padding:6,marginBottom:24},input:{flex:1,paddingHorizontal:12,fontSize:14,color:"#111827",fontFamily:r.Platform.OS==="ios"?"Menlo":"monospace"},copyBtn:{paddingVertical:8,paddingHorizontal:16,borderRadius:8,justifyContent:"center",alignItems:"center"},copyBtnText:{color:"#fff",fontSize:13,fontWeight:"600"},shareLabelContainer:{flexDirection:"row",alignItems:"center",marginBottom:16,marginTop:8},divider:{flex:1,height:1,backgroundColor:"#e5e7eb"},shareLabel:{marginHorizontal:12,fontSize:13,fontWeight:"500",color:"#6b7280"},socialContainer:{flexDirection:"row",justifyContent:"center",gap:12},socialBtn:{width:44,height:44,borderRadius:22,backgroundColor:"#fff",borderWidth:1,justifyContent:"center",alignItems:"center"},iconLgContainer:{width:60,height:60,justifyContent:"center",alignItems:"center",marginBottom:16},bowContainerLg:{flexDirection:"row",position:"absolute",top:2,zIndex:2},bowLoopLg:{width:20,height:16,borderWidth:3,borderColor:"#e5e7eb",borderRadius:10},bowLeftLg:{marginRight:-2,transform:[{rotate:"-15deg"}]},bowRightLg:{marginLeft:-2,transform:[{rotate:"15deg"}]},giftLidLg:{width:44,height:10,backgroundColor:"#f3f4f6",borderRadius:2,marginTop:12,zIndex:3},giftBodyLg:{width:36,height:24,borderWidth:3,borderTopWidth:0,borderBottomLeftRadius:4,borderBottomRightRadius:4,justifyContent:"center",alignItems:"center",marginTop:-2},ribbonVerticalLg:{width:4,height:"100%",opacity:.1},footer:{marginTop:32,alignItems:"center"},poweredBy:{fontSize:11,color:"#9ca3af"},brand:{fontWeight:"600",textDecorationLine:"underline"}});var C=_(require("react")),w=require("react-native"),ne=({onPress:i,options:e})=>{let t=e?.modal?.theme?.primaryColor||"#2563eb",c=e?.trigger?.position||"bottom-right",y=(e?.trigger?.displayMode||"floating")==="edge",h=c.includes("right"),b=c.includes("top"),l=(0,C.useRef)(new w.Animated.Value(-1)).current;(0,C.useEffect)(()=>{let I=()=>{l.setValue(-1),w.Animated.timing(l,{toValue:2,duration:3e3,easing:w.Easing.linear,useNativeDriver:!0}).start(()=>{setTimeout(I,1e3)})};I()},[]);let R=l.interpolate({inputRange:[-1,2],outputRange:y?[-50,100]:[-60,120]}),u={"bottom-right":{bottom:100,right:0},"bottom-left":{bottom:100,left:0},"top-right":{top:100,right:0},"top-left":{top:100,left:0}};y||(u["bottom-right"]={bottom:30,right:20},u["bottom-left"]={bottom:30,left:20},u["top-right"]={top:60,right:20},u["top-left"]={top:60,left:20});let V=()=>C.default.createElement(w.View,{style:k.iconContainer},C.default.createElement(w.View,{style:k.bowContainer},C.default.createElement(w.View,{style:[k.bowLoop,k.bowLeft]}),C.default.createElement(w.View,{style:[k.bowLoop,k.bowRight]})),C.default.createElement(w.View,{style:k.giftLid}),C.default.createElement(w.View,{style:k.giftBody},C.default.createElement(w.View,{style:k.ribbonVertical})));return C.default.createElement(w.TouchableOpacity,{style:[y?k.edgeButton:k.floatingButton,{backgroundColor:t},u[c],y&&(h?k.edgeRight:k.edgeLeft)],onPress:i,activeOpacity:.9},C.default.createElement(w.View,{style:k.content},V()),C.default.createElement(w.Animated.View,{style:[k.shimmer,{transform:[{translateX:R},{skewX:"-15deg"}]}]}))},k=w.StyleSheet.create({floatingButton:{position:"absolute",width:60,height:60,borderRadius:30,justifyContent:"center",alignItems:"center",overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:8},shadowOpacity:.2,shadowRadius:12,elevation:8,zIndex:2147483646},edgeButton:{position:"absolute",width:50,height:50,justifyContent:"center",alignItems:"center",overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:4},shadowOpacity:.1,shadowRadius:8,elevation:4,zIndex:2147483646},edgeRight:{borderTopLeftRadius:8,borderBottomLeftRadius:8,paddingLeft:4},edgeLeft:{borderTopRightRadius:8,borderBottomRightRadius:8,paddingRight:4},content:{justifyContent:"center",alignItems:"center"},iconContainer:{width:28,height:28,justifyContent:"center",alignItems:"center"},bowContainer:{flexDirection:"row",position:"absolute",top:0,zIndex:2},bowLoop:{width:10,height:8,borderWidth:2,borderColor:"#fff",borderRadius:5},bowLeft:{marginRight:-1,transform:[{rotate:"-15deg"}]},bowRight:{marginLeft:-1,transform:[{rotate:"15deg"}]},giftLid:{width:22,height:5,backgroundColor:"#fff",borderRadius:1,marginTop:6,zIndex:3},giftBody:{width:18,height:12,borderWidth:2,borderColor:"#fff",borderTopWidth:0,borderBottomLeftRadius:2,borderBottomRightRadius:2,justifyContent:"center",alignItems:"center",marginTop:-1},ribbonVertical:{width:2,height:"100%",backgroundColor:"#fff",opacity:.8},shimmer:{position:"absolute",top:0,left:0,width:50,height:"100%",backgroundColor:"rgba(255, 255, 255, 0.25)"}});var v=_(require("react")),m=require("react-native"),{width:Ve}=m.Dimensions.get("window"),we=({visible:i,options:e,onClose:t})=>{let c=(0,v.useRef)(new m.Animated.Value(0)).current,[a,y]=(0,v.useState)(i),h=e.position.includes("bottom")?"bottom":"top",b=e.position.includes("left")?"left":e.position.includes("right")?"right":"center";if((0,v.useEffect)(()=>{i?(y(!0),m.Animated.spring(c,{toValue:1,useNativeDriver:!0,tension:50,friction:8}).start()):m.Animated.timing(c,{toValue:0,duration:250,useNativeDriver:!0}).start(()=>{y(!1)})},[i]),!a)return null;let l=c.interpolate({inputRange:[0,1],outputRange:[h==="top"?-100:100,0]}),R=c.interpolate({inputRange:[0,1],outputRange:[0,1]}),u={position:"absolute",[h]:m.Platform.OS==="ios"?60:40,zIndex:9999};return b==="center"?(u.left=20,u.right=20,u.alignItems="center"):u[b]=20,v.default.createElement(m.Animated.View,{style:[u,{transform:[{translateY:l}],opacity:R}]},v.default.createElement(m.View,{style:[Z.banner,{backgroundColor:e.themeColor||"#2563eb"}]},v.default.createElement(m.Text,{style:Z.icon},"\u{1F381}"),v.default.createElement(m.Text,{style:Z.text},e.text),v.default.createElement(m.TouchableOpacity,{onPress:t,style:Z.closeBtn},v.default.createElement(m.Text,{style:Z.closeIcon},"\u2715"))))},Z=m.StyleSheet.create({banner:{flexDirection:"row",alignItems:"center",padding:12,paddingHorizontal:16,borderRadius:12,shadowColor:"#000",shadowOffset:{width:0,height:4},shadowOpacity:.15,shadowRadius:10,elevation:5,maxWidth:Ve-40},icon:{fontSize:18,marginRight:10},text:{color:"#fff",fontSize:14,fontWeight:"600",flex:1},closeBtn:{marginLeft:10,padding:4},closeIcon:{color:"rgba(255, 255, 255, 0.8)",fontSize:14,fontWeight:"bold"}});var be=(0,s.createContext)(void 0),ye=({children:i,userId:e,debug:t=!1,...c})=>{let[a,y]=(0,s.useState)(!1),[h,b]=(0,s.useState)(!1),[l,R]=(0,s.useState)(null),u=(0,s.useRef)(new te(c)),V=(0,s.useRef)(!1),I=(0,s.useRef)(null),[P,K]=(0,s.useState)(!1),[q,ae]=(0,s.useState)(),[ge,S]=(0,s.useState)(!1),[D,ue]=(0,s.useState)(),[X,W]=(0,s.useState)(!1),[se,le]=(0,s.useState)();(0,s.useEffect)(()=>{u.current.setDebug(t),t&&f.log("info","Provider","Debug mode enabled")},[t]);let A=(0,s.useCallback)(()=>{let{width:g,height:d}=T.Dimensions.get("window"),p=Intl.DateTimeFormat().resolvedOptions().locale,L=T.PixelRatio.get(),x,j;if(T.Platform.OS==="android"){let de=T.Platform.constants;x=de.Model,j=de.Brand||de.Manufacturer}return{os:T.Platform.OS,platform:T.Platform.OS,osVersion:String(T.Platform.Version),screenWidth:g,screenHeight:d,timezoneOffsetInMinutes:String(new Date().getTimezoneOffset()),locale:p,language:p,deviceModel:x,deviceBrand:j,pixelRatio:L}},[]),N=(0,s.useCallback)(async(g,d)=>{t&&f.log("info","Referral",`Tracking referral code: ${g}`,{referralCode:g,rewardEventName:d});let p=await u.current.trackReferral(g,d);return await O.setTrackedReferral(p.referralTrackingId),t&&f.log("success","Referral",`Referral tracked \u2192 ${p.referralTrackingId}`,p),p},[t]),z=(0,s.useCallback)(async g=>{try{t&&f.log("info","DeepLink",`Received deep link: ${g}`);let d=g.match(/[?&]referralCode=([^&]+)/);d&&(console.log("GrowthRail: Tracking referral from deep link",d[1]),t&&f.log("info","DeepLink",`Found referral code in deep link: ${d[1]}`),await N(d[1]))}catch(d){console.warn("GrowthRail: Error handling deep link",d),t&&f.log("error","DeepLink","Deep link handling failed",{error:String(d)})}},[N,t]),M=(0,s.useCallback)(async()=>{try{t&&f.log("info","Deferred","Checking deferred deep link via fingerprint");let g=A(),{referralTrackingId:d}=await u.current.checkDeferredDeepLink(g);d?(console.log("GrowthRail: Found deferred referral via fingerprinting",d),t&&f.log("success","Deferred",`Deferred referral matched \u2192 ${d}`),await O.setTrackedReferral(d)):t&&f.log("info","Deferred","No deferred referral match found")}catch(g){console.warn("GrowthRail: Fingerprint check failed",g),t&&f.log("error","Deferred","Fingerprint check failed",{error:String(g)})}},[A,t]),G=(0,s.useCallback)(()=>new Promise(g=>{me.default.getInstallReferrerInfo(async(d,p)=>{if(!p&&d&&d.installReferrer)try{let x=decodeURIComponent(d.installReferrer).match(/[?&]referralCode=([^&]+)/);if(x){let{referralTrackingId:j}=await N(x[1]);await O.setTrackedReferral(j),console.log("GrowthRail: Tracked referral via Install Referrer")}}catch(L){console.warn("GrowthRail: Failed to process install referrer",L)}else await M();g()})}),[N,M]),B=(0,s.useCallback)(async g=>{if(l?.id===g&&h)return t&&f.log("info","User",`User ${g} already initialized, skipping`),l;t&&f.log("info","User",`Initializing user: ${g}`);let d=(async()=>{try{let p=await u.current.initAppUser(g),L={...p,id:g};return R(L),b(!0),t&&f.log("success","User",`User initialized: ${g}`,{referralCode:p.referralCode,referralLink:p.referralLink}),p.referrerExperience?.trigger?.show&&(S(!0),t&&f.log("info","UI","Trigger button shown (server-configured)")),L}catch(p){throw I.current=null,t&&f.log("error","User",`User init failed for ${g}`,{error:String(p)}),p}})();return I.current=d,d},[l,h,t]),ce=(0,s.useCallback)(async g=>{t&&f.log("info","Reward",`Tracking reward event: ${g||"(default)"}`);let d=l;!d&&I.current&&(d=await I.current);let p=d?.id||e;if(!p)throw t&&f.log("error","Reward","Cannot track reward \u2014 user not initialized"),new Error("GrowthRail: User must be initialized before tracking a reward event.");let L=await O.getTrackedReferral();if(!L){console.warn("GrowthRail: No referral tracking ID found. Reward event not attributed."),t&&f.log("warn","Reward","No referral tracking ID in storage \u2014 reward not attributed");return}let x=await u.current.trackRewardEvent(p,L,g);return t&&f.log("success","Reward",`Reward event tracked for user ${p}`),x},[l,e,t]),F=(0,s.useCallback)(async()=>await O.getTrackedReferral(),[]),U=(0,s.useCallback)(async()=>{if(l)return l.referralLink;if(I.current)return(await I.current).referralLink},[l]);(0,s.useEffect)(()=>{if(V.current)return;V.current=!0,t&&f.log("info","Provider","SDK initializing\u2026",{platform:T.Platform.OS}),(async()=>{let d=await T.Linking.getInitialURL();t&&f.log("info","DeepLink",`Initial URL: ${d??"(none)"}`),d&&await z(d);let p=T.Linking.addEventListener("url",({url:x})=>z(x)),L=await O.isFirstLaunch();t&&f.log("info","Provider",`First launch: ${L}`),L&&(T.Platform.OS==="android"?await M():await M()),e&&await B(e);try{let x=await u.current.getNewUserExperience(),j=await O.getTrackedReferral();x.banner?.show&&j&&(le(x.banner),W(!0),t&&f.log("info","UI","Banner shown for new user experience"))}catch(x){t&&f.log("warn","Provider","New user experience fetch failed",{error:String(x)})}return y(!0),t&&f.log("success","Provider","SDK initialization complete \u2713"),()=>p.remove()})()},[]),(0,s.useEffect)(()=>{a&&e&&l?.id!==e&&B(e)},[e,a]);let $={isInitialized:a,isUserReady:h,currentUser:l,referralCode:l?.referralCode,referralLink:l?.referralLink,showDashboard:g=>{ae(g),K(!0)},hideDashboard:()=>K(!1),initUser:B,trackReward:ce,getReferralTrackingId:F,getReferralLink:U,isDebugEnabled:t};if(!a)return null;let E=l?.referrerExperience,ke={...E?.modal,...q,position:E?.trigger?.position||q?.position||"bottom-right",theme:{...E?.modal?.theme||{},...q?.theme||{}}},Ce={...E,...D,trigger:{...E?.trigger||{},...D?.trigger||{}}};return s.default.createElement(be.Provider,{value:$},i,s.default.createElement(oe,{visible:P,options:ke,onClose:()=>K(!1)}),ge&&s.default.createElement(ne,{options:Ce,onPress:()=>$.showDashboard()}),X&&s.default.createElement(we,{visible:X,options:se,onClose:()=>W(!1)}))},ie=()=>{let i=(0,s.useContext)(be);if(!i)throw new Error("useGrowthRail must be used within a GrowthRailProvider");return i};var H=require("react");var Re=()=>{let[i,e]=(0,H.useState)(!1),[t,c]=(0,H.useState)(null),a=ie(),y=(0,H.useCallback)(async b=>{e(!0),c(null);try{return await a.initUser(b)}catch(l){let R=l instanceof Error?l:new Error(String(l));throw c(R),R}finally{e(!1)}},[a]),h=(0,H.useCallback)(async b=>{e(!0),c(null);try{await a.trackReward(b)}catch(l){let R=l instanceof Error?l:new Error(String(l));throw c(R),R}finally{e(!1)}},[a]);return{initAppUser:y,trackRewardEvent:h,showReferralDashboard:a.showDashboard,hideReferralDashboard:a.hideDashboard,getReferralTrackingId:a.getReferralTrackingId,getReferralLink:a.getReferralLink,isLoading:i,isInitialized:a.isInitialized,isUserReady:a.isUserReady,error:t,isDebugEnabled:a.isDebugEnabled}};0&&(module.exports={GrowthRailProvider,ReferralDashboard,TriggerButton,useGrowthRail});
|
package/dist/index.mjs
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
var Ve=Object.defineProperty;var Oe=(n,e,s)=>e in n?Ve(n,e,{enumerable:!0,configurable:!0,writable:!0,value:s}):n[e]=s;var ce=(n,e,s)=>Oe(n,typeof e!="symbol"?e+"":e,s);import oe,{createContext as Ze,useContext as et,useEffect as ve,useState as E,useRef as we,useCallback as U}from"react";import{Linking as Le,Platform as N,Dimensions as tt,PixelRatio as rt}from"react-native";import ot from"react-native-play-install-referrer";var J=class{constructor(e){ce(this,"baseUrl");ce(this,"projectSecretKey");let s=typeof process<"u"&&process.env?.ENV_BASE_URL;this.baseUrl=s||"https://api.growthrail.dev",this.projectSecretKey=e.projectSecretKey}async request(e,s,o){let l={"Content-Type":"application/json","X-GrowthRail-ProjectSecret":this.projectSecretKey,Origin:this.baseUrl};try{let c=await fetch(`${this.baseUrl}${e}`,{method:s,headers:l,body:o?JSON.stringify(o):void 0});if(!c.ok){let m="";try{m=await c.text()}catch{}throw new Error(`GrowthRail API Error ${c.status} (${this.baseUrl}${e}): ${c.statusText} - ${m}`)}return c.json()}catch(c){throw console.error("GrowthRail SDK Error:",c),c}}async getNewUserExperience(){return this.request("/api/v1/sdk/new-user-experience","POST")}async initAppUser(e){return this.request("/api/v1/sdk/init-app-user","POST",{clientProvidedId:e})}async trackReferral(e,s){return this.request("/api/v1/sdk/track-referral","POST",{referralCode:e,rewardEventName:s})}async trackRewardEvent(e,s,o){return this.request("/api/v1/sdk/track-reward-event","POST",{refereeId:e,referralTrackingId:s,eventName:o})}async checkDeferredDeepLink(e){return this.request("/api/v1/sdk/match-link","POST",{fingerprint:e})}};import Q from"@react-native-async-storage/async-storage";var Z={TRACKED_REFERRAL:"growth-rail-tracked-referral",IS_FIRST_LAUNCH:"growth-rail-is-first-launch"},P=class{static async setTrackedReferral(e){await Q.setItem(Z.TRACKED_REFERRAL,e)}static async getTrackedReferral(){return await Q.getItem(Z.TRACKED_REFERRAL)}static async isFirstLaunch(){return await Q.getItem(Z.IS_FIRST_LAUNCH)===null?(await Q.setItem(Z.IS_FIRST_LAUNCH,"false"),!0):!1}};import t,{useEffect as De,useState as de,useRef as ye}from"react";import{Modal as Ee,View as u,Text as x,StyleSheet as Re,TouchableOpacity as ee,Clipboard as Ue,ActivityIndicator as Ae,Linking as Ce,ScrollView as Ge,Platform as ke,Animated as B,Dimensions as ze}from"react-native";var{width:Te,height:fe}=ze.get("window"),ge=({visible:n,options:e,onClose:s})=>{let o=te(),[l,c]=de(null),[m,a]=de(!1),[g,p]=de(!1),d=ye(new B.Value(0)).current,b=ye(new B.Value(0)).current,z=e?.title||"Invite Friends & Earn",v=e?.theme?.primaryColor||"#2563eb",M=e?.theme?.backgroundColor||"#ffffff",ie=e?.theme?.tintColor||"#000000",ne=e?.theme?.tintAlpha??.5,y=(e?.componentType||"modal")==="drawer",O=e?.position||"bottom-right",q="#111827",j="#6b7280";De(()=>{n?(l||ae(),B.parallel([B.timing(b,{toValue:1,duration:300,useNativeDriver:!0}),B.spring(d,{toValue:1,useNativeDriver:!0,tension:50,friction:8})]).start()):(d.setValue(0),b.setValue(0))},[n]);let $=()=>{B.parallel([B.timing(b,{toValue:0,duration:250,useNativeDriver:!0}),B.timing(d,{toValue:0,duration:250,useNativeDriver:!0})]).start(()=>{s()})},ae=async()=>{a(!0);try{if(o.isUserReady){let G=await o.getReferralLink();c(G||null)}}catch(G){console.error("GrowthRail: Failed to load referral link",G)}finally{a(!1)}},X=()=>{l&&(Ue.setString(l),p(!0),setTimeout(()=>p(!1),2e3))},L=G=>{if(!l)return;let S="",C=encodeURIComponent(l),H=encodeURIComponent("Check this out!");switch(G){case"twitter":S=`https://twitter.com/intent/tweet?url=${C}&text=${H}`;break;case"facebook":S=`https://www.facebook.com/sharer/sharer.php?u=${C}`;break;case"linkedin":S=`https://www.linkedin.com/sharing/share-offsite/?url=${C}`;break;case"whatsapp":S=`whatsapp://send?text=${H}%20${C}`;break;case"email":S=`mailto:?subject=Check this out&body=${H}%20${C}`;break}S&&Ce.openURL(S).catch(()=>{})},Y=()=>ie||"#000000",I=[];y?O.includes("right")?I=[{translateX:d.interpolate({inputRange:[0,1],outputRange:[Te,0]})}]:O.includes("left")?I=[{translateX:d.interpolate({inputRange:[0,1],outputRange:[-Te,0]})}]:I=[{translateY:d.interpolate({inputRange:[0,1],outputRange:[fe,0]})}]:I=[{scale:d.interpolate({inputRange:[0,1],outputRange:[.9,1]})},{translateY:d.interpolate({inputRange:[0,1],outputRange:[20,0]})}];let be=O.includes("top"),R=O.includes("right"),D=O.includes("left"),se=y?{position:"absolute",...be?{top:100}:{bottom:100},...R?{right:0}:D?{left:0}:{left:0,right:0},width:R||D?"85%":"100%",maxHeight:R||D?fe-200:"80%"}:{};return t.createElement(Ee,{visible:n,transparent:!0,animationType:"none",onRequestClose:$},t.createElement(u,{style:r.fullScreen},t.createElement(B.View,{style:[r.overlay,{backgroundColor:Y(),opacity:b.interpolate({inputRange:[0,1],outputRange:[0,ne]})}]},t.createElement(ee,{style:r.fullScreen,activeOpacity:1,onPress:$})),t.createElement(B.View,{style:[y?se:r.centeredModalContainer,{backgroundColor:M,transform:I},y&&R&&{borderTopLeftRadius:24,borderBottomLeftRadius:24},y&&D&&{borderTopRightRadius:24,borderBottomRightRadius:24},y&&!R&&!D&&O.includes("bottom")&&{borderTopLeftRadius:24,borderTopRightRadius:24},y&&!R&&!D&&O.includes("top")&&{borderBottomLeftRadius:24,borderBottomRightRadius:24},!y&&{borderRadius:24,opacity:d}]},t.createElement(u,{style:r.contentWrapper},y&&!R&&!D&&O.includes("bottom")&&t.createElement(u,{style:r.drawerHandle}),t.createElement(ee,{style:r.closeBtn,onPress:$},t.createElement(x,{style:r.closeIcon},"\u2715")),t.createElement(Ge,{contentContainerStyle:r.scrollContent,showsVerticalScrollIndicator:!1},t.createElement(u,{style:r.header},t.createElement(u,{style:r.iconLgContainer},t.createElement(u,{style:r.bowContainerLg},t.createElement(u,{style:[r.bowLoopLg,r.bowLeftLg,{borderColor:v}]}),t.createElement(u,{style:[r.bowLoopLg,r.bowRightLg,{borderColor:v}]})),t.createElement(u,{style:[r.giftLidLg,{backgroundColor:v}]}),t.createElement(u,{style:[r.giftBodyLg,{borderColor:v}]},t.createElement(u,{style:[r.ribbonVerticalLg,{backgroundColor:"#fff",opacity:.3}]}))),t.createElement(x,{style:[r.title,{color:q}]},z),t.createElement(x,{style:[r.subtitle,{color:j}]},e?.description||"Share your unique link and earn rewards for every friend who signs up.")),o.isUserReady?m?t.createElement(Ae,{size:"large",color:v,style:r.loader}):t.createElement(u,{style:r.mainContent},t.createElement(x,{style:r.label},"YOUR UNIQUE LINK"),t.createElement(u,{style:r.inputGroup},t.createElement(x,{numberOfLines:1,style:r.input},l||"Initializing..."),t.createElement(ee,{style:[r.copyBtn,{backgroundColor:g?"#10b981":v}],onPress:X},t.createElement(x,{style:r.copyBtnText},g?"Copied!":"Copy"))),t.createElement(u,{style:r.shareLabelContainer},t.createElement(u,{style:r.divider}),t.createElement(x,{style:r.shareLabel},"Share via"),t.createElement(u,{style:r.divider})),t.createElement(u,{style:r.socialContainer},t.createElement(_,{platform:"twitter",color:"#000000",onPress:()=>L("twitter"),icon:"\u{1D54F}"}),t.createElement(_,{platform:"facebook",color:"#1877F2",onPress:()=>L("facebook"),icon:"f"}),t.createElement(_,{platform:"linkedin",color:"#0A66C2",onPress:()=>L("linkedin"),icon:"in"}),t.createElement(_,{platform:"whatsapp",color:"#25D366",onPress:()=>L("whatsapp"),icon:"W"}),t.createElement(_,{platform:"email",color:"#888888",onPress:()=>L("email"),icon:"\u2709"}))):t.createElement(u,{style:r.loadingBox},t.createElement(x,{style:{color:j,textAlign:"center"}},"Please log in to view your referral dashboard.")),t.createElement(u,{style:r.footer},t.createElement(x,{style:r.poweredBy},"Powered by ",t.createElement(x,{style:r.brand,onPress:()=>Ce.openURL("https://growthrail.com")},"Growth Rail"))))))))},_=({color:n,onPress:e,icon:s})=>t.createElement(ee,{style:[r.socialBtn,{borderColor:"#e5e7eb"}],onPress:e},t.createElement(x,{style:[r.socialIcon,{color:n}]},s)),r=Re.create({fullScreen:{flex:1},overlay:{...Re.absoluteFillObject},centeredModalContainer:{width:"90%",maxWidth:400,alignSelf:"center",marginTop:fe*.15,overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:20},shadowOpacity:.25,shadowRadius:30,elevation:20},contentWrapper:{paddingTop:ke.OS==="ios"?20:0},drawerHandle:{width:40,height:4,backgroundColor:"#e5e7eb",borderRadius:2,alignSelf:"center",marginTop:12},scrollContent:{padding:32,alignItems:"center"},closeBtn:{position:"absolute",top:20,right:20,zIndex:10,width:32,height:32,borderRadius:16,backgroundColor:"#f3f4f6",justifyContent:"center",alignItems:"center"},closeIcon:{fontSize:14,color:"#9ca3af"},header:{alignItems:"center",marginBottom:28,width:"100%"},socialIcon:{fontSize:18,fontWeight:"bold"},title:{fontSize:22,fontWeight:"700",letterSpacing:-.5,marginBottom:8,textAlign:"center"},subtitle:{fontSize:14,lineHeight:20,textAlign:"center",paddingHorizontal:10},loadingBox:{padding:16,backgroundColor:"#f9fafb",borderRadius:12,width:"100%",alignItems:"center",borderWidth:1,borderColor:"#e5e7eb",marginTop:20},loader:{marginVertical:20},mainContent:{width:"100%"},label:{fontSize:12,fontWeight:"600",color:"#6b7280",marginBottom:8,letterSpacing:.5},inputGroup:{flexDirection:"row",alignItems:"center",backgroundColor:"#f9fafb",borderWidth:1,borderColor:"#e5e7eb",borderRadius:12,padding:6,marginBottom:24},input:{flex:1,paddingHorizontal:12,fontSize:14,color:"#111827",fontFamily:ke.OS==="ios"?"Menlo":"monospace"},copyBtn:{paddingVertical:8,paddingHorizontal:16,borderRadius:8,justifyContent:"center",alignItems:"center"},copyBtnText:{color:"#fff",fontSize:13,fontWeight:"600"},shareLabelContainer:{flexDirection:"row",alignItems:"center",marginBottom:16,marginTop:8},divider:{flex:1,height:1,backgroundColor:"#e5e7eb"},shareLabel:{marginHorizontal:12,fontSize:13,fontWeight:"500",color:"#6b7280"},socialContainer:{flexDirection:"row",justifyContent:"center",gap:12},socialBtn:{width:44,height:44,borderRadius:22,backgroundColor:"#fff",borderWidth:1,justifyContent:"center",alignItems:"center"},iconLgContainer:{width:60,height:60,justifyContent:"center",alignItems:"center",marginBottom:16},bowContainerLg:{flexDirection:"row",position:"absolute",top:2,zIndex:2},bowLoopLg:{width:20,height:16,borderWidth:3,borderColor:"#e5e7eb",borderRadius:10},bowLeftLg:{marginRight:-2,transform:[{rotate:"-15deg"}]},bowRightLg:{marginLeft:-2,transform:[{rotate:"15deg"}]},giftLidLg:{width:44,height:10,backgroundColor:"#f3f4f6",borderRadius:2,marginTop:12,zIndex:3},giftBodyLg:{width:36,height:24,borderWidth:3,borderTopWidth:0,borderBottomLeftRadius:4,borderBottomRightRadius:4,justifyContent:"center",alignItems:"center",marginTop:-2},ribbonVerticalLg:{width:4,height:"100%",opacity:.1},footer:{marginTop:32,alignItems:"center"},poweredBy:{fontSize:11,color:"#9ca3af"},brand:{fontWeight:"600",textDecorationLine:"underline"}});import V,{useEffect as Fe,useRef as We}from"react";import{TouchableOpacity as Ne,View as A,StyleSheet as Me,Animated as ue,Easing as je}from"react-native";var pe=({onPress:n,options:e})=>{let s=e?.modal?.theme?.primaryColor||"#2563eb",o=e?.trigger?.position||"bottom-right",c=(e?.trigger?.displayMode||"floating")==="edge",m=o.includes("right"),a=o.includes("top"),g=We(new ue.Value(-1)).current;Fe(()=>{let z=()=>{g.setValue(-1),ue.timing(g,{toValue:2,duration:3e3,easing:je.linear,useNativeDriver:!0}).start(()=>{setTimeout(z,1e3)})};z()},[]);let p=g.interpolate({inputRange:[-1,2],outputRange:c?[-50,100]:[-60,120]}),d={"bottom-right":{bottom:100,right:0},"bottom-left":{bottom:100,left:0},"top-right":{top:100,right:0},"top-left":{top:100,left:0}};c||(d["bottom-right"]={bottom:30,right:20},d["bottom-left"]={bottom:30,left:20},d["top-right"]={top:60,right:20},d["top-left"]={top:60,left:20});let b=()=>V.createElement(A,{style:w.iconContainer},V.createElement(A,{style:w.bowContainer},V.createElement(A,{style:[w.bowLoop,w.bowLeft]}),V.createElement(A,{style:[w.bowLoop,w.bowRight]})),V.createElement(A,{style:w.giftLid}),V.createElement(A,{style:w.giftBody},V.createElement(A,{style:w.ribbonVertical})));return V.createElement(Ne,{style:[c?w.edgeButton:w.floatingButton,{backgroundColor:s},d[o],c&&(m?w.edgeRight:w.edgeLeft)],onPress:n,activeOpacity:.9},V.createElement(A,{style:w.content},b()),V.createElement(ue.View,{style:[w.shimmer,{transform:[{translateX:p},{skewX:"-15deg"}]}]}))},w=Me.create({floatingButton:{position:"absolute",width:60,height:60,borderRadius:30,justifyContent:"center",alignItems:"center",overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:8},shadowOpacity:.2,shadowRadius:12,elevation:8,zIndex:2147483646},edgeButton:{position:"absolute",width:50,height:50,justifyContent:"center",alignItems:"center",overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:4},shadowOpacity:.1,shadowRadius:8,elevation:4,zIndex:2147483646},edgeRight:{borderTopLeftRadius:8,borderBottomLeftRadius:8,paddingLeft:4},edgeLeft:{borderTopRightRadius:8,borderBottomRightRadius:8,paddingRight:4},content:{justifyContent:"center",alignItems:"center"},iconContainer:{width:28,height:28,justifyContent:"center",alignItems:"center"},bowContainer:{flexDirection:"row",position:"absolute",top:0,zIndex:2},bowLoop:{width:10,height:8,borderWidth:2,borderColor:"#fff",borderRadius:5},bowLeft:{marginRight:-1,transform:[{rotate:"-15deg"}]},bowRight:{marginLeft:-1,transform:[{rotate:"15deg"}]},giftLid:{width:22,height:5,backgroundColor:"#fff",borderRadius:1,marginTop:6,zIndex:3},giftBody:{width:18,height:12,borderWidth:2,borderColor:"#fff",borderTopWidth:0,borderBottomLeftRadius:2,borderBottomRightRadius:2,justifyContent:"center",alignItems:"center",marginTop:-1},ribbonVertical:{width:2,height:"100%",backgroundColor:"#fff",opacity:.8},shimmer:{position:"absolute",top:0,left:0,width:50,height:"100%",backgroundColor:"rgba(255, 255, 255, 0.25)"}});import W,{useEffect as $e,useRef as He,useState as _e}from"react";import{View as Ke,Text as he,StyleSheet as qe,Animated as re,Dimensions as Xe,TouchableOpacity as Ye,Platform as Je}from"react-native";var{width:Qe}=Xe.get("window"),xe=({visible:n,options:e,onClose:s})=>{let o=He(new re.Value(0)).current,[l,c]=_e(n),m=e.position.includes("bottom")?"bottom":"top",a=e.position.includes("left")?"left":e.position.includes("right")?"right":"center";if($e(()=>{n?(c(!0),re.spring(o,{toValue:1,useNativeDriver:!0,tension:50,friction:8}).start()):re.timing(o,{toValue:0,duration:250,useNativeDriver:!0}).start(()=>{c(!1)})},[n]),!l)return null;let g=o.interpolate({inputRange:[0,1],outputRange:[m==="top"?-100:100,0]}),p=o.interpolate({inputRange:[0,1],outputRange:[0,1]}),d={position:"absolute",[m]:Je.OS==="ios"?60:40,zIndex:9999};return a==="center"?(d.left=20,d.right=20,d.alignItems="center"):d[a]=20,W.createElement(re.View,{style:[d,{transform:[{translateY:g}],opacity:p}]},W.createElement(Ke,{style:[K.banner,{backgroundColor:e.themeColor||"#2563eb"}]},W.createElement(he,{style:K.icon},"\u{1F381}"),W.createElement(he,{style:K.text},e.text),W.createElement(Ye,{onPress:s,style:K.closeBtn},W.createElement(he,{style:K.closeIcon},"\u2715"))))},K=qe.create({banner:{flexDirection:"row",alignItems:"center",padding:12,paddingHorizontal:16,borderRadius:12,shadowColor:"#000",shadowOffset:{width:0,height:4},shadowOpacity:.15,shadowRadius:10,elevation:5,maxWidth:Qe-40},icon:{fontSize:18,marginRight:10},text:{color:"#fff",fontSize:14,fontWeight:"600",flex:1},closeBtn:{marginLeft:10,padding:4},closeIcon:{color:"rgba(255, 255, 255, 0.8)",fontSize:14,fontWeight:"bold"}});var Ie=Ze(void 0),it=({children:n,userId:e,...s})=>{let[o,l]=E(!1),[c,m]=E(!1),[a,g]=E(null),p=we(new J(s)),d=we(!1),b=we(null),[z,v]=E(!1),[M,ie]=E(),[ne,me]=E(!1),[y,O]=E(),[q,j]=E(!1),[$,ae]=E(),X=U(()=>{let{width:f,height:i}=tt.get("window"),h=Intl.DateTimeFormat().resolvedOptions().locale,k=rt.get(),T,F;if(N.OS==="android"){let le=N.constants;T=le.Model,F=le.Brand||le.Manufacturer}return{os:N.OS,platform:N.OS,osVersion:String(N.Version),screenWidth:f,screenHeight:i,timezoneOffsetInMinutes:String(new Date().getTimezoneOffset()),locale:h,language:h,deviceModel:T,deviceBrand:F,pixelRatio:k}},[]),L=U(async(f,i)=>{let h=await p.current.trackReferral(f,i);return await P.setTrackedReferral(h.referralTrackingId),h},[]),Y=U(async f=>{try{let i=f.match(/[?&]referralCode=([^&]+)/);i&&(console.log("GrowthRail: Tracking referral from deep link",i[1]),await L(i[1]))}catch(i){console.warn("GrowthRail: Error handling deep link",i)}},[L]),I=U(async()=>{try{let f=X(),{referralTrackingId:i}=await p.current.checkDeferredDeepLink(f);i&&(console.log("GrowthRail: Found deferred referral via fingerprinting",i),await P.setTrackedReferral(i))}catch(f){console.warn("GrowthRail: Fingerprint check failed",f)}},[X]),be=U(()=>new Promise(f=>{ot.getInstallReferrerInfo(async(i,h)=>{if(!h&&i&&i.installReferrer)try{let T=decodeURIComponent(i.installReferrer).match(/[?&]referralCode=([^&]+)/);if(T){let{referralTrackingId:F}=await L(T[1]);await P.setTrackedReferral(F),console.log("GrowthRail: Tracked referral via Install Referrer")}}catch(k){console.warn("GrowthRail: Failed to process install referrer",k)}else await I();f()})}),[L,I]),R=U(async f=>{if(a?.id===f&&c)return a;let i=(async()=>{try{let h=await p.current.initAppUser(f),k={...h,id:f};return g(k),m(!0),h.referrerExperience?.trigger?.show&&me(!0),k}catch(h){throw b.current=null,h}})();return b.current=i,i},[a,c]),D=U(async f=>{let i=a;!i&&b.current&&(i=await b.current);let h=i?.id||e;if(!h)throw new Error("GrowthRail: User must be initialized before tracking a reward event.");let k=await P.getTrackedReferral();if(!k){console.warn("GrowthRail: No referral tracking ID found. Reward event not attributed.");return}return p.current.trackRewardEvent(h,k,f)},[a,e]),se=U(async()=>await P.getTrackedReferral(),[]),G=U(async()=>{if(a)return a.referralLink;if(b.current)return(await b.current).referralLink},[a]);ve(()=>{if(d.current)return;d.current=!0,(async()=>{let i=await Le.getInitialURL();i&&await Y(i);let h=Le.addEventListener("url",({url:T})=>Y(T));await P.isFirstLaunch()&&(N.OS==="android"?await I():await I()),e&&await R(e);try{let T=await p.current.getNewUserExperience(),F=await P.getTrackedReferral();T.banner?.show&&F&&(ae(T.banner),j(!0))}catch{}return l(!0),()=>h.remove()})()},[]),ve(()=>{o&&e&&a?.id!==e&&R(e)},[e,o]);let S={isInitialized:o,isUserReady:c,currentUser:a,referralCode:a?.referralCode,referralLink:a?.referralLink,showDashboard:f=>{ie(f),v(!0)},hideDashboard:()=>v(!1),initUser:R,trackReward:D,getReferralTrackingId:se,getReferralLink:G};if(!o)return null;let C=a?.referrerExperience,H={...C?.modal,...M,position:C?.trigger?.position||M?.position||"bottom-right",theme:{...C?.modal?.theme||{},...M?.theme||{}}},Be={...C,...y,trigger:{...C?.trigger||{},...y?.trigger||{}}};return oe.createElement(Ie.Provider,{value:S},n,oe.createElement(ge,{visible:z,options:H,onClose:()=>v(!1)}),ne&&oe.createElement(pe,{options:Be,onPress:()=>S.showDashboard()}),q&&oe.createElement(xe,{visible:q,options:$,onClose:()=>j(!1)}))},te=()=>{let n=et(Ie);if(!n)throw new Error("useGrowthRail must be used within a GrowthRailProvider");return n};import{useState as Se,useCallback as Pe}from"react";var nt=()=>{let[n,e]=Se(!1),[s,o]=Se(null),l=te(),c=Pe(async a=>{e(!0),o(null);try{return await l.initUser(a)}catch(g){let p=g instanceof Error?g:new Error(String(g));throw o(p),p}finally{e(!1)}},[l]),m=Pe(async a=>{e(!0),o(null);try{await l.trackReward(a)}catch(g){let p=g instanceof Error?g:new Error(String(g));throw o(p),p}finally{e(!1)}},[l]);return{initAppUser:c,trackRewardEvent:m,showReferralDashboard:l.showDashboard,hideReferralDashboard:l.hideDashboard,getReferralTrackingId:l.getReferralTrackingId,getReferralLink:l.getReferralLink,isLoading:n,isInitialized:l.isInitialized,isUserReady:l.isUserReady,error:s}};export{it as GrowthRailProvider,ge as ReferralDashboard,pe as TriggerButton,nt as useGrowthRail};
|
|
1
|
+
var Ue=Object.defineProperty;var Ee=(f,e,t)=>e in f?Ue(f,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):f[e]=t;var Q=(f,e,t)=>Ee(f,typeof e!="symbol"?e+"":e,t);import ne,{createContext as rt,useContext as ot,useEffect as me,useState as V,useRef as be,useCallback as U}from"react";import{Linking as Ie,Platform as W,Dimensions as it,PixelRatio as nt}from"react-native";import at from"react-native-play-install-referrer";var Pe="[GrowthRail]",s={log(f,e,t,c){let n=`${Pe} [${e}] ${t}`;switch(f){case"error":console.error(n,c||"");break;case"warn":console.warn(n,c||"");break;default:console.log(n,c||"");break}}};var Z=class{constructor(e){Q(this,"baseUrl");Q(this,"projectSecretKey");Q(this,"_debug",!1);let t=typeof process<"u"&&process.env?.ENV_BASE_URL;this.baseUrl=t||"https://api.growthrail.dev",this.projectSecretKey=e.projectSecretKey}setDebug(e){this._debug=e}async request(e,t,c){let n={"Content-Type":"application/json","X-GrowthRail-ProjectSecret":this.projectSecretKey,Origin:this.baseUrl},w=Date.now();this._debug&&s.log("api","APIClient",`${t} ${e}`,{method:t,endpoint:e,body:c??null});try{let g=await fetch(`${this.baseUrl}${e}`,{method:t,headers:n,body:c?JSON.stringify(c):void 0}),p=Date.now()-w;if(!g.ok){let i="";try{i=await g.text()}catch{}let m=`GrowthRail API Error ${g.status} (${this.baseUrl}${e}): ${g.statusText} - ${i}`;throw this._debug&&s.log("api","APIClient",`\u2717 ${t} ${e}`,{method:t,endpoint:e,status:g.status,duration:p,error:i}),new Error(m)}return this._debug&&s.log("api","APIClient",`\u2713 ${t} ${e}`,{method:t,endpoint:e,status:g.status,duration:p}),g.json()}catch(g){let p=Date.now()-w;throw this._debug&&s.log("api","APIClient",`\u2717 ${t} ${e} \u2014 Network error`,{method:t,endpoint:e,duration:p,error:String(g)}),console.error("GrowthRail SDK Error:",g),g}}async getNewUserExperience(){return this.request("/api/v1/sdk/new-user-experience","POST")}async initAppUser(e){return this.request("/api/v1/sdk/init-app-user","POST",{clientProvidedId:e})}async trackReferral(e,t){return this.request("/api/v1/sdk/track-referral","POST",{referralCode:e,rewardEventName:t})}async trackRewardEvent(e,t,c){return this.request("/api/v1/sdk/track-reward-event","POST",{refereeId:e,referralTrackingId:t,eventName:c})}async checkDeferredDeepLink(e){return this.request("/api/v1/sdk/match-link","POST",{fingerprint:e})}};import ee from"@react-native-async-storage/async-storage";var te={TRACKED_REFERRAL:"growth-rail-tracked-referral",IS_FIRST_LAUNCH:"growth-rail-is-first-launch"},v=class{static async setTrackedReferral(e){await ee.setItem(te.TRACKED_REFERRAL,e)}static async getTrackedReferral(){return await ee.getItem(te.TRACKED_REFERRAL)}static async isFirstLaunch(){return await ee.getItem(te.IS_FIRST_LAUNCH)===null?(await ee.setItem(te.IS_FIRST_LAUNCH,"false"),!0):!1}};import r,{useEffect as Ae,useState as fe,useRef as ke}from"react";import{Modal as ze,View as h,Text as L,StyleSheet as Ce,TouchableOpacity as re,Clipboard as Ge,ActivityIndicator as $e,Linking as Te,ScrollView as Fe,Platform as xe,Animated as I,Dimensions as We}from"react-native";var{width:Le,height:ge}=We.get("window"),ue=({visible:f,options:e,onClose:t})=>{let c=oe(),[n,w]=fe(null),[g,p]=fe(!1),[i,m]=fe(!1),d=ke(new I.Value(0)).current,D=ke(new I.Value(0)).current,k=e?.title||"Invite Friends & Earn",E=e?.theme?.primaryColor||"#2563eb",K=e?.theme?.backgroundColor||"#ffffff",q=e?.theme?.tintColor||"#000000",ae=e?.theme?.tintAlpha??.5,C=(e?.componentType||"modal")==="drawer",T=e?.position||"bottom-right",Re="#111827",X="#6b7280";Ae(()=>{f?(n||se(),I.parallel([I.timing(D,{toValue:1,duration:300,useNativeDriver:!0}),I.spring(d,{toValue:1,useNativeDriver:!0,tension:50,friction:8})]).start()):(d.setValue(0),D.setValue(0))},[f]);let N=()=>{I.parallel([I.timing(D,{toValue:0,duration:250,useNativeDriver:!0}),I.timing(d,{toValue:0,duration:250,useNativeDriver:!0})]).start(()=>{t()})},se=async()=>{p(!0);try{if(c.isUserReady){let F=await c.getReferralLink();w(F||null)}}catch(F){console.error("GrowthRail: Failed to load referral link",F)}finally{p(!1)}},le=()=>{n&&(Ge.setString(n),m(!0),setTimeout(()=>m(!1),2e3))},P=F=>{if(!n)return;let B="",G=encodeURIComponent(n),O=encodeURIComponent("Check this out!");switch(F){case"twitter":B=`https://twitter.com/intent/tweet?url=${G}&text=${O}`;break;case"facebook":B=`https://www.facebook.com/sharer/sharer.php?u=${G}`;break;case"linkedin":B=`https://www.linkedin.com/sharing/share-offsite/?url=${G}`;break;case"whatsapp":B=`whatsapp://send?text=${O}%20${G}`;break;case"email":B=`mailto:?subject=Check this out&body=${O}%20${G}`;break}B&&Te.openURL(B).catch(()=>{})},M=()=>q||"#000000",A=[];C?T.includes("right")?A=[{translateX:d.interpolate({inputRange:[0,1],outputRange:[Le,0]})}]:T.includes("left")?A=[{translateX:d.interpolate({inputRange:[0,1],outputRange:[-Le,0]})}]:A=[{translateY:d.interpolate({inputRange:[0,1],outputRange:[ge,0]})}]:A=[{scale:d.interpolate({inputRange:[0,1],outputRange:[.9,1]})},{translateY:d.interpolate({inputRange:[0,1],outputRange:[20,0]})}];let j=T.includes("top"),z=T.includes("right"),x=T.includes("left"),ce=C?{position:"absolute",...j?{top:100}:{bottom:100},...z?{right:0}:x?{left:0}:{left:0,right:0},width:z||x?"85%":"100%",maxHeight:z||x?ge-200:"80%"}:{};return r.createElement(ze,{visible:f,transparent:!0,animationType:"none",onRequestClose:N},r.createElement(h,{style:o.fullScreen},r.createElement(I.View,{style:[o.overlay,{backgroundColor:M(),opacity:D.interpolate({inputRange:[0,1],outputRange:[0,ae]})}]},r.createElement(re,{style:o.fullScreen,activeOpacity:1,onPress:N})),r.createElement(I.View,{style:[C?ce:o.centeredModalContainer,{backgroundColor:K,transform:A},C&&z&&{borderTopLeftRadius:24,borderBottomLeftRadius:24},C&&x&&{borderTopRightRadius:24,borderBottomRightRadius:24},C&&!z&&!x&&T.includes("bottom")&&{borderTopLeftRadius:24,borderTopRightRadius:24},C&&!z&&!x&&T.includes("top")&&{borderBottomLeftRadius:24,borderBottomRightRadius:24},!C&&{borderRadius:24,opacity:d}]},r.createElement(h,{style:o.contentWrapper},C&&!z&&!x&&T.includes("bottom")&&r.createElement(h,{style:o.drawerHandle}),r.createElement(re,{style:o.closeBtn,onPress:N},r.createElement(L,{style:o.closeIcon},"\u2715")),r.createElement(Fe,{contentContainerStyle:o.scrollContent,showsVerticalScrollIndicator:!1},r.createElement(h,{style:o.header},r.createElement(h,{style:o.iconLgContainer},r.createElement(h,{style:o.bowContainerLg},r.createElement(h,{style:[o.bowLoopLg,o.bowLeftLg,{borderColor:E}]}),r.createElement(h,{style:[o.bowLoopLg,o.bowRightLg,{borderColor:E}]})),r.createElement(h,{style:[o.giftLidLg,{backgroundColor:E}]}),r.createElement(h,{style:[o.giftBodyLg,{borderColor:E}]},r.createElement(h,{style:[o.ribbonVerticalLg,{backgroundColor:"#fff",opacity:.3}]}))),r.createElement(L,{style:[o.title,{color:Re}]},k),r.createElement(L,{style:[o.subtitle,{color:X}]},e?.description||"Share your unique link and earn rewards for every friend who signs up.")),c.isUserReady?g?r.createElement($e,{size:"large",color:E,style:o.loader}):r.createElement(h,{style:o.mainContent},r.createElement(L,{style:o.label},"YOUR UNIQUE LINK"),r.createElement(h,{style:o.inputGroup},r.createElement(L,{numberOfLines:1,style:o.input},n||"Initializing..."),r.createElement(re,{style:[o.copyBtn,{backgroundColor:i?"#10b981":E}],onPress:le},r.createElement(L,{style:o.copyBtnText},i?"Copied!":"Copy"))),r.createElement(h,{style:o.shareLabelContainer},r.createElement(h,{style:o.divider}),r.createElement(L,{style:o.shareLabel},"Share via"),r.createElement(h,{style:o.divider})),r.createElement(h,{style:o.socialContainer},r.createElement(Y,{platform:"twitter",color:"#000000",onPress:()=>P("twitter"),icon:"\u{1D54F}"}),r.createElement(Y,{platform:"facebook",color:"#1877F2",onPress:()=>P("facebook"),icon:"f"}),r.createElement(Y,{platform:"linkedin",color:"#0A66C2",onPress:()=>P("linkedin"),icon:"in"}),r.createElement(Y,{platform:"whatsapp",color:"#25D366",onPress:()=>P("whatsapp"),icon:"W"}),r.createElement(Y,{platform:"email",color:"#888888",onPress:()=>P("email"),icon:"\u2709"}))):r.createElement(h,{style:o.loadingBox},r.createElement(L,{style:{color:X,textAlign:"center"}},"Please log in to view your referral dashboard.")),r.createElement(h,{style:o.footer},r.createElement(L,{style:o.poweredBy},"Powered by ",r.createElement(L,{style:o.brand,onPress:()=>Te.openURL("https://growthrail.com")},"Growth Rail"))))))))},Y=({color:f,onPress:e,icon:t})=>r.createElement(re,{style:[o.socialBtn,{borderColor:"#e5e7eb"}],onPress:e},r.createElement(L,{style:[o.socialIcon,{color:f}]},t)),o=Ce.create({fullScreen:{flex:1},overlay:{...Ce.absoluteFillObject},centeredModalContainer:{width:"90%",maxWidth:400,alignSelf:"center",marginTop:ge*.15,overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:20},shadowOpacity:.25,shadowRadius:30,elevation:20},contentWrapper:{paddingTop:xe.OS==="ios"?20:0},drawerHandle:{width:40,height:4,backgroundColor:"#e5e7eb",borderRadius:2,alignSelf:"center",marginTop:12},scrollContent:{padding:32,alignItems:"center"},closeBtn:{position:"absolute",top:20,right:20,zIndex:10,width:32,height:32,borderRadius:16,backgroundColor:"#f3f4f6",justifyContent:"center",alignItems:"center"},closeIcon:{fontSize:14,color:"#9ca3af"},header:{alignItems:"center",marginBottom:28,width:"100%"},socialIcon:{fontSize:18,fontWeight:"bold"},title:{fontSize:22,fontWeight:"700",letterSpacing:-.5,marginBottom:8,textAlign:"center"},subtitle:{fontSize:14,lineHeight:20,textAlign:"center",paddingHorizontal:10},loadingBox:{padding:16,backgroundColor:"#f9fafb",borderRadius:12,width:"100%",alignItems:"center",borderWidth:1,borderColor:"#e5e7eb",marginTop:20},loader:{marginVertical:20},mainContent:{width:"100%"},label:{fontSize:12,fontWeight:"600",color:"#6b7280",marginBottom:8,letterSpacing:.5},inputGroup:{flexDirection:"row",alignItems:"center",backgroundColor:"#f9fafb",borderWidth:1,borderColor:"#e5e7eb",borderRadius:12,padding:6,marginBottom:24},input:{flex:1,paddingHorizontal:12,fontSize:14,color:"#111827",fontFamily:xe.OS==="ios"?"Menlo":"monospace"},copyBtn:{paddingVertical:8,paddingHorizontal:16,borderRadius:8,justifyContent:"center",alignItems:"center"},copyBtnText:{color:"#fff",fontSize:13,fontWeight:"600"},shareLabelContainer:{flexDirection:"row",alignItems:"center",marginBottom:16,marginTop:8},divider:{flex:1,height:1,backgroundColor:"#e5e7eb"},shareLabel:{marginHorizontal:12,fontSize:13,fontWeight:"500",color:"#6b7280"},socialContainer:{flexDirection:"row",justifyContent:"center",gap:12},socialBtn:{width:44,height:44,borderRadius:22,backgroundColor:"#fff",borderWidth:1,justifyContent:"center",alignItems:"center"},iconLgContainer:{width:60,height:60,justifyContent:"center",alignItems:"center",marginBottom:16},bowContainerLg:{flexDirection:"row",position:"absolute",top:2,zIndex:2},bowLoopLg:{width:20,height:16,borderWidth:3,borderColor:"#e5e7eb",borderRadius:10},bowLeftLg:{marginRight:-2,transform:[{rotate:"-15deg"}]},bowRightLg:{marginLeft:-2,transform:[{rotate:"15deg"}]},giftLidLg:{width:44,height:10,backgroundColor:"#f3f4f6",borderRadius:2,marginTop:12,zIndex:3},giftBodyLg:{width:36,height:24,borderWidth:3,borderTopWidth:0,borderBottomLeftRadius:4,borderBottomRightRadius:4,justifyContent:"center",alignItems:"center",marginTop:-2},ribbonVerticalLg:{width:4,height:"100%",opacity:.1},footer:{marginTop:32,alignItems:"center"},poweredBy:{fontSize:11,color:"#9ca3af"},brand:{fontWeight:"600",textDecorationLine:"underline"}});import S,{useEffect as Ne,useRef as Me}from"react";import{TouchableOpacity as je,View as $,StyleSheet as _e,Animated as he,Easing as He}from"react-native";var pe=({onPress:f,options:e})=>{let t=e?.modal?.theme?.primaryColor||"#2563eb",c=e?.trigger?.position||"bottom-right",w=(e?.trigger?.displayMode||"floating")==="edge",g=c.includes("right"),p=c.includes("top"),i=Me(new he.Value(-1)).current;Ne(()=>{let k=()=>{i.setValue(-1),he.timing(i,{toValue:2,duration:3e3,easing:He.linear,useNativeDriver:!0}).start(()=>{setTimeout(k,1e3)})};k()},[]);let m=i.interpolate({inputRange:[-1,2],outputRange:w?[-50,100]:[-60,120]}),d={"bottom-right":{bottom:100,right:0},"bottom-left":{bottom:100,left:0},"top-right":{top:100,right:0},"top-left":{top:100,left:0}};w||(d["bottom-right"]={bottom:30,right:20},d["bottom-left"]={bottom:30,left:20},d["top-right"]={top:60,right:20},d["top-left"]={top:60,left:20});let D=()=>S.createElement($,{style:b.iconContainer},S.createElement($,{style:b.bowContainer},S.createElement($,{style:[b.bowLoop,b.bowLeft]}),S.createElement($,{style:[b.bowLoop,b.bowRight]})),S.createElement($,{style:b.giftLid}),S.createElement($,{style:b.giftBody},S.createElement($,{style:b.ribbonVertical})));return S.createElement(je,{style:[w?b.edgeButton:b.floatingButton,{backgroundColor:t},d[c],w&&(g?b.edgeRight:b.edgeLeft)],onPress:f,activeOpacity:.9},S.createElement($,{style:b.content},D()),S.createElement(he.View,{style:[b.shimmer,{transform:[{translateX:m},{skewX:"-15deg"}]}]}))},b=_e.create({floatingButton:{position:"absolute",width:60,height:60,borderRadius:30,justifyContent:"center",alignItems:"center",overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:8},shadowOpacity:.2,shadowRadius:12,elevation:8,zIndex:2147483646},edgeButton:{position:"absolute",width:50,height:50,justifyContent:"center",alignItems:"center",overflow:"hidden",shadowColor:"#000",shadowOffset:{width:0,height:4},shadowOpacity:.1,shadowRadius:8,elevation:4,zIndex:2147483646},edgeRight:{borderTopLeftRadius:8,borderBottomLeftRadius:8,paddingLeft:4},edgeLeft:{borderTopRightRadius:8,borderBottomRightRadius:8,paddingRight:4},content:{justifyContent:"center",alignItems:"center"},iconContainer:{width:28,height:28,justifyContent:"center",alignItems:"center"},bowContainer:{flexDirection:"row",position:"absolute",top:0,zIndex:2},bowLoop:{width:10,height:8,borderWidth:2,borderColor:"#fff",borderRadius:5},bowLeft:{marginRight:-1,transform:[{rotate:"-15deg"}]},bowRight:{marginLeft:-1,transform:[{rotate:"15deg"}]},giftLid:{width:22,height:5,backgroundColor:"#fff",borderRadius:1,marginTop:6,zIndex:3},giftBody:{width:18,height:12,borderWidth:2,borderColor:"#fff",borderTopWidth:0,borderBottomLeftRadius:2,borderBottomRightRadius:2,justifyContent:"center",alignItems:"center",marginTop:-1},ribbonVertical:{width:2,height:"100%",backgroundColor:"#fff",opacity:.8},shimmer:{position:"absolute",top:0,left:0,width:50,height:"100%",backgroundColor:"rgba(255, 255, 255, 0.25)"}});import H,{useEffect as Ke,useRef as qe,useState as Xe}from"react";import{View as Ye,Text as we,StyleSheet as Je,Animated as ie,Dimensions as Qe,TouchableOpacity as Ze,Platform as et}from"react-native";var{width:tt}=Qe.get("window"),ve=({visible:f,options:e,onClose:t})=>{let c=qe(new ie.Value(0)).current,[n,w]=Xe(f),g=e.position.includes("bottom")?"bottom":"top",p=e.position.includes("left")?"left":e.position.includes("right")?"right":"center";if(Ke(()=>{f?(w(!0),ie.spring(c,{toValue:1,useNativeDriver:!0,tension:50,friction:8}).start()):ie.timing(c,{toValue:0,duration:250,useNativeDriver:!0}).start(()=>{w(!1)})},[f]),!n)return null;let i=c.interpolate({inputRange:[0,1],outputRange:[g==="top"?-100:100,0]}),m=c.interpolate({inputRange:[0,1],outputRange:[0,1]}),d={position:"absolute",[g]:et.OS==="ios"?60:40,zIndex:9999};return p==="center"?(d.left=20,d.right=20,d.alignItems="center"):d[p]=20,H.createElement(ie.View,{style:[d,{transform:[{translateY:i}],opacity:m}]},H.createElement(Ye,{style:[J.banner,{backgroundColor:e.themeColor||"#2563eb"}]},H.createElement(we,{style:J.icon},"\u{1F381}"),H.createElement(we,{style:J.text},e.text),H.createElement(Ze,{onPress:t,style:J.closeBtn},H.createElement(we,{style:J.closeIcon},"\u2715"))))},J=Je.create({banner:{flexDirection:"row",alignItems:"center",padding:12,paddingHorizontal:16,borderRadius:12,shadowColor:"#000",shadowOffset:{width:0,height:4},shadowOpacity:.15,shadowRadius:10,elevation:5,maxWidth:tt-40},icon:{fontSize:18,marginRight:10},text:{color:"#fff",fontSize:14,fontWeight:"600",flex:1},closeBtn:{marginLeft:10,padding:4},closeIcon:{color:"rgba(255, 255, 255, 0.8)",fontSize:14,fontWeight:"bold"}});var Se=rt(void 0),st=({children:f,userId:e,debug:t=!1,...c})=>{let[n,w]=V(!1),[g,p]=V(!1),[i,m]=V(null),d=be(new Z(c)),D=be(!1),k=be(null),[E,K]=V(!1),[q,ae]=V(),[ye,C]=V(!1),[T,Re]=V(),[X,N]=V(!1),[se,le]=V();me(()=>{d.current.setDebug(t),t&&s.log("info","Provider","Debug mode enabled")},[t]);let P=U(()=>{let{width:l,height:a}=it.get("window"),u=Intl.DateTimeFormat().resolvedOptions().locale,R=nt.get(),y,_;if(W.OS==="android"){let de=W.constants;y=de.Model,_=de.Brand||de.Manufacturer}return{os:W.OS,platform:W.OS,osVersion:String(W.Version),screenWidth:l,screenHeight:a,timezoneOffsetInMinutes:String(new Date().getTimezoneOffset()),locale:u,language:u,deviceModel:y,deviceBrand:_,pixelRatio:R}},[]),M=U(async(l,a)=>{t&&s.log("info","Referral",`Tracking referral code: ${l}`,{referralCode:l,rewardEventName:a});let u=await d.current.trackReferral(l,a);return await v.setTrackedReferral(u.referralTrackingId),t&&s.log("success","Referral",`Referral tracked \u2192 ${u.referralTrackingId}`,u),u},[t]),A=U(async l=>{try{t&&s.log("info","DeepLink",`Received deep link: ${l}`);let a=l.match(/[?&]referralCode=([^&]+)/);a&&(console.log("GrowthRail: Tracking referral from deep link",a[1]),t&&s.log("info","DeepLink",`Found referral code in deep link: ${a[1]}`),await M(a[1]))}catch(a){console.warn("GrowthRail: Error handling deep link",a),t&&s.log("error","DeepLink","Deep link handling failed",{error:String(a)})}},[M,t]),j=U(async()=>{try{t&&s.log("info","Deferred","Checking deferred deep link via fingerprint");let l=P(),{referralTrackingId:a}=await d.current.checkDeferredDeepLink(l);a?(console.log("GrowthRail: Found deferred referral via fingerprinting",a),t&&s.log("success","Deferred",`Deferred referral matched \u2192 ${a}`),await v.setTrackedReferral(a)):t&&s.log("info","Deferred","No deferred referral match found")}catch(l){console.warn("GrowthRail: Fingerprint check failed",l),t&&s.log("error","Deferred","Fingerprint check failed",{error:String(l)})}},[P,t]),z=U(()=>new Promise(l=>{at.getInstallReferrerInfo(async(a,u)=>{if(!u&&a&&a.installReferrer)try{let y=decodeURIComponent(a.installReferrer).match(/[?&]referralCode=([^&]+)/);if(y){let{referralTrackingId:_}=await M(y[1]);await v.setTrackedReferral(_),console.log("GrowthRail: Tracked referral via Install Referrer")}}catch(R){console.warn("GrowthRail: Failed to process install referrer",R)}else await j();l()})}),[M,j]),x=U(async l=>{if(i?.id===l&&g)return t&&s.log("info","User",`User ${l} already initialized, skipping`),i;t&&s.log("info","User",`Initializing user: ${l}`);let a=(async()=>{try{let u=await d.current.initAppUser(l),R={...u,id:l};return m(R),p(!0),t&&s.log("success","User",`User initialized: ${l}`,{referralCode:u.referralCode,referralLink:u.referralLink}),u.referrerExperience?.trigger?.show&&(C(!0),t&&s.log("info","UI","Trigger button shown (server-configured)")),R}catch(u){throw k.current=null,t&&s.log("error","User",`User init failed for ${l}`,{error:String(u)}),u}})();return k.current=a,a},[i,g,t]),ce=U(async l=>{t&&s.log("info","Reward",`Tracking reward event: ${l||"(default)"}`);let a=i;!a&&k.current&&(a=await k.current);let u=a?.id||e;if(!u)throw t&&s.log("error","Reward","Cannot track reward \u2014 user not initialized"),new Error("GrowthRail: User must be initialized before tracking a reward event.");let R=await v.getTrackedReferral();if(!R){console.warn("GrowthRail: No referral tracking ID found. Reward event not attributed."),t&&s.log("warn","Reward","No referral tracking ID in storage \u2014 reward not attributed");return}let y=await d.current.trackRewardEvent(u,R,l);return t&&s.log("success","Reward",`Reward event tracked for user ${u}`),y},[i,e,t]),F=U(async()=>await v.getTrackedReferral(),[]),B=U(async()=>{if(i)return i.referralLink;if(k.current)return(await k.current).referralLink},[i]);me(()=>{if(D.current)return;D.current=!0,t&&s.log("info","Provider","SDK initializing\u2026",{platform:W.OS}),(async()=>{let a=await Ie.getInitialURL();t&&s.log("info","DeepLink",`Initial URL: ${a??"(none)"}`),a&&await A(a);let u=Ie.addEventListener("url",({url:y})=>A(y)),R=await v.isFirstLaunch();t&&s.log("info","Provider",`First launch: ${R}`),R&&(W.OS==="android"?await j():await j()),e&&await x(e);try{let y=await d.current.getNewUserExperience(),_=await v.getTrackedReferral();y.banner?.show&&_&&(le(y.banner),N(!0),t&&s.log("info","UI","Banner shown for new user experience"))}catch(y){t&&s.log("warn","Provider","New user experience fetch failed",{error:String(y)})}return w(!0),t&&s.log("success","Provider","SDK initialization complete \u2713"),()=>u.remove()})()},[]),me(()=>{n&&e&&i?.id!==e&&x(e)},[e,n]);let G={isInitialized:n,isUserReady:g,currentUser:i,referralCode:i?.referralCode,referralLink:i?.referralLink,showDashboard:l=>{ae(l),K(!0)},hideDashboard:()=>K(!1),initUser:x,trackReward:ce,getReferralTrackingId:F,getReferralLink:B,isDebugEnabled:t};if(!n)return null;let O=i?.referrerExperience,Oe={...O?.modal,...q,position:O?.trigger?.position||q?.position||"bottom-right",theme:{...O?.modal?.theme||{},...q?.theme||{}}},Ve={...O,...T,trigger:{...O?.trigger||{},...T?.trigger||{}}};return ne.createElement(Se.Provider,{value:G},f,ne.createElement(ue,{visible:E,options:Oe,onClose:()=>K(!1)}),ye&&ne.createElement(pe,{options:Ve,onPress:()=>G.showDashboard()}),X&&ne.createElement(ve,{visible:X,options:se,onClose:()=>N(!1)}))},oe=()=>{let f=ot(Se);if(!f)throw new Error("useGrowthRail must be used within a GrowthRailProvider");return f};import{useState as De,useCallback as Be}from"react";var lt=()=>{let[f,e]=De(!1),[t,c]=De(null),n=oe(),w=Be(async p=>{e(!0),c(null);try{return await n.initUser(p)}catch(i){let m=i instanceof Error?i:new Error(String(i));throw c(m),m}finally{e(!1)}},[n]),g=Be(async p=>{e(!0),c(null);try{await n.trackReward(p)}catch(i){let m=i instanceof Error?i:new Error(String(i));throw c(m),m}finally{e(!1)}},[n]);return{initAppUser:w,trackRewardEvent:g,showReferralDashboard:n.showDashboard,hideReferralDashboard:n.hideDashboard,getReferralTrackingId:n.getReferralTrackingId,getReferralLink:n.getReferralLink,isLoading:f,isInitialized:n.isInitialized,isUserReady:n.isUserReady,error:t,isDebugEnabled:n.isDebugEnabled}};export{st as GrowthRailProvider,ue as ReferralDashboard,pe as TriggerButton,lt as useGrowthRail};
|