linkAndroid function
void
linkAndroid(})
Configures android/build.gradle (or .kts) so the generated Kotlin bridge
files in lib/src/generated/kotlin/ are compiled as part of the Android build.
Without the kotlin.srcDirs entry, all .bridge.g.kt files are generated but
never compiled — causing "Unresolved reference: XxxJniBridge" errors at build time.
Implementation
void linkAndroid(String pluginName, List<String> moduleLibs, {String baseDir = '.', List<ModuleInfo>? moduleInfos}) {
File? buildGradle;
for (final candidate in [
File(p.join(baseDir, 'android', 'build.gradle')),
File(p.join(baseDir, 'android', 'build.gradle.kts')),
]) {
if (candidate.existsSync()) {
buildGradle = candidate;
break;
}
}
if (buildGradle == null) return;
var content = buildGradle.readAsStringSync();
bool modified = false;
final isKts = buildGradle.path.endsWith('.kts');
final srcDirsLine = isKts
? r' kotlin.srcDirs += setOf("${project.projectDir}/../lib/src/generated/kotlin")'
: r' kotlin.srcDirs += "${project.projectDir}/../lib/src/generated/kotlin"';
// 1. Ensure kotlin.srcDirs for generated Kotlin bridges.
// .bridge.g.kt files live in lib/src/generated/kotlin/ — Gradle must see
// that directory as a Kotlin source root or the JNI bridge classes won't compile.
// Note: add to kotlin.srcDirs ONLY, NOT java.srcDirs — in AGP 8.x, routing
// .kt through the Java compiler path causes "Unresolved reference: XxxJniBridge".
if (!content.contains('generated/kotlin')) {
final sourceSetsMatch = RegExp(r'\bsourceSets\s*\{').firstMatch(content);
if (sourceSetsMatch != null) {
// sourceSets block exists — look for main {} inside it.
final afterSourceSets = content.substring(sourceSetsMatch.end);
final mainInBlock = RegExp(r'\bmain\s*\{').firstMatch(afterSourceSets);
if (mainInBlock != null) {
final mainAbsStart = sourceSetsMatch.end + mainInBlock.start;
// Find the { of main {} and then its matching }
final openBrace = content.indexOf('{', mainAbsStart + mainInBlock.group(0)!.length - 1);
if (openBrace >= 0) {
final mainClose = _findBlockEnd(content, openBrace);
if (mainClose > 0) {
content = content.replaceRange(mainClose, mainClose, '\n$srcDirsLine\n ');
modified = true;
}
}
} else {
// sourceSets exists but no main {} — add main {} before sourceSets closing brace
final sourceSetsClose = _findBlockEnd(content, sourceSetsMatch.end - 1);
if (sourceSetsClose > 0) {
content = content.replaceRange(
sourceSetsClose,
sourceSetsClose,
' main {\n$srcDirsLine\n }\n ',
);
modified = true;
}
}
} else {
// No sourceSets block — inject one inside android {}
final androidMatch = RegExp(r'\bandroid\s*\{').firstMatch(content);
if (androidMatch != null) {
content = content.replaceRange(
androidMatch.end,
androidMatch.end,
'\n sourceSets {\n main {\n$srcDirsLine\n }\n }',
);
} else {
content += '\nandroid {\n sourceSets {\n main {\n$srcDirsLine\n }\n }\n}\n';
}
modified = true;
}
}
// 2. Ensure kotlinOptions { jvmTarget = "17" } for correct bytecode target.
if (!content.contains('kotlinOptions')) {
final androidMatch = RegExp(r'\bandroid\s*\{').firstMatch(content);
if (androidMatch != null) {
content = content.replaceRange(
androidMatch.end,
androidMatch.end,
'\n kotlinOptions { jvmTarget = "17" }',
);
modified = true;
}
}
// 3. Ensure kotlinx-coroutines (required for generated Kotlin suspend bridge functions).
if (!content.contains('kotlinx-coroutines')) {
final depsMatch = RegExp(r'\bdependencies\s*\{').firstMatch(content);
if (depsMatch != null) {
content = content.replaceRange(
depsMatch.end,
depsMatch.end,
'\n implementation "org.jetbrains.kotlinx:kotlinx-coroutines-android:1.7.3"\n implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:1.7.3"',
);
} else {
content +=
'\ndependencies {\n implementation "org.jetbrains.kotlinx:kotlinx-coroutines-android:1.7.3"\n implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:1.7.3"\n}\n';
}
modified = true;
}
if (modified) buildGradle.writeAsStringSync(content);
}