react-native-acoustic-connect-beta 18.0.34 → 18.0.36
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 +2 -0
- package/README.md +117 -2
- package/android/src/main/assets/ConnectBasicConfig.properties +1 -1
- package/app.plugin.js +15 -0
- package/package.json +15 -1
- package/plugin/build/index.d.ts +25 -0
- package/plugin/build/index.d.ts.map +1 -0
- package/plugin/build/index.js +44 -0
- package/plugin/build/index.js.map +1 -0
- package/plugin/build/withConnectNCE.d.ts +88 -0
- package/plugin/build/withConnectNCE.d.ts.map +1 -0
- package/plugin/build/withConnectNCE.js +366 -0
- package/plugin/build/withConnectNCE.js.map +1 -0
- package/plugin/build/withConnectNSE.d.ts +120 -0
- package/plugin/build/withConnectNSE.d.ts.map +1 -0
- package/plugin/build/withConnectNSE.js +512 -0
- package/plugin/build/withConnectNSE.js.map +1 -0
- package/plugin/src/index.ts +50 -0
- package/plugin/src/withConnectNCE.ts +487 -0
- package/plugin/src/withConnectNSE.ts +689 -0
- package/plugin/swift/NotificationService.swift +34 -0
- package/plugin/swift/NotificationViewController.swift +34 -0
|
@@ -0,0 +1,487 @@
|
|
|
1
|
+
// Copyright (C) 2026 Acoustic, L.P. All rights reserved.
|
|
2
|
+
//
|
|
3
|
+
// NOTICE: This file contains material that is confidential and proprietary to
|
|
4
|
+
// Acoustic, L.P. and/or other developers. No license is granted under any
|
|
5
|
+
// intellectual or industrial property rights of Acoustic, L.P. except as may
|
|
6
|
+
// be provided in an agreement with Acoustic, L.P. Any unauthorized copying or
|
|
7
|
+
// distribution of content from this file is prohibited.
|
|
8
|
+
|
|
9
|
+
import {
|
|
10
|
+
withXcodeProject,
|
|
11
|
+
withDangerousMod,
|
|
12
|
+
type ConfigPlugin,
|
|
13
|
+
} from '@expo/config-plugins'
|
|
14
|
+
import type { ExpoConfig } from '@expo/config-types'
|
|
15
|
+
import * as fs from 'fs'
|
|
16
|
+
import * as path from 'path'
|
|
17
|
+
import {
|
|
18
|
+
resolveAppGroupIdentifier,
|
|
19
|
+
substituteSwiftTemplate,
|
|
20
|
+
getHostBuildSettingForConfig,
|
|
21
|
+
buildConnectPodTargetBlock,
|
|
22
|
+
withNSEEntitlements,
|
|
23
|
+
type ConnectPluginProps,
|
|
24
|
+
} from './withConnectNSE'
|
|
25
|
+
|
|
26
|
+
// XcodeProject is declared in the `xcode` package which ships no TypeScript
|
|
27
|
+
// types of its own. We derive the type from the withXcodeProject callback
|
|
28
|
+
// so the compiler can still check our usage without requiring @types/xcode.
|
|
29
|
+
type XcodeProject = Parameters<
|
|
30
|
+
Parameters<typeof withXcodeProject>[1]
|
|
31
|
+
>[0]['modResults']
|
|
32
|
+
|
|
33
|
+
// ─── Target / config constants ─────────────────────────────────────────────
|
|
34
|
+
|
|
35
|
+
const NSE_TARGET_NAME = 'ConnectNSE'
|
|
36
|
+
const NCE_TARGET_NAME = 'ConnectNCE'
|
|
37
|
+
|
|
38
|
+
// Configs belonging to either extension target must be skipped when mirroring
|
|
39
|
+
// host build settings. The NCE mod runs AFTER the NSE mod in the composition,
|
|
40
|
+
// so by the time this executes the pbxproj already carries ConnectNSE configs;
|
|
41
|
+
// without skipping both, the NCE could mirror the NSE target's settings rather
|
|
42
|
+
// than the host app's.
|
|
43
|
+
const EXTENSION_TARGET_NAMES = [NSE_TARGET_NAME, NCE_TARGET_NAME]
|
|
44
|
+
|
|
45
|
+
// ─── Plist helpers ────────────────────────────────────────────────────────────
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Generates the Info.plist content for the ConnectNCE target.
|
|
49
|
+
*
|
|
50
|
+
* The NSExtensionAttributes mirror the canonical bare-workflow reference
|
|
51
|
+
* (Examples/bare-workflow/ios/ConnectNCE/Info.plist), which itself mirrors the
|
|
52
|
+
* iOS-SDK NCE 1:1:
|
|
53
|
+
* - UNNotificationExtensionCategory: the category identifiers the Connect
|
|
54
|
+
* APNs payload routes rich-media notifications through. Must match the
|
|
55
|
+
* category the SDK/Collector sets — keep in sync with the iOS-SDK NCE.
|
|
56
|
+
* - UNNotificationExtensionDefaultContentHidden=false: keep the default
|
|
57
|
+
* system body alongside the custom content view.
|
|
58
|
+
* - UNNotificationExtensionInitialContentSizeRatio=1.0: start full-height;
|
|
59
|
+
* the SDK content controller resizes to fit the rendered media.
|
|
60
|
+
*
|
|
61
|
+
* RCTNewArchEnabled is intentionally omitted: the NCE links the Connect SDK
|
|
62
|
+
* only — no React Native runtime is present in the extension process. (The
|
|
63
|
+
* bare-workflow reference carries it as an Xcode-template leftover; the
|
|
64
|
+
* plugin-generated target drops it, consistent with the NSE template.)
|
|
65
|
+
*/
|
|
66
|
+
export function buildNCEInfoPlist(): string {
|
|
67
|
+
return `<?xml version="1.0" encoding="UTF-8"?>
|
|
68
|
+
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
|
69
|
+
<plist version="1.0">
|
|
70
|
+
<dict>
|
|
71
|
+
\t<key>CFBundleDevelopmentRegion</key>
|
|
72
|
+
\t<string>$(DEVELOPMENT_LANGUAGE)</string>
|
|
73
|
+
\t<key>CFBundleDisplayName</key>
|
|
74
|
+
\t<string>ConnectNCE</string>
|
|
75
|
+
\t<key>CFBundleExecutable</key>
|
|
76
|
+
\t<string>$(EXECUTABLE_NAME)</string>
|
|
77
|
+
\t<key>CFBundleIdentifier</key>
|
|
78
|
+
\t<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
|
|
79
|
+
\t<key>CFBundleInfoDictionaryVersion</key>
|
|
80
|
+
\t<string>6.0</string>
|
|
81
|
+
\t<key>CFBundleName</key>
|
|
82
|
+
\t<string>$(PRODUCT_NAME)</string>
|
|
83
|
+
\t<key>CFBundlePackageType</key>
|
|
84
|
+
\t<string>$(PRODUCT_BUNDLE_PACKAGE_TYPE)</string>
|
|
85
|
+
\t<key>CFBundleShortVersionString</key>
|
|
86
|
+
\t<string>$(MARKETING_VERSION)</string>
|
|
87
|
+
\t<key>CFBundleVersion</key>
|
|
88
|
+
\t<string>$(CURRENT_PROJECT_VERSION)</string>
|
|
89
|
+
\t<key>NSExtension</key>
|
|
90
|
+
\t<dict>
|
|
91
|
+
\t\t<key>NSExtensionAttributes</key>
|
|
92
|
+
\t\t<dict>
|
|
93
|
+
\t\t\t<key>UNNotificationExtensionCategory</key>
|
|
94
|
+
\t\t\t<array>
|
|
95
|
+
\t\t\t\t<string>ACOUSTIC_RICH_NOTIFICATION</string>
|
|
96
|
+
\t\t\t\t<string>ACTIONABLE_NOTIFICATION</string>
|
|
97
|
+
\t\t\t</array>
|
|
98
|
+
\t\t\t<key>UNNotificationExtensionDefaultContentHidden</key>
|
|
99
|
+
\t\t\t<false/>
|
|
100
|
+
\t\t\t<key>UNNotificationExtensionInitialContentSizeRatio</key>
|
|
101
|
+
\t\t\t<real>1.0</real>
|
|
102
|
+
\t\t</dict>
|
|
103
|
+
\t\t<key>NSExtensionPointIdentifier</key>
|
|
104
|
+
\t\t<string>com.apple.usernotifications.content-extension</string>
|
|
105
|
+
\t\t<key>NSExtensionPrincipalClass</key>
|
|
106
|
+
\t\t<string>$(PRODUCT_MODULE_NAME).NotificationViewController</string>
|
|
107
|
+
\t</dict>
|
|
108
|
+
</dict>
|
|
109
|
+
</plist>
|
|
110
|
+
`
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
/** Generates the entitlements plist content for the ConnectNCE target. */
|
|
114
|
+
export function buildNCEEntitlements(appGroupIdentifier: string): string {
|
|
115
|
+
return `<?xml version="1.0" encoding="UTF-8"?>
|
|
116
|
+
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
|
117
|
+
<plist version="1.0">
|
|
118
|
+
<dict>
|
|
119
|
+
\t<key>com.apple.security.application-groups</key>
|
|
120
|
+
\t<array>
|
|
121
|
+
\t\t<string>${appGroupIdentifier}</string>
|
|
122
|
+
\t</array>
|
|
123
|
+
</dict>
|
|
124
|
+
</plist>
|
|
125
|
+
`
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// ─── Podfile injection ────────────────────────────────────────────────────────
|
|
129
|
+
|
|
130
|
+
const PODFILE_MARKER =
|
|
131
|
+
'# @generated ConnectNCE target (react-native-acoustic-connect-beta)'
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Returns the Ruby snippet to inject into the Expo-generated Podfile for the
|
|
135
|
+
* ConnectNCE target. Thin wrapper over the shared
|
|
136
|
+
* {@link buildConnectPodTargetBlock} (single source of truth for the
|
|
137
|
+
* AcousticConnect pod-resolution helper), parameterised with the NCE marker,
|
|
138
|
+
* helper name, and target name.
|
|
139
|
+
*/
|
|
140
|
+
export function buildPodfileBlock(): string {
|
|
141
|
+
return buildConnectPodTargetBlock(
|
|
142
|
+
PODFILE_MARKER,
|
|
143
|
+
'acoustic_connect_pod_nce',
|
|
144
|
+
NCE_TARGET_NAME
|
|
145
|
+
)
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
/**
|
|
149
|
+
* Injects the ConnectNCE Podfile block into `podfileContent` if not already
|
|
150
|
+
* present (idempotent, guarded by PODFILE_MARKER).
|
|
151
|
+
*/
|
|
152
|
+
export function injectPodfileBlock(podfileContent: string): string {
|
|
153
|
+
if (podfileContent.includes(PODFILE_MARKER)) {
|
|
154
|
+
return podfileContent
|
|
155
|
+
}
|
|
156
|
+
return podfileContent + buildPodfileBlock()
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
// ─── Xcode project mod ───────────────────────────────────────────────────────
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* Adds the ConnectNCE Xcode target (app_extension) to the project.
|
|
163
|
+
*
|
|
164
|
+
* Sibling to withNSEXcodeProject; the only differences are the target name,
|
|
165
|
+
* the Swift source file (NotificationViewController.swift), the NCE Info.plist
|
|
166
|
+
* (content-extension NSExtension keys), and skipping BOTH extension targets'
|
|
167
|
+
* configs when mirroring host build settings.
|
|
168
|
+
*
|
|
169
|
+
* Idempotent: if a target named ConnectNCE already exists, skips target
|
|
170
|
+
* creation but still (re)writes the source files.
|
|
171
|
+
*/
|
|
172
|
+
export function withNCEXcodeProject(
|
|
173
|
+
config: ExpoConfig,
|
|
174
|
+
appGroupIdentifier: string,
|
|
175
|
+
swiftContent: string
|
|
176
|
+
): ExpoConfig {
|
|
177
|
+
return withXcodeProject(config, (c) => {
|
|
178
|
+
const xcodeProject: XcodeProject = c.modResults
|
|
179
|
+
|
|
180
|
+
// ios/ directory — modRequest.platformProjectRoot is the canonical
|
|
181
|
+
// native project root. (xcodeProject.filepath points INSIDE the
|
|
182
|
+
// .xcodeproj bundle, so deriving from it misplaces the files.)
|
|
183
|
+
const iosDir = c.modRequest.platformProjectRoot
|
|
184
|
+
const nceDir = path.join(iosDir, NCE_TARGET_NAME)
|
|
185
|
+
|
|
186
|
+
// Always write / overwrite the source files (idempotent)
|
|
187
|
+
fs.mkdirSync(nceDir, { recursive: true })
|
|
188
|
+
|
|
189
|
+
fs.writeFileSync(
|
|
190
|
+
path.join(nceDir, 'NotificationViewController.swift'),
|
|
191
|
+
swiftContent,
|
|
192
|
+
'utf8'
|
|
193
|
+
)
|
|
194
|
+
fs.writeFileSync(
|
|
195
|
+
path.join(nceDir, 'Info.plist'),
|
|
196
|
+
buildNCEInfoPlist(),
|
|
197
|
+
'utf8'
|
|
198
|
+
)
|
|
199
|
+
fs.writeFileSync(
|
|
200
|
+
path.join(nceDir, `${NCE_TARGET_NAME}.entitlements`),
|
|
201
|
+
buildNCEEntitlements(appGroupIdentifier),
|
|
202
|
+
'utf8'
|
|
203
|
+
)
|
|
204
|
+
|
|
205
|
+
// Idempotency check — skip if target already registered in the pbxproj.
|
|
206
|
+
// Note: pbxTargetByName misses targets on a REPARSED project because the
|
|
207
|
+
// xcode lib stores written names with literal quotes ('"ConnectNCE"'),
|
|
208
|
+
// so compare quote-stripped names across the native-target section.
|
|
209
|
+
const nativeTargets = xcodeProject.pbxNativeTargetSection() as Record<
|
|
210
|
+
string,
|
|
211
|
+
{ name?: string } | string
|
|
212
|
+
>
|
|
213
|
+
const targetExists = Object.values(nativeTargets).some(
|
|
214
|
+
(t) =>
|
|
215
|
+
typeof t === 'object' &&
|
|
216
|
+
typeof t.name === 'string' &&
|
|
217
|
+
t.name.replace(/"/g, '') === NCE_TARGET_NAME
|
|
218
|
+
)
|
|
219
|
+
if (targetExists) {
|
|
220
|
+
return c
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
// Derive the host bundle id — the NCE bundle id must be
|
|
224
|
+
// `<hostBundleId>.ConnectNCE`. Fail fast rather than emit a placeholder
|
|
225
|
+
// (a wrong bundle id breaks App Group pairing and App Store submission).
|
|
226
|
+
const hostBundleId = c.ios?.bundleIdentifier as string | undefined
|
|
227
|
+
if (!hostBundleId) {
|
|
228
|
+
throw new Error(
|
|
229
|
+
`[react-native-acoustic-connect-beta] ios.bundleIdentifier is not set.\n\n` +
|
|
230
|
+
`The ConnectNCE extension bundle id is derived as ` +
|
|
231
|
+
`"<ios.bundleIdentifier>.ConnectNCE", so the host bundle id must be ` +
|
|
232
|
+
`defined in app.json before prebuild:\n` +
|
|
233
|
+
` { "expo": { "ios": { "bundleIdentifier": "com.example.app" } } }`
|
|
234
|
+
)
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
// Add the native target (app_extension)
|
|
238
|
+
const nceTarget = xcodeProject.addTarget(
|
|
239
|
+
NCE_TARGET_NAME,
|
|
240
|
+
'app_extension',
|
|
241
|
+
NCE_TARGET_NAME,
|
|
242
|
+
`${hostBundleId}.ConnectNCE`
|
|
243
|
+
)
|
|
244
|
+
|
|
245
|
+
if (!nceTarget) {
|
|
246
|
+
throw new Error(
|
|
247
|
+
`[react-native-acoustic-connect-beta] Failed to add ConnectNCE Xcode target.`
|
|
248
|
+
)
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
// Add the build phase for Swift sources
|
|
252
|
+
xcodeProject.addBuildPhase(
|
|
253
|
+
['NotificationViewController.swift'],
|
|
254
|
+
'PBXSourcesBuildPhase',
|
|
255
|
+
'Sources',
|
|
256
|
+
nceTarget.uuid
|
|
257
|
+
)
|
|
258
|
+
|
|
259
|
+
// Add Resources build phase (empty, but required by Xcode)
|
|
260
|
+
xcodeProject.addBuildPhase(
|
|
261
|
+
[],
|
|
262
|
+
'PBXResourcesBuildPhase',
|
|
263
|
+
'Resources',
|
|
264
|
+
nceTarget.uuid
|
|
265
|
+
)
|
|
266
|
+
|
|
267
|
+
// Add Frameworks build phase (empty — CocoaPods populates it)
|
|
268
|
+
xcodeProject.addBuildPhase(
|
|
269
|
+
[],
|
|
270
|
+
'PBXFrameworksBuildPhase',
|
|
271
|
+
'Frameworks',
|
|
272
|
+
nceTarget.uuid
|
|
273
|
+
)
|
|
274
|
+
|
|
275
|
+
// Make the host app target depend on ConnectNCE. CocoaPods resolves an
|
|
276
|
+
// extension's host target through PBXTargetDependency
|
|
277
|
+
// (xcodeproj's host_targets_for_embedded_target), NOT through the embed
|
|
278
|
+
// copy phase — without this edge `pod install` fails with "Unable to find
|
|
279
|
+
// host target(s) for ConnectNCE".
|
|
280
|
+
// The xcode lib SILENTLY no-ops addTargetDependency when these sections
|
|
281
|
+
// are absent from the parsed project — create them first. (The NSE mod
|
|
282
|
+
// already creates them when it runs before this; guarding here keeps the
|
|
283
|
+
// NCE mod correct if it ever runs first.)
|
|
284
|
+
const objects = xcodeProject.hash.project.objects
|
|
285
|
+
objects['PBXTargetDependency'] = objects['PBXTargetDependency'] ?? {}
|
|
286
|
+
objects['PBXContainerItemProxy'] = objects['PBXContainerItemProxy'] ?? {}
|
|
287
|
+
const hostTargetUuid = xcodeProject.getFirstTarget().uuid
|
|
288
|
+
xcodeProject.addTargetDependency(hostTargetUuid, [nceTarget.uuid])
|
|
289
|
+
|
|
290
|
+
// Create a PBXGroup for ConnectNCE source files
|
|
291
|
+
const nceGroup = xcodeProject.addPbxGroup(
|
|
292
|
+
[
|
|
293
|
+
'NotificationViewController.swift',
|
|
294
|
+
'Info.plist',
|
|
295
|
+
`${NCE_TARGET_NAME}.entitlements`,
|
|
296
|
+
],
|
|
297
|
+
NCE_TARGET_NAME,
|
|
298
|
+
NCE_TARGET_NAME
|
|
299
|
+
)
|
|
300
|
+
|
|
301
|
+
// Add the group to the main project group
|
|
302
|
+
const mainGroup = xcodeProject.getFirstProject().firstProject.mainGroup
|
|
303
|
+
xcodeProject.addToPbxGroup(nceGroup.uuid, mainGroup)
|
|
304
|
+
|
|
305
|
+
// Set per-configuration build settings — mirror each host configuration
|
|
306
|
+
// individually so Debug gets Debug values and Release gets Release values.
|
|
307
|
+
const buildConfigurations = xcodeProject.pbxXCConfigurationList()
|
|
308
|
+
const nceConfigListUuid = nceTarget.pbxNativeTarget.buildConfigurationList
|
|
309
|
+
|
|
310
|
+
if (nceConfigListUuid) {
|
|
311
|
+
const configList = buildConfigurations[nceConfigListUuid]
|
|
312
|
+
if (
|
|
313
|
+
configList &&
|
|
314
|
+
Array.isArray(
|
|
315
|
+
(configList as { buildConfigurations?: unknown[] })
|
|
316
|
+
.buildConfigurations
|
|
317
|
+
)
|
|
318
|
+
) {
|
|
319
|
+
const configs = xcodeProject.pbxXCBuildConfigurationSection()
|
|
320
|
+
for (const entry of (
|
|
321
|
+
configList as { buildConfigurations: Array<{ value: string }> }
|
|
322
|
+
).buildConfigurations) {
|
|
323
|
+
const buildConfig = configs[entry.value]
|
|
324
|
+
if (buildConfig && buildConfig.buildSettings) {
|
|
325
|
+
// Determine which host config name to mirror (Debug → Debug, etc.)
|
|
326
|
+
const configName: string =
|
|
327
|
+
(buildConfig as { name?: string }).name ?? 'Debug'
|
|
328
|
+
|
|
329
|
+
// Mirror per-configuration values; fall back to the other config's
|
|
330
|
+
// value only when the matching one is absent. Skip BOTH extension
|
|
331
|
+
// targets so the NCE never mirrors the NSE target's settings.
|
|
332
|
+
const deploymentTarget = getHostBuildSettingForConfig(
|
|
333
|
+
xcodeProject,
|
|
334
|
+
'IPHONEOS_DEPLOYMENT_TARGET',
|
|
335
|
+
configName,
|
|
336
|
+
// Matches the AcousticConnect podspec floor (iOS >= 15.1) so the
|
|
337
|
+
// NCE never demands a higher minimum than the host app.
|
|
338
|
+
'15.1',
|
|
339
|
+
EXTENSION_TARGET_NAMES
|
|
340
|
+
)
|
|
341
|
+
const swiftVersion = getHostBuildSettingForConfig(
|
|
342
|
+
xcodeProject,
|
|
343
|
+
'SWIFT_VERSION',
|
|
344
|
+
configName,
|
|
345
|
+
'5.0',
|
|
346
|
+
EXTENSION_TARGET_NAMES
|
|
347
|
+
)
|
|
348
|
+
const marketingVersion = getHostBuildSettingForConfig(
|
|
349
|
+
xcodeProject,
|
|
350
|
+
'MARKETING_VERSION',
|
|
351
|
+
configName,
|
|
352
|
+
'1.0',
|
|
353
|
+
EXTENSION_TARGET_NAMES
|
|
354
|
+
)
|
|
355
|
+
const currentProjectVersion = getHostBuildSettingForConfig(
|
|
356
|
+
xcodeProject,
|
|
357
|
+
'CURRENT_PROJECT_VERSION',
|
|
358
|
+
configName,
|
|
359
|
+
'1',
|
|
360
|
+
EXTENSION_TARGET_NAMES
|
|
361
|
+
)
|
|
362
|
+
|
|
363
|
+
Object.assign(buildConfig.buildSettings, {
|
|
364
|
+
// Prevent Xcode from synthesising a second Info.plist that would
|
|
365
|
+
// shadow the NSExtension dictionary in ours.
|
|
366
|
+
GENERATE_INFOPLIST_FILE: 'NO',
|
|
367
|
+
INFOPLIST_FILE: `${NCE_TARGET_NAME}/Info.plist`,
|
|
368
|
+
// Extension must target the device SDK and restrict to
|
|
369
|
+
// extension-safe APIs (mirrors bare-workflow reference target).
|
|
370
|
+
SDKROOT: 'iphoneos',
|
|
371
|
+
APPLICATION_EXTENSION_API_ONLY: 'YES',
|
|
372
|
+
IPHONEOS_DEPLOYMENT_TARGET: deploymentTarget,
|
|
373
|
+
SWIFT_VERSION: swiftVersion,
|
|
374
|
+
MARKETING_VERSION: marketingVersion,
|
|
375
|
+
CURRENT_PROJECT_VERSION: currentProjectVersion,
|
|
376
|
+
CODE_SIGN_ENTITLEMENTS: `${NCE_TARGET_NAME}/${NCE_TARGET_NAME}.entitlements`,
|
|
377
|
+
PRODUCT_NAME: NCE_TARGET_NAME,
|
|
378
|
+
PRODUCT_BUNDLE_IDENTIFIER: `${hostBundleId}.ConnectNCE`,
|
|
379
|
+
TARGETED_DEVICE_FAMILY: '"1,2"',
|
|
380
|
+
CODE_SIGN_STYLE: 'Automatic',
|
|
381
|
+
})
|
|
382
|
+
}
|
|
383
|
+
}
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
return c
|
|
388
|
+
})
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
// ─── Podfile mod ─────────────────────────────────────────────────────────────
|
|
392
|
+
|
|
393
|
+
/**
|
|
394
|
+
* Injects the ConnectNCE Podfile target block via withDangerousMod.
|
|
395
|
+
* Guarded by a marker comment — re-runs are no-ops.
|
|
396
|
+
*
|
|
397
|
+
* Throws a clear, actionable error when the Podfile does not exist at
|
|
398
|
+
* mod-execution time, instead of silently returning — a silent skip would
|
|
399
|
+
* ship an NCE whose `import Connect` cannot resolve.
|
|
400
|
+
*/
|
|
401
|
+
export function withNCEPodfile(config: ExpoConfig): ExpoConfig {
|
|
402
|
+
return withDangerousMod(config, [
|
|
403
|
+
'ios',
|
|
404
|
+
async (c) => {
|
|
405
|
+
const podfilePath = path.join(c.modRequest.platformProjectRoot, 'Podfile')
|
|
406
|
+
if (!fs.existsSync(podfilePath)) {
|
|
407
|
+
throw new Error(
|
|
408
|
+
`[react-native-acoustic-connect-beta] ios/Podfile not found at ${podfilePath}.\n\n` +
|
|
409
|
+
`This mod runs after prebuild generates the ios/ directory. ` +
|
|
410
|
+
`If you are running this plugin outside of \`expo prebuild\`, ` +
|
|
411
|
+
`ensure the Podfile exists before the dangerous mod phase executes.\n` +
|
|
412
|
+
`Check prebuild ordering: withDangerousMod('ios') runs after ` +
|
|
413
|
+
`withXcodeProject, so the ios/ directory should already be present.`
|
|
414
|
+
)
|
|
415
|
+
}
|
|
416
|
+
const current = fs.readFileSync(podfilePath, 'utf8')
|
|
417
|
+
const updated = injectPodfileBlock(current)
|
|
418
|
+
if (updated !== current) {
|
|
419
|
+
fs.writeFileSync(podfilePath, updated, 'utf8')
|
|
420
|
+
}
|
|
421
|
+
return c
|
|
422
|
+
},
|
|
423
|
+
])
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
// ─── Composed mod ─────────────────────────────────────────────────────────────
|
|
427
|
+
|
|
428
|
+
/**
|
|
429
|
+
* Expo Config Plugin mod that provisions a Notification Content Extension
|
|
430
|
+
* (NCE) Xcode target named `ConnectNCE`.
|
|
431
|
+
*
|
|
432
|
+
* Applies three mutations — all idempotent:
|
|
433
|
+
* 1. Host app entitlements: merges App Group into
|
|
434
|
+
* `com.apple.security.application-groups` (reuses the NSE host-entitlements
|
|
435
|
+
* mod — the host shares one App Group across app + NSE + NCE; the merge is
|
|
436
|
+
* a no-op when the NSE mod already added it).
|
|
437
|
+
* 2. Xcode project: adds ConnectNCE target + files (skips if already present).
|
|
438
|
+
* 3. Podfile: injects `target 'ConnectNCE'` block (guarded by marker comment).
|
|
439
|
+
*
|
|
440
|
+
* `config._internal.projectRoot` is required (injected by Expo CLI during
|
|
441
|
+
* `expo prebuild`); an actionable error is thrown if absent rather than
|
|
442
|
+
* silently falling back to `process.cwd()`.
|
|
443
|
+
*
|
|
444
|
+
* NSE coupling (intentional, tracked for refactor): this mod deliberately
|
|
445
|
+
* reuses three pieces of withConnectNSE — `withNSEEntitlements` (the host App
|
|
446
|
+
* Group is shared across app + NSE + NCE), `buildConnectPodTargetBlock` (one
|
|
447
|
+
* source for the pod-resolution helper), and `getHostBuildSettingForConfig`
|
|
448
|
+
* (the per-config mirroring logic). The composition always runs withConnectNSE
|
|
449
|
+
* before withConnectNCE, so the shared host entitlement and the
|
|
450
|
+
* PBXTargetDependency/PBXContainerItemProxy sections already exist when this
|
|
451
|
+
* runs. Extracting a shared `withConnectExtension` base for NSE + NCE is
|
|
452
|
+
* tracked as a follow-up refactor (see PR description) rather than done here,
|
|
453
|
+
* to keep the already-landed NSE mod stable.
|
|
454
|
+
*/
|
|
455
|
+
export const withConnectNCE: ConfigPlugin<ConnectPluginProps> = (
|
|
456
|
+
config,
|
|
457
|
+
props = {}
|
|
458
|
+
) => {
|
|
459
|
+
const projectRoot = config._internal?.projectRoot
|
|
460
|
+
if (!projectRoot) {
|
|
461
|
+
throw new Error(
|
|
462
|
+
`[react-native-acoustic-connect-beta] config._internal.projectRoot is not set.\n\n` +
|
|
463
|
+
`This value is injected by Expo CLI during \`expo prebuild\`. ` +
|
|
464
|
+
`If you are calling this plugin outside of Expo CLI (e.g. in a test or ` +
|
|
465
|
+
`custom script), set config._internal = { projectRoot: '/abs/path/to/project' } ` +
|
|
466
|
+
`before invoking the plugin.`
|
|
467
|
+
)
|
|
468
|
+
}
|
|
469
|
+
|
|
470
|
+
const appGroupIdentifier = resolveAppGroupIdentifier(projectRoot, props)
|
|
471
|
+
|
|
472
|
+
const swiftTemplatePath = path.join(
|
|
473
|
+
__dirname,
|
|
474
|
+
'..',
|
|
475
|
+
'swift',
|
|
476
|
+
'NotificationViewController.swift'
|
|
477
|
+
)
|
|
478
|
+
const swiftContent = substituteSwiftTemplate(
|
|
479
|
+
swiftTemplatePath,
|
|
480
|
+
appGroupIdentifier
|
|
481
|
+
)
|
|
482
|
+
|
|
483
|
+
let result = withNSEEntitlements(config, appGroupIdentifier)
|
|
484
|
+
result = withNCEXcodeProject(result, appGroupIdentifier, swiftContent)
|
|
485
|
+
result = withNCEPodfile(result)
|
|
486
|
+
return result
|
|
487
|
+
}
|