{"id":"0bf161c9-9cfe-4a49-a049-a7ca0217d6ef","shortId":"X2NF6p","kind":"skill","title":"cometchat-native-calls","tagline":"CometChat Calls SDK integration for React Native (Expo managed + bare CLI). Covers @cometchat/calls-sdk-react-native install, dual-SDK init, native module linking (iOS pods, Android Gradle), VoIP push via react-native-callkeep + react-native-voip-push-notification + @react-native","description":"## Purpose\n\nProduction-grade voice + video calling for React Native (Expo managed + bare CLI). Loaded by `cometchat-calls` when `framework` is `expo` or `react-native`. Operates in two modes:\n\n- **Standalone** — calls is the product. Chat SDK (signaling) + Calls SDK (WebRTC) + your own RN screens. **VoIP push is mandatory** — same rule as native iOS / Android.\n- **Additive** — calls layered onto an existing CometChat React Native UI Kit integration. Adds call buttons inline, mounts `CometChatIncomingCall` at app root.\n\n**Read these other skills first:**\n- `cometchat-calls` — dispatcher (modes, hard rules, anti-patterns)\n- `cometchat-native-core` — Chat SDK init, login, env conventions, gesture handler peer-dep rules\n- Framework path: `cometchat-native-expo-patterns` (managed) OR `cometchat-native-bare-patterns` (CLI)\n\n**Ground truth:**\n- SDK source — `~/Downloads/calls-sdk/calls-sdk-react-native-5/package/`\n- Sample app — `~/Downloads/calls-sdk/calls-sdk-react-native-5/sample-apps/cometchat-calls-sample-app-react-native/`\n- Public docs — https://www.cometchat.com/docs/calls/react-native/overview\n\n---\n\n## 1. Hard rules — RN specialization\n\n### 1.0 Calls SDK login is its own step (v5+)\n\nThe v5 Calls SDK has its own auth state, separate from the Chat SDK. After `CometChat.login(uid, AUTH_KEY)` succeeds, you MUST also call `await CometChatCalls.login(uid, AUTH_KEY)` — without it, the FIRST calls API call (`initiateCall`, `joinSession`, `generateToken`) throws **\"auth token cannot be null\"**. The chat skill's \"login once, persist forever\" pattern does NOT transfer.\n\n```ts\nimport { CometChat } from \"@cometchat/chat-sdk-react-native\";\nimport { CometChatCalls } from \"@cometchat/calls-sdk-react-native\";\n\n// ✓ RIGHT — chat login first, then calls login\nawait CometChat.login(uid, AUTH_KEY);\ntry {\n  const callUser = await CometChatCalls.login(uid, AUTH_KEY);  // dev mode\n  // OR for production:\n  // const callUser = await CometChatCalls.loginWithAuthToken(authToken);\n} catch (e) {\n  // surface to user — common cause: typo in app id / auth key\n}\n```\n\n**Surprises that bite on real devices:**\n- The Chat SDK persists login across launches via AsyncStorage; the **Calls SDK does NOT**. Even if `CometChat.getLoggedinUser()` returns a non-null user on cold start, call `CometChatCalls.login` again before any calls API works.\n- A single-arg `CometChatCalls.login(uid)` overload exists for re-login when the SDK has cached auth — default to the (uid, AUTH_KEY) form for dev to avoid foot-guns.\n- Login errors surface as Promise rejections — wrap in try/catch.\n\n### 1.1 Dual-SDK contract\n\nSame shape as web. `@cometchat/chat-sdk-react-native` initiates ringing; `@cometchat/calls-sdk-react-native` runs the WebRTC session. Both packages.\n\n```ts\nimport { CometChat } from \"@cometchat/chat-sdk-react-native\";\nimport { CometChatCalls } from \"@cometchat/calls-sdk-react-native\";\n\n// Chat SDK — initiate\nconst outgoing = new CometChat.Call(receiverUid, CometChat.CALL_TYPE.VIDEO, CometChat.RECEIVER_TYPE.USER);\nconst initiated = await CometChat.initiateCall(outgoing);\n\n// Calls SDK — generate the token (v5 takes only sessionId; auth is internal after login).\nconst { token: callToken } = await CometChatCalls.generateToken(initiated.getSessionId());\n\n// On RN, session/join is rendered via the SDK's declarative Component.\n// There is NO imperative joinSession(token, settings, viewRef) on RN — the\n// Component IS the call surface. With the kit, <CometChatOngoingCall />\n// wraps this internally. For custom UI, render <CometChatCalls.Component\n// callToken={callToken} /> directly. See references/custom-ui.md.\n```\n\n### 1.2 VoIP push — react-native-callkeep + platform-specific push\n\nVoIP push on RN is the highest-effort piece. The standard production stack:\n\n- **`react-native-callkeep`** — bridges CallKit (iOS) + ConnectionService (Android). Single API for \"report incoming call to OS\"\n- **`react-native-voip-push-notification`** — iOS PushKit token registration + payload delivery\n- **`@react-native-firebase/messaging`** — Android FCM high-priority data messages\n- **Server side** — your push server must split iOS sends to PushKit (VoIP cert) and Android sends to FCM with `priority: \"high\"` and a `data` payload (NOT `notification`, which ConnectionService can't intercept)\n\nThe skill scaffolds all four pieces in standalone mode. Additive mode prompts before adding (it's substantial).\n\n### 1.3 Foreground service — same Android 14+ rules as native\n\nWhen the call is active on Android, an ongoing-call foreground service must run. `react-native-callkeep` handles registration but the app's `AndroidManifest.xml` must declare:\n\n```xml\n<uses-permission android:name=\"android.permission.FOREGROUND_SERVICE\" />\n<uses-permission android:name=\"android.permission.FOREGROUND_SERVICE_PHONE_CALL\" />\n<uses-permission android:name=\"android.permission.FOREGROUND_SERVICE_MICROPHONE\" />\n<uses-permission android:name=\"android.permission.FOREGROUND_SERVICE_CAMERA\" />\n<uses-permission android:name=\"android.permission.MANAGE_OWN_CALLS\" />\n<uses-permission android:name=\"android.permission.BIND_TELECOM_CONNECTION_SERVICE\" />\n```\n\nSame silent-crash failure mode as native Android (cf. `cometchat-android-v5-calls` rule 1.3).\n\nFor Expo managed, these go into `app.json` `expo.android.permissions` AND require a config plugin (`react-native-callkeep`'s plugin) to merge into the generated `AndroidManifest.xml` during prebuild. Bare RN can edit the manifest directly.\n\n### 1.4 Server-minted auth tokens\n\nSame — `cometchat-native-production` covers it.\n\n### 1.5 Hangup cleanup — RTCPeerConnection + audio session\n\nCombined web + iOS rules. RN's WebRTC bridge wraps both:\n\n```ts\nfunction endCall() {\n  CometChatCalls.leaveSession();            // v5 canonical (endSession is a deprecated shim)\n  RNCallKeep.endCall(callUUID);             // tells CallKit/ConnectionService the call ended\n\n  // iOS: matching audio session deactivation happens inside callkeep\n  // Android: foreground service stop happens inside callkeep\n}\n```\n\nSkipping `RNCallKeep.endCall` leaves the system call UI stuck (lock-screen card persists, OS thinks there's an active call). Common bug.\n\n### 1.6 Permissions\n\nRequired:\n\n- iOS `Info.plist` — `NSCameraUsageDescription`, `NSMicrophoneUsageDescription` (same as native)\n- Android — runtime requests via `PermissionsAndroid.requestMultiple` for `RECORD_AUDIO`, `CAMERA`, `POST_NOTIFICATIONS` (Android 13+)\n- Expo managed — declare in `app.json` `expo.ios.infoPlist` and `expo.android.permissions`; the prebuild merges into native manifests\n\n### 1.7 IncomingCall mounted at app root\n\n`<CometChatIncomingCall />` (additive mode) goes inside the root navigator OR in the App.tsx wrapper, ABOVE all stacks/tabs. Same rule as web — calls only ring on screens where the listener exists.\n\nIn standalone mode, CallKit/ConnectionService own the foreground UI; `<CometChatIncomingCall />` is not used. Instead, a `react-native-callkeep` event listener at app root reports new calls to the OS.\n\n### 1.8 Three canonical provider/scaffold patterns (non-negotiable)\n\nValidated across 4 RN cohorts on 2026-05-14. Each bug silently breaks integration in a different way; each fix is one line. Future scaffolds MUST emit all three.\n\n**1.8.a — `getLoggedInUser()` throws on no-session; always `.catch(() => null)`.**\n\n```ts\n// ❌ Throws \"User not found\" on every fresh launch → init fails → app stuck\nconst existing = await CometChatUIKit.getLoggedInUser();\nif (existing) return;\n\n// ✅\nconst existing = await CometChatUIKit.getLoggedInUser().catch(() => null);\nif (existing) return;\n```\n\nThe RN SDK treats \"no logged-in user\" as a thrown error (not `null`), unlike the web SDK. Without the catch, every fresh launch aborts before reaching `login()`.\n\n**1.8.b — Render `e.message`, not `String(e)`.**\n\n```ts\n// ❌ Most CometChat SDK errors are plain objects, not Error subclasses\n// → setError(String(e)) shows \"[object Object]\" on screen\ncatch (e) {\n  setError(String(e));\n}\n\n// ✅\ncatch (e) {\n  initialized = false; // let hot-reload retry\n  const msg = e instanceof Error ? e.message : JSON.stringify(e);\n  setError(msg);\n}\n```\n\nHiding the underlying error from the user is the single biggest debugging-time-sink in this stack. Always render `e.message` (or `JSON.stringify(e)` as fallback) so the actionable text reaches the screen.\n\n**1.8.c — DO NOT pass `onAccept` to `<CometChatIncomingCall>`.**\n\n```tsx\n// ❌ Short-circuits the kit's internal acceptCall + OngoingCall transition.\n// Symptom: callee taps Accept; caller's outgoing screen stays on \"Calling…\"\n// indefinitely; the call connects at server level but UI never transitions.\n<CometChatIncomingCall call={call} onAccept={(c) => navigate('OngoingCall', ...)} ... />\n\n// ✅ Let the kit own the accept path; only handle decline + error\n<CometChatIncomingCall\n  call={call}\n  onDecline={() => setCallReceived(false)}\n  onError={() => setCallReceived(false)}\n/>\n```\n\nThe kit calls `CometChat.acceptCall` internally and pushes its own OngoingCall surface. Providing `onAccept` replaces that behavior with the caller's function — typically incomplete, never matches what the kit does.\n\n**Also recommended** — guard creds at init time so undefined `@env`/`process.env.EXPO_PUBLIC_*` values surface as actionable errors:\n\n```ts\nif (!appId || !region) {\n  throw new Error(\n    `Missing CometChat credentials at init time: appId=${JSON.stringify(appId)}, region=${JSON.stringify(region)}. ` +\n    `Check .env defines COMETCHAT_APP_ID/COMETCHAT_REGION/COMETCHAT_AUTH_KEY ` +\n    `and restart Metro with cache wipe.`,\n  );\n}\n```\n\nWithout this guard, undefined env values produce opaque `TypeError: undefined is not a function` from deep inside the SDK — the most expensive failure mode of a typo'd env name.\n\n---\n\n## 2. Setup\n\n### Bare RN CLI\n\n```bash\nnpm install @cometchat/chat-sdk-react-native @cometchat/calls-sdk-react-native\nnpm install react-native-callkeep react-native-voip-push-notification @react-native-firebase/app @react-native-firebase/messaging\nnpm install react-native-webrtc           # Calls SDK peer dep\nnpm install react-native-gesture-handler react-native-reanimated  # already installed if using UI Kit\n```\n\n**iOS — four hardening steps before `pod install`** (validated 2026-05-14 on Apple Silicon, iOS 26.5 sim):\n\n1. **`USE_FRAMEWORKS=static pod install`** — required for WebRTC + CometChat pods. Default dynamic linkage silently produces a binary that can't load WebRTC at runtime. Set in shell rc or invoke every time:\n\n   ```bash\n   cd ios && USE_FRAMEWORKS=static pod install && cd ..\n   # Subsequent builds:\n   USE_FRAMEWORKS=static npx react-native run-ios\n   ```\n\n2. **Remove `EXCLUDED_ARCHS arm64 i386` from `ios/Podfile`** (Apple Silicon). RN templates often add this Intel-era workaround to the post-install hook; on Apple Silicon it blocks `react-native-webrtc`/`JitsiWebRTC` arm64 slices from linking against the simulator. Delete the line:\n\n   ```ruby\n   # ❌ Remove this from post_install on Apple Silicon\n   config.build_settings['EXCLUDED_ARCHS[sdk=iphonesimulator*]'] = 'arm64 i386'\n   ```\n\n3. **`ios/Info.plist`** — minimum set for calls:\n\n   ```xml\n   <key>NSCameraUsageDescription</key>\n   <string>Camera access for video calls</string>\n   <key>NSMicrophoneUsageDescription</key>\n   <string>Microphone access for voice and video calls</string>\n   <key>NSBluetoothAlwaysUsageDescription</key>\n   <string>Bluetooth access for using headsets during calls</string>\n   <key>UIBackgroundModes</key>\n   <array>\n     <string>audio</string>\n     <string>voip</string>\n     <string>remote-notification</string>\n   </array>\n   ```\n\n   `UIBackgroundModes: audio` is the load-bearing one for call-audio survival when the app backgrounds; without it, audio cuts the instant the app loses foreground. `NSBluetoothAlways` prevents a crash when the user connects a Bluetooth headset mid-call.\n\n4. **`ios/.xcode.env.local` NODE_BINARY** — point at a stable Homebrew Node path, not an nvm path that may have been cleaned up:\n\n   ```bash\n   # ios/.xcode.env.local\n   export NODE_BINARY=/opt/homebrew/opt/node@20/bin/node\n   ```\n\n   nvm paths in `~/.nvm/versions/node/v20.x.y/bin/node` go stale when nvm prunes — Xcode build fails with `node: command not found`. Homebrew `/opt/homebrew/opt/node@20/bin/node` is symlinked to whatever node@20.x is currently installed; survives `brew upgrade`.\n\n**iOS-on-RN works even where iOS V5 NATIVE is blocked.** The native iOS V5 cohort is gated upstream by a Cloudsmith 404 on `cometchat-calls-ios`. The RN-on-iOS path links a different WebRTC surface (`react-native-webrtc` + `JitsiWebRTC` pod transitives), so customers on RN + iOS are NOT blocked by the native cohort's vendor issue.\n\n**Android — manifest permissions** (rule 1.3 + 1.6), Firebase config (`google-services.json` in `android/app/`), service registration:\n\n```xml\n<service\n  android:name=\"io.wazo.callkeep.RNCallKeepBackgroundMessagingService\"\n  android:foregroundServiceType=\"phoneCall|microphone|camera\"\n  android:exported=\"false\" />\n```\n\n### Expo managed\n\nCalls SDK requires native modules — Expo managed CANNOT run it without a custom dev client. The skill detects the project mode:\n\n- **Managed + has `expo-dev-client`**: scaffolds config plugins for callkeep/firebase/voip-push, regenerates native projects, builds dev client\n- **Managed without dev client**: prompts the user — calls require either ejecting to bare or adding `expo-dev-client`\n- **EAS Build**: configures `eas.json` profiles + build commands\n\nExpo Go (the public dev client) cannot run calls. The skill states this clearly and refuses to scaffold without a dev client.\n\n#### ⚠️ Real build-time landmines on Expo SDK 54 + chat-uikit-react-native 5.3.x (validated 2026-05-14, 4 cohorts)\n\nA previous version of this doc listed three \"Expo SDK 54 build traps\" (document-picker removal, react-native-worklets install, NDK override). Re-validation on 2026-05-14 across **expo-new, expo-existing, rn-new, rn-existing** showed **none of those three fired** on the current combo (`chat-uikit-react-native@5.3.5` + `calls-sdk-react-native@5.0.0`/`4.4.1` + Expo SDK 54). Removed. The real landmines on this combo are different:\n\n1. **`@cometchat/calls-lib-webrtc` is Cloudsmith-only, NOT on npm.** `npm install @cometchat/calls-lib-webrtc` returns 404. The package lives on CometChat's Cloudsmith registry. Use the tarball URL:\n\n   ```bash\n   npm install --legacy-peer-deps \\\n     'https://dl.cloudsmith.io/public/cometchat/cometchat/raw/files/cometchat-calls-lib-webrtc-346a46ff.tgz'\n   ```\n\n   The exact revision hash may roll forward — check the Cloudsmith page for the current version. Without this, runtime fails when the WebRTC layer initializes.\n\n2. **`--legacy-peer-deps` silently strips peer deps the kit needs at runtime.** chat-uikit-react-native does not declare all its transitive runtime peers in `peerDependencies` (kit + calls SDK together pull in `expo-linking`, `expo-constants`, `expo-asset`, `expo-font` via expo-router, plus `valibot`, `zustand`, `@xmldom/xmldom`, `abab`, `promise.allsettled`, `text-encoding`, `react-native-url-polyfill`, `react-native-performance`). With `--legacy-peer-deps`, npm skips them. Each one bites on first bundle as `Unable to resolve module ...`. Reinstall them explicitly:\n\n   ```bash\n   # Bare RN\n   npm install --legacy-peer-deps \\\n     valibot zustand @xmldom/xmldom abab promise.allsettled text-encoding \\\n     react-native-url-polyfill react-native-performance\n\n   # Expo (resolves to SDK-compatible versions)\n   npx expo install \\\n     expo-linking expo-constants expo-asset expo-font \\\n     -- --legacy-peer-deps\n   ```\n\n3. **`expo.extra` (app.json) caches on the Expo dev client manifest.** Edits to `app.json → expo.extra` after `expo prebuild` do NOT reload to the device — `Constants.expoConfig.extra` keeps reading the prebuild-time snapshot. For dev iteration on credentials, EITHER hardcode in `src/config/*.ts` (Metro hot-bundles source changes) OR run `expo prebuild --clean && expo run:android` after every `app.json → extra` change.\n\n4. **`react-native start` does NOT run `adb reverse` (bare RN only).** Only `react-native run-android` sets `adb reverse tcp:8081 tcp:8081`. When you restart Metro standalone (e.g. after `.env` changes), the device loses port-forwarding and shows \"unable to load scripts.\" Fix:\n\n   ```bash\n   adb reverse tcp:8081 tcp:8081\n   ```\n\n5. **`react-native-dotenv` needs Metro `--reset-cache` after `.env` changes (bare RN only).** The babel plugin processes `.env` at compile time, not runtime. Workflow:\n\n   ```bash\n   pkill -9 -f \"react-native start\"\n   npx react-native start --reset-cache\n   adb reverse tcp:8081 tcp:8081\n   adb shell am force-stop com.<package> && adb shell monkey -p com.<package> -c android.intent.category.LAUNCHER 1\n   ```\n\n### Init\n\n```ts\n// cometchat/init.ts\nimport { CometChat } from \"@cometchat/chat-sdk-react-native\";\nimport { CometChatCalls } from \"@cometchat/calls-sdk-react-native\";\n\nlet initialized = false;\n\nexport async function initCometChat() {\n  if (initialized) return;\n\n  const appSettings = new CometChat.AppSettingsBuilder()\n    .subscribePresenceForAllUsers()\n    .setRegion(process.env.EXPO_PUBLIC_COMETCHAT_REGION!)\n    .build();\n\n  await CometChat.init(process.env.EXPO_PUBLIC_COMETCHAT_APP_ID!, appSettings);\n\n  const callAppSettings = new CometChatCalls.CallAppSettingsBuilder()\n    .setAppId(process.env.EXPO_PUBLIC_COMETCHAT_APP_ID!)\n    .setRegion(process.env.EXPO_PUBLIC_COMETCHAT_REGION!)\n    .build();\n\n  CometChatCalls.init(callAppSettings);\n\n  initialized = true;\n}\n```\n\n(Bare RN uses `react-native-dotenv` and `@env` imports instead of `process.env.EXPO_PUBLIC_*` — see `cometchat-native-bare-patterns`.)\n\n---\n\n## 3. Components catalog\n\n### Calls SDK primitives\n\nSame names + shapes as the JavaScript SDK (Section 3 of `cometchat-react-calls`). The RN SDK adds platform-specific helpers — `CometChatCalls.setUserVideoProxy` for native track piping etc. — covered in deeper sample-app references.\n\n### UI Kit views (additive mode — `@cometchat/chat-uikit-react-native`)\n\n| Component | Purpose |\n|---|---|\n| `<CometChatCallButtons user={u} group={g} />` | Voice + video icon row (typically inside `CometChatMessageHeader`). **Group + user semantics differ** — see callout below |\n| `<CometChatIncomingCall />` | Root-mounted listener |\n| `<CometChatOutgoingCall />` | Auto-mounted by IncomingCall on initiate |\n| `<CometChatOngoingCall />` | Active call view |\n| `<CometChatCallLogs onItemClick={fn} />` | History |\n\n#### ⚠️ Group calls use message-based join, not the ringing channel (validated 2026-05-15)\n\n`<CometChatCallButtons group={group} />` does NOT call `CometChat.initiateCall` like the 1:1 user variant does. Source: `node_modules/@cometchat/chat-uikit-react-native/src/calls/CometChatCallButtons/CometChatCallButtons.tsx:138-201`.\n\n| Surface | What `<CometChatCallButtons>` does |\n|---|---|\n| `user={u}` | `CometChat.initiateCall(call)` → standard Ringing flow → `onIncomingCallReceived` fires on peer's `CallListener` |\n| `group={g}` | `CometChat.sendCustomMessage(meetingMessage)` → meeting-card message in the group; caller jumps straight to in-call surface |\n\n**Implication for receivers:**\n- **Other kit-based clients** (apps using `<CometChatMessageList />` to render group messages) — get the \"Join meeting\" card rendered automatically; tap to join. No additional plumbing.\n- **Custom-UI clients** (apps that render their own message list, or no list at all) — receive NOTHING on the `CallListener` channel for group calls. To handle group meetings, add a `CometChat.addMessageListener` and check for the custom meeting type:\n\n   ```ts\n   CometChat.addMessageListener('GROUP_MEETING_LISTENER', new CometChat.MessageListener({\n     onCustomMessageReceived: (msg) => {\n       if (msg.getCategory() === CometChat.CATEGORY_CUSTOM && msg.getType() === 'meeting') {\n         const sessionId = (msg.getCustomData() as any)?.sessionId;\n         const callType = (msg.getCustomData() as any)?.callType;  // \"audio\" | \"video\"\n         // Show your own \"incoming group call\" UI; tap to navigate to ongoing-call with sessionId\n       }\n     },\n   }));\n   ```\n\nThis semantic is the same across all CometChat kits (React, Angular, native iOS, native Android, Flutter) — group calls broadcast via custom message, NOT the ringing channel. Document loudly because the symptom (group-call recipient sees nothing) looks like a bug but is by design.\n\n---\n\n## 4. Standalone integration\n\nWhen `product === \"voice-video\"` and there is no existing UI Kit.\n\n**Split by calling mode — these are two different shapes:**\n\n### 4a. Standalone — Session mode (meeting-room UX, no ringing)\n\nCalls SDK ONLY. NO Chat SDK. Matches `~/Downloads/calls-sdk/calls-sdk-react-native-5/sample-apps/cometchat-calls-sample-app-react-native/`.\n\n**MANDATORY install set (MUST run BEFORE scaffolding files — bundle will fail with `Unable to resolve module <name>` for each one missing):**\n\n```bash\n# Bare RN — session-only mode (all version pins are load-bearing, see notes below)\nnpm install --legacy-peer-deps \\\n  '@cometchat/calls-sdk-react-native@^5.0.0' \\\n  'https://dl.cloudsmith.io/public/cometchat/cometchat/raw/files/cometchat-calls-lib-webrtc-346a46ff.tgz' \\\n  'react-native-webrtc@^124.0.0' \\\n  'react-native-permissions@^5.0.0' \\\n  'react-native-safe-area-context@^5.0.0' \\\n  '@react-native-async-storage/async-storage@^2.2.0' \\\n  '@xmldom/xmldom@^0.8.11' \\\n  'react-native-svg@^15.0.0' \\\n  'react-native-background-timer@^2.4.1' \\\n  'react-native-performance@^5.1.0' \\\n  'react-native-url-polyfill@^2.0.0' \\\n  'valibot@^1.2.0' \\\n  'zustand@^5.0.0' \\\n  'text-encoding@^0.7.0' \\\n  'abab@^2.0.6' \\\n  'promise.allsettled@^1.0.7'\n\n# The Cloudsmith tarball sometimes silently skips on the same install line as\n# other packages — re-run it on its own if it's missing afterwards:\n[ -d node_modules/@cometchat/calls-lib-webrtc ] || \\\n  npm install --save --force \\\n    'https://dl.cloudsmith.io/public/cometchat/cometchat/raw/files/cometchat-calls-lib-webrtc-346a46ff.tgz'\n```\n\n**Critical version pins** — empirically validated 2026-05-15 (test #3, RN bare on Pixel 3 + RN 0.85):\n\n| Pin | Why |\n|---|---|\n| `@react-native-async-storage/async-storage@^2.2.0` | v3.x splits Android native code into a separate Maven artifact (`org.asyncstorage.shared_storage:storage-android:1.0.0`) that isn't widely published. Gradle fails with `Could not find org.asyncstorage.shared_storage:...`. Pin to ^2.2.0 (self-contained Android). |\n| `@xmldom/xmldom@^0.8.11` | The Calls SDK's polyfill (`dist/polyfills/browser.js`) assumes the xmldom 0.8 prototype shape. v0.9.x reorganized the prototype chain; runtime crashes with `Cannot set property 'innerHTML' of undefined` during polyfill init. Pin to ^0.8.11 (matches SDK's declared peer range). |\n| `react-native-webrtc@^124.0.0` | Older versions don't support React Native 0.74+ Fabric. |\n| `valibot@^1.2.0` / `zustand@^5.0.0` | SDK declares these specific majors; minor upgrades have been API-stable but pin to the declared range to avoid surprises. |\n\n**Metro config patch: route the calls SDK to its .mjs entry.** `@cometchat/calls-sdk-react-native@5.0.0` ships a broken CJS bundle at `dist/index.js` — line 1 is `import \"./polyfills\"` (ESM syntax inside a CJS file). Metro can't parse it; the import returns `undefined` and crashes at runtime as `Cannot read property 'CometChatCalls' of undefined` in your `init.ts`. The `dist/index.mjs` is correctly formed. Patch `metro.config.js`:\n\n```js\nconst { getDefaultConfig, mergeConfig } = require('@react-native/metro-config');\nconst path = require('path');\n\nconst config = {\n  resolver: {\n    resolveRequest: (context, moduleName, platform) => {\n      if (moduleName === '@cometchat/calls-sdk-react-native') {\n        return {\n          filePath: path.resolve(\n            __dirname,\n            'node_modules/@cometchat/calls-sdk-react-native/dist/index.mjs',\n          ),\n          type: 'sourceFile',\n        };\n      }\n      return context.resolveRequest(context, moduleName, platform);\n    },\n  },\n};\n\nmodule.exports = mergeConfig(getDefaultConfig(__dirname), config);\n```\n\nAfter patching, **`npx react-native start --reset-cache`** to clear Metro's resolver cache, then `run-android` / `run-ios`. Customer-found 2026-05-15 (test #3).\n\n**11 peer deps total** — derived empirically (test #3, 2026-05-15) by parsing the SDK's actual bundle imports (`node_modules/@cometchat/calls-sdk-react-native/dist/index.js` + `dist/polyfills/*.js`). Every one is a real bundle-time or runtime requirement. The npm CLI `--legacy-peer-deps` flag silently strips them, so they must be listed explicitly.\n\n**Failure-mode breakdown:**\n\n| Missing peer | Symptom |\n|---|---|\n| `@react-native-async-storage/async-storage` | `Unable to resolve module @react-native-async-storage/async-storage` at first bundle |\n| `react-native-svg` | `Unable to resolve module react-native-svg` at first bundle |\n| `react-native-background-timer` | `Unable to resolve module react-native-background-timer` (polyfills/browser.js) |\n| `valibot`, `zustand`, `react-native-webrtc` | Same — bundle-time imports in SDK index |\n| `@xmldom/xmldom`, `abab`, `promise.allsettled` | Same — polyfills/browser.js DOM + Promise polyfills |\n| `text-encoding`, `react-native-url-polyfill`, `react-native-performance` | Runtime — used inside SDK init code paths |\n\n**Also: DO NOT install `@cometchat/chat-sdk-react-native` in session-only mode.** Dead weight in source (never imported, never initialized) AND it transitively pulls async-storage which conflicts with our own peer-dep management.\n\nVerify checks: `rn_webrtc_peers` (11-peer presence) + `no_chat_sdk_in_session_only` (Chat SDK absence).\n\nThe skill then scaffolds:\n\n1. **`cometchat/init.ts`** — `CometChatCalls.init({ appId, region, authKey })` ONLY. No `CometChat.init`, no `CometChat.login`. Pass `authKey` at init so `CometChatCalls.login(uid)` needs no second arg.\n2. **`screens/JoinSession.tsx`** — UID picker + Start/Join meeting + state machine (`inMeeting && callToken`).\n3. **`screens/CallRoom.tsx`** — Renders `<CometChatCalls.Component callToken={callToken} />` inside `SafeAreaView`. `onConnectionClosed` listener resets state. See `references/call-session.md`.\n4. **No VoIP push** — session mode is link-driven, not ringing-driven. No `react-native-callkeep`, no PushKit, no FCM data messages.\n5. **Native config** — Camera + microphone permissions only.\n\n**Why no Chat SDK / no VoIP push:** session mode never initiates a call entity. Customers tap a meeting link, the app generates a token, joins the session. There's nothing to \"ring.\" Initializing both SDKs + the VoIP-push stack adds substantial complexity for zero benefit.\n\n### 4b. Standalone — Ringing mode (CallKeep + CallKit/ConnectionService + Incoming/Outgoing/Ongoing kit)\n\nDual-SDK: Chat SDK signaling channel + Calls SDK media channel. The skill scaffolds:\n\n1. **`cometchat/init.ts`** — Chat SDK + Calls SDK init (sequential).\n2. **`cometchat/CometChatProvider.tsx`** — Provider with init+login gate.\n3. **`hooks/useCallKeep.ts`** — Sets up `react-native-callkeep`, registers event listeners (didReceiveStartCallAction, answerCall, endCall).\n4. **`services/voipPush.ts`** — Combines `react-native-voip-push-notification` (iOS) + `@react-native-firebase/messaging` (Android). Handles incoming-call payloads → `RNCallKeep.displayIncomingCall(...)`.\n5. **`screens/ProfileScreen.tsx`** (or wherever the call trigger lives) — Voice + video buttons.\n6. **`screens/OngoingCallScreen.tsx`** — Hosts call surface via `<CometChatCalls.Component />` OR custom UI overlay. Implements rule 1.5 cleanup (`CometChatCalls.leaveSession()` + `RNCallKeep.endCall(callUUID)`).\n7. **`screens/CallLogsScreen.tsx`** — Paginated history.\n8. **Native config** — Info.plist, AndroidManifest.xml, Firebase setup, capabilities.\n9. **Server-side push docs** — describes the VoIP cert + FCM key requirements; cannot automate.\n\n## 5. Additive integration\n\nWhen `cometchat-native-core` integration already exists. The skill:\n\n1. Adds `@cometchat/calls-sdk-react-native` + the four push deps.\n2. Patches `cometchat/init.ts` to add `CometChatCalls.init` after `CometChat.init`.\n3. Mounts `<CometChatIncomingCall />` at app root (rule 1.7).\n4. Wires `CometChatMessageHeader` call buttons (auto-rendered when `user` prop is set).\n5. VoIP push: opt-in (asks user — substantial native config).\n6. Adds a `CallLogsScreen` to the existing navigator if the user picked \"dedicated screen\".\n\n## 6. Anti-patterns\n\n1. **Using Expo Go for calls.** Calls require native modules that Expo Go can't load. Either eject or use a dev client. The skill refuses to scaffold against Expo Go.\n2. **Skipping `cd ios && pod install`** after install. Symbols missing → \"Native module CometChatCalls is null\" runtime error. Bare RN only.\n3. **Mounting `<CometChatIncomingCall />` inside a stack/tab navigator.** Loses the listener on stack push. Mount in App.tsx above the navigator (rule 1.7).\n4. **Forgetting `RNCallKeep.endCall`** in the hangup path. CallKit/ConnectionService thinks the call is still active; lock-screen UI gets stuck.\n5. **Sending Android push as `notification` instead of `data`.** ConnectionService cannot intercept `notification` payloads. Server must send `data: { type: \"incoming_call\", sessionId: \"...\" }` with `priority: \"high\"`.\n6. **Missing Firebase `google-services.json`** for Android. Builds fail at compile time but the error is buried in Gradle output.\n7. **Mixing `react-native-callkeep` v4 with RN <0.70.** Older RN versions need callkeep v3.x; the skill checks RN version.\n8. **Passing `onAccept` to `<CometChatIncomingCall>`** (rule 1.8.c). Short-circuits the kit's internal `acceptCall` + OngoingCall transition — callee's UI moves but caller stays on \"Calling…\" indefinitely. Only handle `onDecline` + `onError`; let the kit own the accept path.\n9. **`setError(String(e))` in catch blocks** (rule 1.8.b). Most SDK errors aren't `Error` subclasses → screen shows `[object Object]`. Use `e instanceof Error ? e.message : JSON.stringify(e)`.\n10. **`await CometChatUIKit.getLoggedInUser()` without `.catch(() => null)`** (rule 1.8.a). RN SDK throws \"User not found\" on no-session — kills init before login can run.\n11. **`pod install` without `USE_FRAMEWORKS=static`.** Default dynamic linkage produces a binary that can't load WebRTC at runtime. iOS only.\n12. **`EXCLUDED_ARCHS = 'arm64 i386'` in Podfile post_install on Apple Silicon.** Intel-era workaround that breaks arm64 simulator linking for WebRTC pods. Remove on Apple Silicon hosts.\n13. **`react-native start` followed by app reload without `adb reverse tcp:8081 tcp:8081`.** Only `run-android` sets the port forward; standalone Metro restart loses it. Symptom: \"unable to load scripts.\" Bare RN only.\n14. **Editing `.env` and expecting Metro hot-reload to pick it up.** `react-native-dotenv` is a babel-time plugin — `.env` changes need Metro `--reset-cache`. Bare RN only.\n15. **Installing CometChat deps with `--legacy-peer-deps` and expecting all transitive peers to land.** npm silently skips peers — install `valibot`, `zustand`, `@xmldom/xmldom`, `abab`, `promise.allsettled`, `text-encoding`, `react-native-url-polyfill`, `react-native-performance`, plus Expo's `expo-linking`/`expo-constants`/`expo-asset`/`expo-font` explicitly. See §2 setup landmines.\n16. **`npm install @cometchat/calls-lib-webrtc`.** Returns 404 — package is Cloudsmith-only. Use the `https://dl.cloudsmith.io/...` tarball URL.\n\n## 7. Verification checklist\n\n**Static:**\n\n- [ ] All seven packages: chat-sdk-react-native, calls-sdk-react-native, callkeep, voip-push-notification, firebase/app, firebase/messaging, webrtc\n- [ ] `@cometchat/calls-lib-webrtc` installed via Cloudsmith tarball (not npm — npm returns 404)\n- [ ] iOS: `Info.plist` has UIBackgroundModes (audio + voip + remote-notification) + camera/mic/Bluetooth strings (rule 1.6 + §2 setup)\n- [ ] iOS: PushKit token registration in App.tsx (or a hook called from there)\n- [ ] iOS: `USE_FRAMEWORKS=static` used for `pod install`\n- [ ] iOS (Apple Silicon): no `EXCLUDED_ARCHS = 'arm64 i386'` in Podfile post_install\n- [ ] iOS: `.xcode.env.local` NODE_BINARY points at stable Homebrew node path\n- [ ] Android: manifest has all four FOREGROUND_SERVICE_* permissions + MANAGE_OWN_CALLS + BIND_TELECOM_CONNECTION_SERVICE\n- [ ] Android: `google-services.json` in `android/app/`\n- [ ] Android: callkeep service registered with correct `foregroundServiceType`\n- [ ] CometChatIncomingCall mounted at App.tsx (additive mode), no `onAccept` prop (rule 1.8.c)\n- [ ] CometChatUIKit.getLoggedInUser called with `.catch(() => null)` (rule 1.8.a)\n- [ ] Error rendering uses `e.message` not `String(e)` (rule 1.8.b)\n- [ ] Env-var guard: provider throws actionable error when appId/region undefined (rule 1.8.c \"Also recommended\")\n- [ ] Hangup path includes `endSession` + `RNCallKeep.endCall`\n- [ ] Module-level `initialized` flag\n\n**Runtime (real devices, both platforms):**\n\n- [ ] iOS — terminated app, lock screen rings on incoming call\n- [ ] iOS — answer from lock screen → opens app, joins ongoing call\n- [ ] Android — terminated app, ConnectionService rings (full-screen heads-up) on incoming call\n- [ ] Android — answer from notification → opens app, joins ongoing call\n- [ ] Both — outgoing call connects, two-way audio + video\n- [ ] Both — hangup releases camera + mic, no lingering system call UI\n- [ ] Both — Android 14+: ongoing-call notification visible, doesn't get killed by swipe\n- [ ] Expo: only on dev client / standalone build, not Expo Go\n\n## 8. Pointers\n\n- `cometchat-native-core` — provider, init, gesture handler peer deps\n- `cometchat-native-{expo,bare}-patterns` — pod install, gesture handler, dev client setup\n- `cometchat-native-components` — full UI Kit catalog (additive mode)\n- `cometchat-native-push` — APNs + FCM for chat (overlap with VoIP push but distinct paths — chat push is APNs/FCM standard, VoIP push is PushKit/FCM data-message)\n- `cometchat-native-production` — server-minted tokens\n- `cometchat-native-troubleshooting` — Metro cache, pod install failures, privacy manifest, gesture handler conflicts","tags":["cometchat","native","calls","skills","agent-skills","ai-agent","chat","claude-code","cursor","messaging","nextjs","react"],"capabilities":["skill","source-cometchat","skill-cometchat-native-calls","topic-agent-skills","topic-ai-agent","topic-chat","topic-claude-code","topic-cometchat","topic-cursor","topic-messaging","topic-nextjs","topic-react","topic-react-native","topic-ui-kit"],"categories":["cometchat-skills"],"synonyms":[],"warnings":[],"endpointUrl":"https://skills.sh/cometchat/cometchat-skills/cometchat-native-calls","protocol":"skill","transport":"skills-sh","auth":{"type":"none","details":{"cli":"npx skills add cometchat/cometchat-skills","source_repo":"https://github.com/cometchat/cometchat-skills","install_from":"skills.sh"}},"qualityScore":"0.463","qualityRationale":"deterministic score 0.46 from registry signals: · indexed on github topic:agent-skills · 27 github stars · SKILL.md body (34,169 chars)","verified":false,"liveness":"unknown","lastLivenessCheck":null,"agentReviews":{"count":0,"score_avg":null,"cost_usd_avg":null,"success_rate":null,"latency_p50_ms":null,"narrative_summary":null,"summary_updated_at":null},"enrichmentModel":"deterministic:skill-github:v1","enrichmentVersion":1,"enrichedAt":"2026-05-18T19:04:54.014Z","embedding":null,"createdAt":"2026-05-18T07:04:27.553Z","updatedAt":"2026-05-18T19:04:54.014Z","lastSeenAt":"2026-05-18T19:04:54.014Z","tsv":"'-05':894,1310,1753,1786,2398,2804,3069,3082 '-14':895,1311,1754,1787 '-15':2399,2805,3070,3083 '-201':2419 '-9':2184 '/...':4047 '/.nvm/versions/node/v20.x.y/bin/node':1541 '/app':1268 '/async-storage':2723,2822,3137,3147 '/docs/calls/react-native/overview':181 '/downloads/calls-sdk/calls-sdk-react-native-5/package':173 '/downloads/calls-sdk/calls-sdk-react-native-5/sample-apps/cometchat-calls-sample-app-react-native':176,2652 '/messaging':550,1273,3466 '/metro-config':3008 '/opt/homebrew/opt/node':1536,1556 '/polyfills':2963 '/public/cometchat/cometchat/raw/files/cometchat-calls-lib-webrtc-346a46ff.tgz''':1871,2700,2797 '0.7.0':2756 '0.70':3750 '0.74':2912 '0.8':2871 '0.8.11':2726,2861,2893 '0.85':2814 '1':182,1318,1836,2218,2409,2410,2960,3277,3423,3542,3606 '1.0':187 '1.0.0':2839 '1.0.7':2760 '1.1':386 '1.2':492 '1.2.0':2750,2915 '1.3':607,661,1637 '1.4':696 '1.5':709,3497 '1.6':780,1638,4097 '1.7':817,3563,3676 '1.8':879,916,985,1068,3767,3808,3835,4178,4186,4196,4210 '10':3828 '11':3073,3261,3853 '12':3875 '124.0.0':2705,2904 '13':802,3904 '138':2418 '14':612,3941,4292 '15':3974 '15.0.0':2731 '16':4032 '2':1242,1372,1896,3299,3431,3549,3637,4029,4098 '2.0.0':2748 '2.0.6':2758 '2.2.0':2724,2823,2855 '2.4.1':2737 '20':1563 '20/bin/node':1537,1557 '2026':893,1309,1752,1785,2397,2803,3068,3081 '26.5':1316 '3':1434,2039,2299,2313,2807,2812,3072,3080,3309,3438,3557,3657 '4':889,1510,1755,2099,2611,3323,3452,3564,3677 '4.4.1':1823 '404':1594,1849,4037,4084 '4a':2635 '4b':3401 '5':2155,3348,3474,3529,3577,3697 '5.0.0':1822,2697,2710,2717,2752,2917,2951 '5.1.0':2742 '5.3':1749 '5.3.5':1816 '54':1743,1767,1826 '6':3485,3588,3602,3722 '7':3502,3741,4050 '8':3506,3762,4314 '8081':2123,2125,2152,2154,2201,2203,3917,3919 '9':3514,3800 'abab':1951,1999,2757,3196,3998 'abort':981 'absenc':3272 'accept':1089,1120,3798 'acceptcal':1083,3776 'access':1443,1449,1457 'across':316,888,1788,2571 'action':1063,1179,4204 'activ':620,776,2378,3690 'actual':3089 'ad':603,1701 'adb':2107,2120,2149,2198,2204,2211,3914 'add':114,1385,2322,2511,3395,3543,3553,3589 'addit':102,599,823,2343,2480,3530,4172,4347 'afterward':2786 'alreadi':1295,3538 'also':218,1164,3222,4212 'alway':924,1053 'android':28,101,525,551,572,611,622,653,657,751,790,801,1633,2093,2118,2580,2826,2838,2859,3061,3467,3699,3727,3923,4142,4157,4161,4248,4262,4291 'android.intent.category.launcher':2217 'android/app':1643,4160 'androidmanifest.xml':641,686,3510 'angular':2576 'answer':4239,4263 'answercal':3450 'anti':136,3604 'anti-pattern':135,3603 'api':230,343,527,2928 'api-st':2927 'apn':4353 'apns/fcm':4367 'app':121,175,301,639,821,871,938,1204,1484,1493,2256,2267,2338,2463,2486,3375,3560,3911,4231,4244,4250,4267 'app.json':668,807,2041,2051,2096 'app.tsx':833,3671,4105,4171 'appid':1183,1194,1196,3280 'appid/region':4207 'appl':1313,1380,1398,1424,3885,3901,4121 'appset':2241,2258 'arch':1375,1429,3877,4125 'area':2715 'aren':3813 'arg':348,3298 'arm64':1376,1407,1432,3878,3893,4126 'artifact':2833 'ask':3583 'asset':1939,2031,4023 'assum':2868 'async':2234,2721,2820,3135,3145,3245 'async-storag':3244 'asyncstorag':319 'audio':713,745,797,1464,1470,1480,1488,2548,4089,4278 'auth':203,213,223,236,272,280,303,362,367,438,700 'authkey':3282,3289 'authtoken':291 'auto':2372,3570 'auto-mount':2371 'auto-rend':3569 'autom':3528 'automat':2475 'avoid':373,2937 'await':220,269,277,289,426,446,942,949,2251,3829 'b':986,3809,4197 'babel':2172,3961 'babel-tim':3960 'background':1485,2735,3169,3178 'bare':14,58,166,689,1244,1699,1988,2109,2168,2279,2297,2674,2809,3654,3938,3971,4330 'base':2390,2461 'bash':1247,1351,1531,1862,1987,2148,2182,2673 'bear':1475,2686 'behavior':1150 'benefit':3400 'biggest':1045 'binari':1335,1513,1535,3865,4135 'bind':4153 'bite':307,1975 'block':1401,1582,1625,3806 'bluetooth':1456,1505 'break':899,3892 'breakdown':3128 'brew':1569 'bridg':521,722 'broadcast':2584 'broken':2954 'bug':779,897,2606 'build':1361,1548,1684,1707,1711,1737,1768,2250,2274,3728,4310 'build-tim':1736 'bundl':1978,2083,2661,2956,3090,3103,3150,3165,3189 'bundle-tim':3102,3188 'buri':3737 'button':116,3484,3568 'c':1069,1112,2216,3768,4179,4211 'cach':361,1210,2042,2164,2197,3051,3057,3970,4389 'call':4,6,52,64,78,85,103,115,130,188,198,219,229,231,267,321,337,342,429,474,531,618,626,659,741,763,777,842,875,1096,1099,1109,1110,1127,1128,1137,1280,1439,1446,1454,1462,1479,1509,1598,1649,1694,1721,1818,1926,2302,2318,2379,2386,2405,2426,2453,2506,2555,2563,2583,2599,2628,2645,2863,2944,3367,3416,3427,3471,3479,3488,3567,3611,3612,3687,3717,3787,4063,4109,4152,4181,4237,4247,4261,4270,4273,4288,4295 'call-audio':1478 'callappset':2260,2276 'calle':1087,3779 'caller':1090,1153,2447,3784 'callkeep':36,498,520,634,678,750,757,867,1257,3341,3405,3445,3746,3755,4067,4162 'callkeep/firebase/voip-push':1680 'callkit':522 'callkit/connectionservice':739,854,3406,3684 'calllisten':2435,2502 'calllogsscreen':3591 'callout':2365 'calls-sdk-react-n':1817,4062 'calltoken':445,487,488,3308,3313,3314 'calltyp':2543,2547 'callus':276,288 'calluuid':737,3501 'camera':798,1442,3351,4283 'camera/mic/bluetooth':4094 'cannot':238,1656,1719,2882,2984,3527,3707 'canon':730,881 'capabl':3513 'card':769,2442,2473 'catalog':2301,4346 'catch':292,925,951,977,1011,1016,3805,3832,4183 'caus':298 'cd':1352,1359,3639 'cert':570,3523 'cf':654 'chain':2878 'chang':2085,2098,2134,2167,3965 'channel':2395,2503,2591,3415,3419 'chat':82,142,208,242,263,312,414,1745,1812,1911,2649,3265,3270,3357,3412,3425,4058,4356,4364 'chat-sdk-react-n':4057 'chat-uikit-react-n':1744,1811,1910 'check':1200,1879,2515,3257,3759 'checklist':4052 'circuit':1078,3771 'cjs':2955,2968 'clean':1529,2090 'cleanup':711,3498 'clear':1726,3053 'cli':15,59,168,1246,3110 'client':1663,1675,1686,1690,1705,1718,1734,2047,2462,2485,3628,4308,4337 'cloudsmith':1593,1840,1856,1881,2762,4041,4078 'cloudsmith-on':1839,4040 'code':2828,3220 'cohort':891,1587,1629,1756 'cold':335 'com':2210,2215 'combin':715,3454 'combo':1810,1833 'cometchat':2,5,63,108,129,139,157,164,255,407,656,704,994,1189,1203,1327,1597,1854,2223,2248,2255,2266,2272,2295,2316,2573,3534,3976,4317,4327,4340,4350,4377,4385 'cometchat-android-v5-calls':655 'cometchat-cal':62,128 'cometchat-calls-io':1596 'cometchat-n':4326 'cometchat-native-bare-pattern':163,2294 'cometchat-native-cal':1 'cometchat-native-compon':4339 'cometchat-native-cor':138,3533,4316 'cometchat-native-expo-pattern':156 'cometchat-native-product':703,4376 'cometchat-native-push':4349 'cometchat-native-troubleshoot':4384 'cometchat-react-cal':2315 'cometchat.acceptcall':1138 'cometchat.addmessagelistener':2513,2522 'cometchat.appsettingsbuilder':2243 'cometchat.call':420 'cometchat.call_type.video':422 'cometchat.category':2532 'cometchat.getloggedinuser':327 'cometchat.init':2252,3285,3556 'cometchat.initiatecall':427,2406,2425 'cometchat.login':211,270,3287 'cometchat.messagelistener':2527 'cometchat.receiver_type.user':423 'cometchat.sendcustommessage':2438 'cometchat/calls-lib-webrtc':1837,1847,2790,4035,4075 'cometchat/calls-sdk-react-native':17,261,398,413,1251,2229,2696,2950,3022,3544 'cometchat/calls-sdk-react-native/dist/index.js':3094 'cometchat/calls-sdk-react-native/dist/index.mjs':3029 'cometchat/chat-sdk-react-native':257,395,409,1250,2225,3226 'cometchat/chat-uikit-react-native':2345 'cometchat/chat-uikit-react-native/src/calls/cometchatcallbuttons/cometchatcallbuttons.tsx':2417 'cometchat/cometchatprovider.tsx':3432 'cometchat/init.ts':2221,3278,3424,3551 'cometchatcal':259,411,2227,2987,3649 'cometchatcallbutton':2348,2400 'cometchatcalllog':2381 'cometchatcalls.callappsettingsbuilder':2262 'cometchatcalls.component':486,3312 'cometchatcalls.generatetoken':447 'cometchatcalls.init':2275,3279,3554 'cometchatcalls.leavesession':728,3499 'cometchatcalls.login':221,278,338,349,3293 'cometchatcalls.loginwithauthtoken':290 'cometchatcalls.setuservideoproxy':2327 'cometchatincomingcal':119,1108,1126,4168 'cometchatmessagehead':2359,3566 'cometchatuikit.getloggedinuser':943,950,3830,4180 'command':1552,1712 'common':297,778 'compat':2018 'compil':2177,3731 'complex':3397 'compon':459,471,2300,2346,4342 'config':673,1640,1677,2940,3014,3041,3350,3508,3587 'config.build':1426 'configur':1708 'conflict':3248,4397 'connect':1100,1503,4155,4274 'connectionservic':524,586,3706,4251 'const':275,287,417,424,443,940,947,1025,2240,2259,2536,2542,3001,3009,3013 'constant':1936,2028,4020 'constants.expoconfig.extra':2062 'contain':2858 'context':2716,3017,3034 'context.resolverequest':3033 'contract':390 'convent':147 'core':141,3536,4319 'correct':2996,4166 'could':2848 'cover':16,707,2333 'crash':648,1499,2880,2980 'cred':1167 'credenti':1190,2074 'critic':2798 'current':1566,1809,1885 'custom':483,1619,1661,2483,2518,2533,2586,3066,3369,3492 'custom-ui':2482 'customer-found':3065 'cut':1489 'd':1239,2787 'data':556,581,3346,3705,3714,4374 'data-messag':4373 'deactiv':747 'dead':3232 'debug':1047 'debugging-time-sink':1046 'declar':458,643,805,1917,2897,2919,2934 'declin':1124 'dedic':3600 'deep':1227 'deeper':2335 'default':363,1329,3860 'defin':1202 'delet':1414 'deliveri':545 'dep':152,1283,1868,1900,1904,1969,1995,2038,2695,3075,3114,3254,3548,3977,3982,4325 'deprec':734 'deriv':3077 'describ':3520 'design':2610 'detect':1666 'dev':282,371,1662,1674,1685,1689,1704,1717,1733,2046,2071,3627,4307,4336 'devic':310,2061,2136,4226 'didreceivestartcallact':3449 'differ':903,1608,1835,2363,2633 'direct':489,695 'dirnam':3026,3040 'dispatch':131 'dist/index.js':2958 'dist/index.mjs':2994 'dist/polyfills':3095 'dist/polyfills/browser.js':2867 'distinct':4362 'dl.cloudsmith.io':1870,2699,2796,4046 'dl.cloudsmith.io/...':4045 'dl.cloudsmith.io/public/cometchat/cometchat/raw/files/cometchat-calls-lib-webrtc-346a46ff.tgz''':1869,2698,2795 'doc':178,1762,3519 'document':1771,2592 'document-pick':1770 'doesn':4298 'dom':3200 'dotenv':2159,2285,3957 'driven':3332,3336 'dual':20,388,3410 'dual-sdk':19,387,3409 'dynam':1330,3861 'e':293,991,1005,1012,1015,1017,1027,1032,1058,3803,3822,3827,4194 'e.g':2131 'e.message':988,1030,1055,3825,4191 'ea':1706 'eas.json':1709 'edit':692,2049,3942 'effort':511 'either':1696,2075,3622 'eject':1697,3623 'emit':913 'empir':2801,3078 'encod':1955,2003,2755,3205,4002 'end':742 'endcal':727,3451 'endsess':731,4217 'entiti':3368 'entri':2949 'env':146,1173,1201,1216,1240,2133,2166,2175,2287,3943,3964,4199 'env-var':4198 'era':1389,3889 'error':378,968,996,1001,1029,1038,1125,1180,1187,3653,3735,3812,3815,3824,4188,4205 'esm':2964 'etc':2332 'even':325,1576 'event':868,3447 'everi':933,978,1349,2095,3097 'exact':1873 'exclud':1374,1428,3876,4124 'exist':107,352,850,941,945,948,954,1794,1800,2623,3539,3594 'expect':3945,3984 'expens':1233 'explicit':1986,3124,4027 'expo':12,56,68,159,663,803,1647,1654,1673,1703,1713,1741,1765,1790,1793,1824,1932,1935,1938,1941,1945,2013,2021,2024,2027,2030,2033,2045,2054,2088,2091,3608,3617,3635,4013,4016,4019,4022,4025,4304,4312,4329 'expo-asset':1937,2029,4021 'expo-const':1934,2026,4018 'expo-dev-cli':1672,1702 'expo-exist':1792 'expo-font':1940,2032,4024 'expo-link':1931,2023,4015 'expo-new':1789 'expo-rout':1944 'expo.android.permissions':669,810 'expo.extra':2040,2052 'expo.ios.infoplist':808 'export':1533,2233 'extra':2097 'f':2185 'fabric':2913 'fail':937,1549,1890,2663,2846,3729 'failur':649,1234,3126,4392 'failure-mod':3125 'fallback':1060 'fals':1019,1131,1134,2232 'fcm':552,575,3345,3524,4354 'file':2660,2969 'filepath':3024 'find':2850 'fire':1806,2431 'firebas':549,1267,1272,1639,3465,3511,3724 'firebase/app':4072 'firebase/messaging':4073 'first':127,228,265,1977,3149,3164 'fix':906,2147 'flag':3115,4223 'flow':2429 'flutter':2581 'fn':2383 'follow':3909 'font':1942,2034,4026 'foot':375 'foot-gun':374 'forc':2208,2794 'force-stop':2207 'foreground':608,627,752,857,1495,4147 'foregroundservicetyp':4167 'forev':248 'forget':3678 'form':369,2997 'forward':1878,2140,3927 'found':931,1554,3067,3842 'four':594,1302,3546,4146 'framework':66,154,1320,1355,1363,3858,4114 'fresh':934,979 'full':4254,4343 'full-screen':4253 'function':726,1155,1225,2235 'futur':910 'g':2352,2437 'gate':1589,3437 'generat':431,685,3376 'generatetoken':234 'gestur':148,1289,4322,4334,4395 'get':2469,3695,4300 'getdefaultconfig':3002,3039 'getloggedinus':918 'go':666,1542,1714,3609,3618,3636,4313 'goe':825 'google-services.json':1641,3725,4158 'grade':49 'gradl':29,2845,3739 'ground':169 'group':2351,2360,2385,2401,2402,2436,2446,2467,2505,2509,2523,2554,2582,2598 'group-cal':2597 'guard':1166,1214,4201 'gun':376 'handl':635,1123,2508,3468,3790 'handler':149,1290,4323,4335,4396 'hangup':710,3682,4214,4281 'happen':748,755 'hard':133,183 'hardcod':2076 'harden':1303 'hash':1875 'head':4257 'heads-up':4256 'headset':1460,1506 'helper':2326 'hide':1035 'high':554,578,3721 'high-prior':553 'highest':510 'highest-effort':509 'histori':2384,3505 'homebrew':1518,1555,4139 'hook':1396,4108 'hooks/usecallkeep.ts':3439 'host':3487,3903 'hot':1022,2082,3948 'hot-bundl':2081 'hot-reload':1021,3947 'i386':1377,1433,3879,4127 'icon':2355 'id':302,2257,2268 'id/cometchat_region/cometchat_auth_key':1205 'imper':463 'implement':3495 'implic':2455 'import':254,258,406,410,2222,2226,2288,2962,2976,3091,3191,3237 'in-cal':2451 'includ':4216 'incom':530,2553,3470,3716,4236,4260 'incoming-cal':3469 'incoming/outgoing/ongoing':3407 'incomingcal':818,2375 'incomplet':1157 'indefinit':1097,3788 'index':3194 'info.plist':784,3509,4086 'init':22,144,936,1169,1192,2219,2890,3219,3291,3429,3435,3848,4321 'init.ts':2992 'initcometchat':2236 'initi':396,416,425,1018,1895,2231,2238,2277,2377,3239,3365,3387,4222 'initiatecal':232 'initiated.getsessionid':448 'inlin':117 'inmeet':3307 'innerhtml':2885 'insid':749,756,826,1228,2358,2966,3217,3315,3659 'instal':18,1249,1253,1275,1285,1296,1307,1323,1358,1395,1422,1567,1778,1846,1864,1991,2022,2654,2691,2770,2792,3225,3642,3644,3855,3883,3975,3994,4034,4076,4119,4131,4333,4391 'instanceof':1028,3823 'instant':1491 'instead':862,2289,3703 'integr':8,113,900,2613,3531,3537 'intel':1388,3888 'intel-era':1387,3887 'intercept':589,3708 'intern':440,481,1082,1139,3775 'invok':1348 'io':26,100,523,540,565,717,743,783,1301,1315,1353,1371,1572,1578,1585,1599,1604,1622,2578,3064,3461,3640,3873,4085,4100,4112,4120,4132,4229,4238 'ios-on-rn':1571 'ios/.xcode.env.local':1511,1532 'ios/info.plist':1435 'ios/podfile':1379 'iphonesimul':1431 'isn':2841 'issu':1632 'iter':2072 'javascript':2310 'jitsiwebrtc':1406,1615 'join':2391,2471,2478,3379,4245,4268 'joinsess':233,464 'js':3000,3096 'json.stringify':1031,1057,1195,1198,3826 'jump':2448 'keep':2063 'key':214,224,273,281,304,368,3525 'kill':3847,4301 'kit':112,478,1080,1117,1136,1162,1300,1906,1925,2341,2460,2574,2625,3408,3773,3795,4345 'kit-bas':2459 'land':3989 'landmin':1739,1830,4031 'launch':317,935,980 'layer':104,1894 'leav':760 'legaci':1866,1898,1967,1993,2036,2693,3112,3980 'legacy-peer-dep':1865,1897,1966,1992,2035,2692,3111,3979 'let':1020,1115,2230,3793 'level':1103,4221 'like':2407,2604 'line':909,1416,2771,2959 'linger':4286 'link':25,1410,1606,1933,2025,3331,3373,3895,4017 'link-driven':3330 'linkag':1331,3862 'list':1763,2492,2495,3123 'listen':849,869,2370,2525,3318,3448,3665 'live':1852,3481 'load':60,1339,1474,2145,2685,3621,3869,3936 'load-bear':1473,2684 'lock':767,3692,4232,4241 'lock-screen':766,3691 'log':962 'logged-in':961 'login':145,190,245,264,268,315,356,377,442,984,3436,3850 'look':2603 'lose':1494,2137,3663,3931 'loud':2593 'machin':3306 'major':2922 'manag':13,57,161,664,804,1648,1655,1670,1687,3255,4150 'mandatori':95,2653 'manifest':694,816,1634,2048,4143,4394 'match':744,1159,2651,2894 'maven':2832 'may':1526,1876 'media':3418 'meet':2441,2472,2510,2519,2524,2535,2640,3304,3372 'meeting-card':2440 'meeting-room':2639 'meetingmessag':2439 'merg':682,813 'mergeconfig':3003,3038 'messag':557,2389,2443,2468,2491,2587,3347,4375 'message-bas':2388 'metro':1208,2080,2129,2161,2939,2970,3054,3929,3946,3967,4388 'metro.config.js':2999 'mic':4284 'microphon':1448,3352 'mid':1508 'mid-cal':1507 'minimum':1436 'minor':2923 'mint':699,4382 'miss':1188,2672,2785,3129,3646,3723 'mix':3742 'mjs':2948 'mode':76,132,283,598,600,650,824,853,1235,1669,2344,2629,2638,2679,3127,3231,3328,3363,3404,4173,4348 'modul':24,1653,1983,2416,2668,2789,3028,3093,3141,3158,3174,3615,3648,4220 'module-level':4219 'module.exports':3037 'modulenam':3018,3021,3035 'monkey':2213 'mount':118,819,2369,2373,3558,3658,3669,4169 'move':3782 'msg':1026,1034,2529 'msg.getcategory':2531 'msg.getcustomdata':2538,2544 'msg.gettype':2534 'must':217,563,629,642,912,2656,3121,3712 'name':1241,2306 'nativ':3,11,23,35,39,45,55,72,99,110,140,158,165,497,519,536,548,615,633,652,677,705,789,815,866,1256,1260,1266,1271,1278,1288,1293,1368,1404,1580,1584,1613,1628,1652,1682,1748,1776,1815,1821,1914,1958,1963,2006,2011,2102,2115,2158,2188,2193,2284,2296,2329,2577,2579,2703,2708,2713,2720,2729,2734,2740,2745,2819,2827,2902,2911,3007,3047,3134,3144,3153,3161,3168,3177,3185,3208,3213,3340,3349,3444,3457,3464,3507,3535,3586,3614,3647,3745,3907,3956,4005,4010,4061,4066,4318,4328,4341,4351,4378,4386 'navig':829,1113,2559,3595,3662,3674 'ndk':1779 'need':1907,2160,3295,3754,3966 'negoti':886 'never':1106,1158,3236,3238,3364 'new':419,874,1186,1791,1797,2242,2261,2526 'no-sess':921,3844 'node':1512,1519,1534,1551,1562,2415,2788,3027,3092,4134,4140 'non':331,885 'non-negoti':884 'non-nul':330 'none':1802 'note':2688 'noth':2499,2602,3384 'notif':42,539,584,800,1263,1468,3460,3702,3709,4071,4093,4265,4296 'npm':1248,1252,1274,1284,1844,1845,1863,1970,1990,2690,2791,3109,3990,4033,4081,4082 'npx':1365,2020,2190,3044 'nsbluetoothalway':1496 'nsbluetoothalwaysusagedescript':1455 'nscamerausagedescript':785,1441 'nsmicrophoneusagedescript':786,1447 'null':240,332,926,952,970,3651,3833,4184 'nvm':1523,1538,1545 'object':999,1007,1008,3819,3820 'often':1384 'older':2905,3751 'onaccept':1073,1111,1147,3764,4175 'onconnectionclos':3317 'oncustommessagereceiv':2528 'ondeclin':1129,3791 'one':908,1476,1974,2671,3098 'onerror':1132,3792 'ongo':625,2562,4246,4269,4294 'ongoing-cal':624,2561,4293 'ongoingcal':1084,1114,1144,3777 'onincomingcallreceiv':2430 'onitemclick':2382 'onto':105 'opaqu':1219 'open':4243,4266 'oper':73 'opt':3581 'opt-in':3580 'org.asyncstorage.shared':2834,2851 'os':533,771,878 'outgo':418,428,1092,4272 'output':3740 'overlap':4357 'overlay':3494 'overload':351 'overrid':1780 'p':2214 'packag':404,1851,2774,4038,4056 'page':1882 'pagin':3504 'pars':2973,3085 'pass':1072,3288,3763 'patch':2941,2998,3043,3550 'path':155,1121,1520,1524,1539,1605,3010,3012,3221,3683,3799,4141,4215,4363 'path.resolve':3025 'pattern':137,160,167,249,883,2298,3605,4331 'payload':544,582,3472,3710 'peer':151,1282,1867,1899,1903,1922,1968,1994,2037,2433,2694,2898,3074,3113,3130,3253,3260,3262,3981,3987,3993,4324 'peer-dep':150,3252 'peerdepend':1924 'perform':1964,2012,2741,3214,4011 'permiss':781,1635,2709,3353,4149 'permissionsandroid.requestmultiple':794 'persist':247,314,770 'pick':3599,3951 'picker':1772,3302 'piec':512,595 'pin':2682,2800,2815,2853,2891,2931 'pipe':2331 'pixel':2811 'pkill':2183 'plain':998 'platform':500,2324,3019,3036,4228 'platform-specif':499,2323 'plugin':674,680,1678,2173,3963 'plumb':2481 'plus':1947,4012 'pod':27,1306,1322,1328,1357,1616,3641,3854,3898,4118,4332,4390 'podfil':3881,4129 'point':1514,4136 'pointer':4315 'polyfil':1960,2008,2747,2866,2889,3202,3210,4007 'polyfills/browser.js':3180,3199 'port':2139,3926 'port-forward':2138 'post':799,1394,1421,3882,4130 'post-instal':1393 'prebuild':688,812,2055,2067,2089 'prebuild-tim':2066 'presenc':3263 'prevent':1497 'previous':1758 'primit':2304 'prioriti':555,577,3720 'privaci':4393 'process':2174 'process.env.expo':1174,2246,2253,2264,2270,2291 'produc':1218,1333,3863 'product':48,81,286,515,706,2615,4379 'production-grad':47 'profil':1710 'project':1668,1683 'promis':381,3201 'promise.allsettled':1952,2000,2759,3197,3999 'prompt':601,1691 'prop':3574,4176 'properti':2884,2986 'prototyp':2872,2877 'provid':1146,3433,4202,4320 'provider/scaffold':882 'prune':1546 'public':177,1175,1716,2247,2254,2265,2271,2292 'publish':2844 'pull':1929,3243 'purpos':46,2347 'push':31,41,93,494,502,504,538,561,1141,1262,3326,3361,3393,3459,3518,3547,3579,3668,3700,4070,4352,4360,4365,4370 'pushkit':541,568,3343,4101 'pushkit/fcm':4372 'rang':2899,2935 'rc':1346 're':355,1782,2776 're-login':354 're-run':2775 're-valid':1781 'reach':983,1065 'react':10,34,38,44,54,71,109,496,518,535,547,632,676,865,1255,1259,1265,1270,1277,1287,1292,1367,1403,1612,1747,1775,1814,1820,1913,1957,1962,2005,2010,2101,2114,2157,2187,2192,2283,2317,2575,2702,2707,2712,2719,2728,2733,2739,2744,2818,2901,2910,3006,3046,3133,3143,3152,3160,3167,3176,3184,3207,3212,3339,3443,3456,3463,3744,3906,3955,4004,4009,4060,4065 'react-nat':43,70,1366,2100,2113,2186,2191,3005,3045,3905 'react-native-async-storag':2718,2817,3132,3142 'react-native-background-tim':2732,3166,3175 'react-native-callkeep':33,495,517,631,675,864,1254,3338,3442,3743 'react-native-dotenv':2156,2282,3954 'react-native-firebas':546,1264,1269,3462 'react-native-gesture-handl':1286 'react-native-perform':1961,2009,2738,3211,4008 'react-native-permiss':2706 'react-native-reanim':1291 'react-native-safe-area-context':2711 'react-native-svg':2727,3151,3159 'react-native-url-polyfil':1956,2004,2743,3206,4003 'react-native-voip-push-notif':37,534,1258,3455 'react-native-webrtc':1276,1402,1611,2701,2900,3183 'react-native-worklet':1774 'read':123,2064,2985 'real':309,1735,1829,3101,4225 'reanim':1294 'receiv':2457,2498 'receiveruid':421 'recipi':2600 'recommend':1165,4213 'record':796 'refer':2339 'references/call-session.md':3322 'references/custom-ui.md':491 'refus':1728,3631 'regener':1681 'region':1184,1197,1199,2249,2273,3281 'regist':3446,4164 'registr':543,636,1645,4103 'registri':1857 'reinstal':1984 'reject':382 'releas':4282 'reload':1023,2058,3912,3949 'remot':1467,4092 'remote-notif':1466,4091 'remov':1373,1418,1773,1827,3899 'render':453,485,987,1054,2466,2474,2488,3311,3571,4189 'reorgan':2875 'replac':1148 'report':529,873 'request':792 'requir':671,782,1324,1651,1695,3004,3011,3107,3526,3613 'reset':2163,2196,3050,3319,3969 'reset-cach':2162,2195,3049,3968 'resolv':1982,2014,2667,3015,3056,3140,3157,3173 'resolverequest':3016 'restart':1207,2128,3930 'retri':1024 'return':328,946,955,1848,2239,2977,3023,3032,4036,4083 'revers':2108,2121,2150,2199,3915 'revis':1874 'right':262 'ring':397,844,2394,2428,2590,2644,3335,3386,3403,4234,4252 'ringing-driven':3334 'rn':90,185,450,469,506,690,719,890,957,1245,1382,1574,1602,1621,1796,1799,1989,2110,2169,2280,2320,2675,2808,2813,3258,3655,3749,3752,3760,3837,3939,3972 'rn-exist':1798 'rn-new':1795 'rn-on-io':1601 'rncallkeep.displayincomingcall':3473 'rncallkeep.endcall':736,759,3500,3679,4218 'roll':1877 'room':2641 'root':122,822,828,872,2368,3561 'root-mount':2367 'rout':2942 'router':1946 'row':2356 'rtcpeerconnect':712 'rubi':1417 'rule':97,134,153,184,613,660,718,839,1636,3496,3562,3675,3766,3807,3834,4096,4177,4185,4195,4209 'run':399,630,1370,1657,1720,2087,2092,2106,2117,2657,2777,3060,3063,3852,3922 'run-android':2116,3059,3921 'run-io':1369,3062 'runtim':791,1342,1889,1909,1921,2180,2879,2982,3106,3215,3652,3872,4224 'safe':2714 'safeareaview':3316 'sampl':174,2337 'sample-app':2336 'save':2793 'scaffold':592,911,1676,1730,2659,3276,3422,3633 'screen':91,768,846,1010,1067,1093,3601,3693,3817,4233,4242,4255 'screens/calllogsscreen.tsx':3503 'screens/callroom.tsx':3310 'screens/joinsession.tsx':3300 'screens/ongoingcallscreen.tsx':3486 'screens/profilescreen.tsx':3475 'script':2146,3937 'sdk':7,21,83,86,143,171,189,199,209,313,322,359,389,415,430,456,958,974,995,1230,1281,1430,1650,1742,1766,1819,1825,1927,2017,2303,2311,2321,2646,2650,2864,2895,2918,2945,3087,3193,3218,3266,3271,3358,3411,3413,3417,3426,3428,3811,3838,4059,4064 'sdk-compat':2016 'sdks':3389 'second':3297 'section':2312 'see':490,2293,2364,2601,2687,3321,4028 'self':2857 'self-contain':2856 'semant':2362,2567 'send':566,573,3698,3713 'separ':205,2831 'sequenti':3430 'server':558,562,698,1102,3516,3711,4381 'server-mint':697,4380 'server-sid':3515 'servic':609,628,753,1644,4148,4156,4163 'services/voippush.ts':3453 'session':402,714,746,923,2637,2677,3229,3268,3327,3362,3381,3846 'session-on':2676,3228 'session/join':451 'sessionid':437,2537,2541,2565,3718 'set':466,1343,1427,1437,2119,2655,2883,3440,3576,3924 'setappid':2263 'setcallreceiv':1130,1133 'seterror':1003,1013,1033,3801 'setregion':2245,2269 'setup':1243,3512,4030,4099,4338 'seven':4055 'shape':392,2307,2634,2873 'shell':1345,2205,2212 'shim':735 'ship':2952 'short':1077,3770 'short-circuit':1076,3769 'show':1006,1801,2142,2550,3818 'side':559,3517 'signal':84,3414 'silent':647,898,1332,1901,2765,3116,3991 'silent-crash':646 'silicon':1314,1381,1399,1425,3886,3902,4122 'sim':1317 'simul':1413,3894 'singl':347,526,1044 'single-arg':346 'sink':1049 'skill':126,243,591,1665,1723,3274,3421,3541,3630,3758 'skill-cometchat-native-calls' 'skip':758,1971,2766,3638,3992 'slice':1408 'snapshot':2069 'sometim':2764 'sourc':172,2084,2414,3235 'source-cometchat' 'sourcefil':3031 'special':186 'specif':501,2325,2921 'split':564,2626,2825 'src/config':2078 'stabl':1517,2929,4138 'stack':516,1052,3394,3667 'stack/tab':3661 'stacks/tabs':837 'stale':1543 'standalon':77,597,852,2130,2612,2636,3402,3928,4309 'standard':514,2427,4368 'start':336,2103,2189,2194,3048,3908 'start/join':3303 'state':204,1724,3305,3320 'static':1321,1356,1364,3859,4053,4115 'stay':1094,3785 'step':194,1304 'still':3689 'stop':754,2209 'storag':2722,2821,2835,2837,2852,3136,3146,3246 'storage-android':2836 'straight':2449 'string':990,1004,1014,3802,4095,4193 'strip':1902,3117 'stuck':765,939,3696 'subclass':1002,3816 'subscribepresenceforallus':2244 'subsequ':1360 'substanti':606,3396,3585 'succeed':215 'support':2909 'surfac':294,379,475,1145,1177,1610,2420,2454,3489 'surpris':305,2938 'surviv':1481,1568 'svg':2730,3154,3162 'swipe':4303 'symbol':3645 'symlink':1559 'symptom':1086,2596,3131,3933 'syntax':2965 'system':762,4287 'take':435 'tap':1088,2476,2557,3370 'tarbal':1860,2763,4048,4079 'tcp':2122,2124,2151,2153,2200,2202,3916,3918 'telecom':4154 'tell':738 'templat':1383 'termin':4230,4249 'test':2806,3071,3079 'text':1064,1954,2002,2754,3204,4001 'text-encod':1953,2001,2753,3203,4000 'think':772,3685 'three':880,915,1764,1805 'throw':235,919,928,1185,3839,4203 'thrown':967 'time':1048,1170,1193,1350,1738,2068,2178,3104,3190,3732,3962 'timer':2736,3170,3179 'togeth':1928 'token':237,433,444,465,542,701,3378,4102,4383 'topic-agent-skills' 'topic-ai-agent' 'topic-chat' 'topic-claude-code' 'topic-cometchat' 'topic-cursor' 'topic-messaging' 'topic-nextjs' 'topic-react' 'topic-react-native' 'topic-ui-kit' 'total':3076 'track':2330 'transfer':252 'transit':1085,1107,1617,1920,3242,3778,3986 'trap':1769 'treat':959 'tri':274 'trigger':3480 'troubleshoot':4387 'true':2278 'truth':170 'try/catch':385 'ts':253,405,725,927,992,1181,2079,2220,2521 'tsx':1075 'two':75,2632,4276 'two-way':4275 'type':2520,3030,3715 'typeerror':1220 'typic':1156,2357 'typo':299,1238 'u':2350,2424 'ui':111,484,764,858,1105,1299,2340,2484,2556,2624,3493,3694,3781,4289,4344 'uibackgroundmod':1463,1469,4088 'uid':212,222,271,279,350,366,3294,3301 'uikit':1746,1813,1912 'unabl':1980,2143,2665,3138,3155,3171,3934 'undefin':1172,1215,1221,2887,2978,2989,4208 'under':1037 'unlik':971 'upgrad':1570,2924 'upstream':1590 'url':1861,1959,2007,2746,3209,4006,4049 'use':861,1298,1319,1354,1362,1459,1858,2281,2387,2464,3216,3607,3625,3821,3857,4043,4113,4116,4190 'user':296,333,929,964,1041,1502,1693,2349,2361,2411,2423,3573,3584,3598,3840 'ux':2642 'v0.9.x':2874 'v3.x':2824,3756 'v4':3747 'v5':195,197,434,658,729,1579,1586 'valibot':1948,1996,2749,2914,3181,3995 'valid':887,1308,1751,1783,2396,2802 'valu':1176,1217 'var':4200 'variant':2412 'vendor':1631 'verif':4051 'verifi':3256 'version':1759,1886,2019,2681,2799,2906,3753,3761 'via':32,318,454,793,1943,2585,3490,4077 'video':51,1445,1453,2354,2549,2618,3483,4279 'view':2342,2380 'viewref':467 'visibl':4297 'voic':50,1451,2353,2617,3482 'voice-video':2616 'voip':30,40,92,493,503,537,569,1261,1465,3325,3360,3392,3458,3522,3578,4069,4090,4359,4369 'voip-push':3391 'voip-push-notif':4068 'way':904,4277 'web':394,716,841,973 'webrtc':87,401,721,1279,1326,1340,1405,1609,1614,1893,2704,2903,3186,3259,3870,3897,4074 'weight':3233 'whatev':1561 'wherev':3477 'wide':2843 'wipe':1211 'wire':3565 'without':225,975,1212,1486,1659,1688,1731,1887,3831,3856,3913 'work':344,1575 'workaround':1390,3890 'workflow':2181 'worklet':1777 'wrap':383,479,723 'wrapper':834 'www.cometchat.com':180 'www.cometchat.com/docs/calls/react-native/overview':179 'x':1564,1750 'xcode':1547 'xcode.env.local':4133 'xml':644,1440,1646 'xmldom':2870 'xmldom/xmldom':1950,1998,2725,2860,3195,3997 'zero':3399 'zustand':1949,1997,2751,2916,3182,3996","prices":[{"id":"72b8a346-e074-4831-932c-e824b878daca","listingId":"0bf161c9-9cfe-4a49-a049-a7ca0217d6ef","amountUsd":"0","unit":"free","nativeCurrency":null,"nativeAmount":null,"chain":null,"payTo":null,"paymentMethod":"skill-free","isPrimary":true,"details":{"org":"cometchat","category":"cometchat-skills","install_from":"skills.sh"},"createdAt":"2026-05-18T07:04:27.553Z"}],"sources":[{"listingId":"0bf161c9-9cfe-4a49-a049-a7ca0217d6ef","source":"github","sourceId":"cometchat/cometchat-skills/cometchat-native-calls","sourceUrl":"https://github.com/cometchat/cometchat-skills/tree/main/skills/cometchat-native-calls","isPrimary":false,"firstSeenAt":"2026-05-18T07:04:27.553Z","lastSeenAt":"2026-05-18T19:04:54.014Z"}],"details":{"listingId":"0bf161c9-9cfe-4a49-a049-a7ca0217d6ef","quickStartSnippet":null,"exampleRequest":null,"exampleResponse":null,"schema":null,"openapiUrl":null,"agentsTxtUrl":null,"citations":[],"useCases":[],"bestFor":[],"notFor":[],"kindDetails":{"org":"cometchat","slug":"cometchat-native-calls","github":{"repo":"cometchat/cometchat-skills","stars":27,"topics":["agent-skills","ai-agent","chat","claude-code","cometchat","cursor","messaging","nextjs","react","react-native","ui-kit"],"license":null,"html_url":"https://github.com/cometchat/cometchat-skills","pushed_at":"2026-05-18T05:04:24Z","description":"Add CometChat chat to any React, Next.js, React Native, Angular, Android, iOS, or Flutter project through your AI coding agent. Works with Claude Code, Cursor, Codex, VS Code Copilot, Windsurf, Cline, Kiro, and 50+ more agents.","skill_md_sha":"6edc1b4beb7134b08b29f99e30d6c445d8cce351","skill_md_path":"skills/cometchat-native-calls/SKILL.md","default_branch":"main","skill_tree_url":"https://github.com/cometchat/cometchat-skills/tree/main/skills/cometchat-native-calls"},"layout":"multi","source":"github","category":"cometchat-skills","frontmatter":{"name":"cometchat-native-calls","license":"MIT","description":"CometChat Calls SDK integration for React Native (Expo managed + bare CLI). Covers @cometchat/calls-sdk-react-native install, dual-SDK init, native module linking (iOS pods, Android Gradle), VoIP push via react-native-callkeep + react-native-voip-push-notification + @react-native-firebase/messaging, CallKit on iOS / ConnectionService on Android, foreground service correctness on Android 14+, gesture handler + reanimated peer deps, Expo-specific config plugins, and additive-vs-standalone modes.","compatibility":"React Native >= 0.70 (>= 0.72 recommended), Expo SDK >= 49 (managed) / bare RN CLI; @cometchat/calls-sdk-react-native ^4.x; @cometchat/chat-sdk-react-native ^4.x; @cometchat/chat-uikit-react-native ^5.x (additive mode)"},"skills_sh_url":"https://skills.sh/cometchat/cometchat-skills/cometchat-native-calls"},"updatedAt":"2026-05-18T19:04:54.014Z"}}