UNPKG

appsonair-react-native-applink

Version:

A self-hosted dynamic link service for app download tracking, deep linking, and engagement analytics.

377 lines (288 loc) 10.7 kB
# AppsOnAir-react-native-AppLink **AppsOnAir-react-native-AppLink** enables seamless handling of deep links and in-app routing within your React Native app. With simple integration, you can configure, manage, and act on links directly from the web dashboard in real time. For more details, refer to the [documentation](https://documentation.appsonair.com/MobileQuickstart/GettingStarted/). ## 🚀 Features - Deep link support (URI schemes and AppLinks) - Fallback behavior (e.g., open Play Store or App Store) - Custom domain support - Seamless migration from Firebase Dynamic Links to AppLink **Note:** For comprehensive instructions on migrating Firebase Dynamic Links to AppLinks, refer to the [documentation](https://documentation.appsonair.com/MobileQuickstart/AppLink/firebase-dynamiclinks-migration). ## Installation Install the **AppsOnAir React Native AppLink** package using `npm`: ```sh npm install appsonair-react-native-applink ``` After installation, navigate to your iOS project directory and install the native dependencies using CocoaPods: ```sh cd ios pod install cd .. ``` ## Minimum Requirements ### Android - Android Gradle Plugin (AGP): version 8.0.2 or higher - Kotlin: version 1.7.10 or higher - Gradle: version 8.0 or higher ### iOS - Minimum deployment target: iOS 13.0 ## Android Setup ### Adding AppId in AndroidManifest.xml Add the following `<meta-data>` tag to your application’s AndroidManifest.xml file inside the `<application>` tag: - Make sure the `android:name` is set to "AppsonairAppId" - Replace the `android:value` with your actual AppId provided by AppsOnAir ```xml </application> ... <meta-data android:name="AppsonairAppId" android:value="********-****-****-****-************" /> </application> ``` ### Adding Intent-Filter in AndroidManifest.xml Add the following `<intent-filter>` inside the `<activity>` tag of your main activity in AndroidManifest.xml: - Replace **your-domain.com** with your actual domain configured in AppsOnAir. ```xml <intent-filter android:autoVerify="true"> <action android:name="android.intent.action.VIEW" /> <category android:name="android.intent.category.DEFAULT" /> <category android:name="android.intent.category.BROWSABLE" /> <data android:host="your-domain.com" android:scheme="https" /> </intent-filter> ``` If you're using a custom URI scheme, add this additional `<intent-filter>` block under the same activity: - Replace **your-domain.com** and **your-scheme** with the custom domain and scheme you’ve defined. ```xml <intent-filter> <action android:name="android.intent.action.VIEW" /> <category android:name="android.intent.category.DEFAULT" /> <category android:name="android.intent.category.BROWSABLE" /> <data android:host="your-domain.com" android:scheme="your-scheme" /> </intent-filter> ``` Update your MainActivity.kt to handle incoming deep links by overriding the onNewIntent method. ### Handling Deep Links in MainActivity Update your `MainActivity.kt` to handle incoming deep links by overriding the onNewIntent method. Make sure the following imports are present at the top of your `MainActivity.kt`: ```kt import android.content.Intent import com.appsonairreactnativeapplink.AppsonairReactNativeApplinkModule import com.facebook.react.ReactApplication ``` In your `MainActivity.kt` class (which should extend ReactActivity), override the `onNewIntent` method as shown below: ```kt class MainActivity : ReactActivity() { ... ... override fun onNewIntent(intent: Intent) { super.onNewIntent(intent) setIntent(intent) val reactContext = (application as? ReactApplication) ?.reactNativeHost ?.reactInstanceManager ?.currentReactContext if (reactContext != null) { reactContext .getNativeModule(AppsonairReactNativeApplinkModule::class.java) ?.onNewIntent(intent) } } } ``` ## iOS Setup ### Adding AppId in Info.plist - Replace the `<string>` value with your actual AppsOnAir AppId ```xml <key>AppsonairAppId</key> <string>XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX</string> ``` ### Universal Links (Associated Domains) To support **Universal Links**, create or edit the `YOUR_PROJECT.entitlements` file and add the following configuration: ```xml <!-- If Using Universal Links --> <key>com.apple.developer.associated-domains</key> <array> <string>applinks:YOUR_DOMAIN</string> <!-- Replace with your actual domain --> </array> ``` > Note: After configuring the Associated Domain for Universal Links, it may take up to 24 hours for the changes to propagate and become active. The setup and verification process is handled by Apple. To support a **Custom URL Scheme**, add the following to your app’s `Info.plist` file: ```xml <!-- If Using Custom Url Schema --> <key>CFBundleURLTypes</key> <array> <dict> <key>CFBundleURLName</key> <string>YOUR_URL_NAME</string> <key>CFBundleURLSchemes</key> <array> <string>YOUR_CUSTOM_URL_SCHEME</string> <!-- Replace with your custom URL scheme --> </array> </dict> </array> ``` > Replace YOUR_URL_NAME with a descriptive name for your app, and YOUR_CUSTOM_URL_SCHEME with the scheme you want to use. ### iOS Deep Link Handling in AppDelegate.swift To handle both Universal Links and Custom URL Schemes, add the following methods to your `AppDelegate.swift` file: ```swift // Step 1: Import AppsOnAir_AppLink import AppsOnAir_AppLink ... // Step 2: AppLink Class instance create var window: UIWindow? let appLinkService = AppLinkService.shared ... // Step 3: Handle AppLink Service func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -> Void) -> Bool { guard userActivity.activityType == NSUserActivityTypeBrowsingWeb, let url = userActivity.webpageURL else { return false } appLinkService.handleAppLink(incomingURL: url) return true } func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey : Any] = [:]) -> Bool { appLinkService.handleAppLink(incomingURL: url) return true } ``` ## Usage ### Function 1: Initialize AppLink Before handling any deep links, you need to initialize the AppsOnAir AppLink SDK. This is typically done in your app’s entry point — such as inside a top-level component or during app startup. ```tsx import React, { useEffect } from 'react'; import { initializeAppLink } from 'appsonair-react-native-applink'; const App = () => { useEffect(() => { // Initialize AppLink on app startup initializeAppLink(); }, []); return ( // your JSX ); }; ``` ### Function 2: Listen for Deep Link Events Use `onDeepLinkProcessed` to listen for incoming deep links after **initialization**. This allows you to respond to navigation events or extract data from the link. ```tsx import React, { useEffect } from 'react'; import { initializeAppLink, onDeepLinkProcessed, } from 'appsonair-react-native-applink'; const App = () => { useEffect(() => { initializeAppLink(); // Subscribe to deep link events const sub = onDeepLinkProcessed(event => { console.log(` Processed:\n${JSON.stringify(event, null, 2)}`); }); // Cleanup on unmount return () => { sub?.remove(); }; }, []); return ( // your UI rendering deepLinkResult ); }; ``` ### Function 3: Listen for Referral Events Use `onReferralLinkDetected` to listen for referral detected after **initialization**. This allows you to respond to navigation events or extract data from the link. ```tsx import React, { useEffect } from 'react'; import { initializeAppLink, onReferralLinkDetected } from 'appsonair-react-native-applink'; const App = () => { useEffect(() => { initializeAppLink(); const sub = onReferralLinkDetected((event) => { console.log(` Referral:\n${JSON.stringify(event, null, 2)}`); }); return () => { sub?.remove(); }; }, []); return ( // your UI rendering deepLinkResult ); }; ``` ### Function 4: Create a New AppLink Use `createAppLink` to programmatically generate a **New AppLink** from your React Native app by passing a structured set of parameters. ```tsx import React, { useState } from 'react'; import { Alert, Button, TextInput, View } from 'react-native'; import { createAppLink, type AppLinkParams, type CreateAppLinkResponse, } from 'appsonair-react-native-applink'; const App = () => { const [linkParams, setLinkParams] = useState<AppLinkParams>({ name: '', url: '', urlPrefix: '', // Replace with your actual domain prefix iosFallbackUrl: '', androidFallbackUrl: '', shortId: '', isOpenInAndroidApp: true, isOpenInBrowserAndroid: false, isOpenInIosApp: true, isOpenInBrowserApple: false, }); const handleCreateLink = async () => { try { const result: CreateAppLinkResponse = await createAppLink(linkParams); if ('error' in result) { throw new Error(result.error); } if ('status' in result && result.status !== 'SUCCESS') { throw new Error(result.message); } if ('data' in result) { const shortUrl = result.data.shortUrl; Alert.alert('AppLink Created', shortUrl); } else { throw new Error( result.message ?? 'No data returned from createAppLink' ); } } catch (err: any) { console.log(err); Alert.alert('Error Creating Link', err.message || 'Unknown error'); } }; return ( <View> <TextInput placeholder="Enter URL" value={linkParams.url} onChangeText={(text) => setLinkParams({ ...linkParams, url: text })} /> <Button title="Create AppLink" onPress={handleCreateLink} /> </View> ); }; ``` ### Function 5: Get Referral Info Use `getReferralInfo` to retrieve any referral data passed through a deep link. ```tsx import { Button } from 'react-native'; import { getReferralInfo } from 'appsonair-react-native-applink'; const App = () => { const handleReferralDetails = async () => { try { const info = await getReferralInfo(); console.log('Referral Info', JSON.stringify(info, null, 2)); } catch (err) { console.log('Error', JSON.stringify(err, null, 2)); } }; return <Button title="Get Referral Info" onPress={handleReferralDetails} />; }; ```