linkMacosSwiftPlugin function

void linkMacosSwiftPlugin(
  1. String pluginName,
  2. List<Map<String, String>> modules, {
  3. String baseDir = '.',
})

Wires non-cpp module registrations into the macOS Swift plugin file.

Mirrors linkSwiftPlugin but targets macos/ instead of ios/. Searches macos/ recursively for *Plugin.swift and injects Registry.register(...) calls for each non-cpp module that doesn't already have one.

Implementation

void linkMacosSwiftPlugin(
  String pluginName,
  List<Map<String, String>> modules, {
  String baseDir = '.',
}) {
  final macosDir = Directory(p.join(baseDir, 'macos'));
  if (!macosDir.existsSync()) return;
  final pluginFiles = macosDir
      .listSync(recursive: true, followLinks: false)
      .whereType<File>()
      .where((f) => !f.path.contains('.symlinks'))
      .where((f) => f.path.endsWith('Plugin.swift'))
      .toList();

  if (pluginFiles.isEmpty) {
    // Create default macOS plugin if missing
    final className = _toPascalCase(pluginName);
    final fileName = '${className}Plugin.swift';
    final targetPath = p.join(macosDir.path, 'Classes', fileName);
    Directory(p.dirname(targetPath)).createSync(recursive: true);
    final stub = st.macosPluginSwiftStub(className);
    File(targetPath).writeAsStringSync(stub);
    pluginFiles.add(File(targetPath));
  }

  final pluginFile = pluginFiles.first;
  var content = pluginFile.readAsStringSync();
  bool modified = false;
  for (final m in modules) {
    final name = m['module']!;
    final lib = (m['lib'] ?? name.toLowerCase()).replaceAll('-', '_');
    final reg = '${name}Registry';
    // Standard implementation naming: BenchmarkImpl or BenchmarkModuleImpl
    final impl = name.endsWith('Module') ? '${name}Impl' : '${name}ModuleImpl';

    // ── 1. No module import needed — bridge .swift files are compiled into
    //        the same CocoaPods pod target. Remove any stale module import.
    final staleImportPattern = RegExp(
      r'#if canImport\(nitro_' +
          RegExp.escape(lib) +
          r'_module\)\s*\nimport nitro_' +
          RegExp.escape(lib) +
          r'_module\s*\n#endif\s*\n?',
    );
    if (staleImportPattern.hasMatch(content)) {
      content = content.replaceAll(staleImportPattern, '');
      modified = true;
    }
    final bareImport = RegExp(
      r'import nitro_' + RegExp.escape(lib) + r'_module[ \t]*\r?\n?',
    );
    if (bareImport.hasMatch(content)) {
      content = content.replaceAll(bareImport, '');
      modified = true;
    }

    // ── 2. Ensure register() call is present ────────────────────────────────
    if (!content.contains('$reg.register')) {
      content = content.replaceFirst(
        'public static func register(with registrar: FlutterPluginRegistrar) {',
        'public static func register(with registrar: FlutterPluginRegistrar) {\n    $reg.register($impl())',
      );
      modified = true;
    }
  }
  if (modified) pluginFile.writeAsStringSync(content);
}