// Copyright 2014 The Flutter Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // @dart = 2.8 import 'package:meta/meta.dart'; import 'package:xml/xml.dart'; import 'package:yaml/yaml.dart'; import '../src/convert.dart'; import 'android/gradle_utils.dart' as gradle; import 'artifacts.dart'; import 'base/common.dart'; import 'base/file_system.dart'; import 'base/logger.dart'; import 'build_info.dart'; import 'bundle.dart' as bundle; import 'features.dart'; import 'flutter_manifest.dart'; import 'globals.dart' as globals; import 'ios/plist_parser.dart'; import 'ios/xcodeproj.dart' as xcode; import 'ios/xcodeproj.dart'; import 'platform_plugins.dart'; import 'plugins.dart'; import 'template.dart'; class FlutterProjectFactory { FlutterProjectFactory({ @required Logger logger, @required FileSystem fileSystem, }) : _logger = logger, _fileSystem = fileSystem; final Logger _logger; final FileSystem _fileSystem; @visibleForTesting final Map<String, FlutterProject> projects = <String, FlutterProject>{}; /// Returns a [FlutterProject] view of the given directory or a ToolExit error, /// if `pubspec.yaml` or `example/pubspec.yaml` is invalid. FlutterProject fromDirectory(Directory directory) { assert(directory != null); return projects.putIfAbsent(directory.path, () { final FlutterManifest manifest = FlutterProject._readManifest( directory.childFile(bundle.defaultManifestPath).path, logger: _logger, fileSystem: _fileSystem, ); final FlutterManifest exampleManifest = FlutterProject._readManifest( FlutterProject._exampleDirectory(directory) .childFile(bundle.defaultManifestPath) .path, logger: _logger, fileSystem: _fileSystem, ); return FlutterProject(directory, manifest, exampleManifest); }); } } /// Represents the contents of a Flutter project at the specified [directory]. /// /// [FlutterManifest] information is read from `pubspec.yaml` and /// `example/pubspec.yaml` files on construction of a [FlutterProject] instance. /// The constructed instance carries an immutable snapshot representation of the /// presence and content of those files. Accordingly, [FlutterProject] instances /// should be discarded upon changes to the `pubspec.yaml` files, but can be /// used across changes to other files, as no other file-level information is /// cached. class FlutterProject { @visibleForTesting FlutterProject(this.directory, this.manifest, this._exampleManifest) : assert(directory != null), assert(manifest != null), assert(_exampleManifest != null); /// Returns a [FlutterProject] view of the given directory or a ToolExit error, /// if `pubspec.yaml` or `example/pubspec.yaml` is invalid. static FlutterProject fromDirectory(Directory directory) => globals.projectFactory.fromDirectory(directory); /// Returns a [FlutterProject] view of the current directory or a ToolExit error, /// if `pubspec.yaml` or `example/pubspec.yaml` is invalid. static FlutterProject current() => globals.projectFactory.fromDirectory(globals.fs.currentDirectory); /// Create a [FlutterProject] and bypass the project caching. @visibleForTesting static FlutterProject fromDirectoryTest(Directory directory, [Logger logger]) { final FileSystem fileSystem = directory.fileSystem; logger ??= BufferLogger.test(); final FlutterManifest manifest = FlutterProject._readManifest( directory.childFile(bundle.defaultManifestPath).path, logger: logger, fileSystem: fileSystem, ); final FlutterManifest exampleManifest = FlutterProject._readManifest( FlutterProject._exampleDirectory(directory) .childFile(bundle.defaultManifestPath) .path, logger: logger, fileSystem: fileSystem, ); return FlutterProject(directory, manifest, exampleManifest); } /// The location of this project. final Directory directory; /// The manifest of this project. final FlutterManifest manifest; /// The manifest of the example sub-project of this project. final FlutterManifest _exampleManifest; /// The set of organization names found in this project as /// part of iOS product bundle identifier, Android application ID, or /// Gradle group ID. Future<Set<String>> get organizationNames async { final List<String> candidates = <String>[ // Don't require iOS build info, this method is only // used during create as best-effort, use the // default target bundle identifier. if (ios.existsSync()) await ios.productBundleIdentifier(null), if (android.existsSync()) ...<String>[ android.applicationId, android.group, ], if (example.android.existsSync()) example.android.applicationId, if (example.ios.existsSync()) await example.ios.productBundleIdentifier(null), ]; return Set<String>.of(candidates .map<String>(_organizationNameFromPackageName) .where((String name) => name != null)); } String _organizationNameFromPackageName(String packageName) { if (packageName != null && 0 <= packageName.lastIndexOf('.')) { return packageName.substring(0, packageName.lastIndexOf('.')); } return null; } /// The iOS sub project of this project. IosProject _ios; IosProject get ios => _ios ??= IosProject.fromFlutter(this); /// The Android sub project of this project. AndroidProject _android; AndroidProject get android => _android ??= AndroidProject._(this); /// The web sub project of this project. WebProject _web; WebProject get web => _web ??= WebProject._(this); /// The MacOS sub project of this project. MacOSProject _macos; MacOSProject get macos => _macos ??= MacOSProject._(this); /// The Linux sub project of this project. LinuxProject _linux; LinuxProject get linux => _linux ??= LinuxProject._(this); /// The Windows sub project of this project. WindowsProject _windows; WindowsProject get windows => _windows ??= WindowsProject._(this); /// The Windows UWP sub project of this project. WindowsUwpProject _windowUwp; WindowsUwpProject get windowsUwp => _windowUwp ??= WindowsUwpProject._(this); /// The Fuchsia sub project of this project. FuchsiaProject _fuchsia; FuchsiaProject get fuchsia => _fuchsia ??= FuchsiaProject._(this); /// The `pubspec.yaml` file of this project. File get pubspecFile => directory.childFile('pubspec.yaml'); /// The `.packages` file of this project. File get packagesFile => directory.childFile('.packages'); /// The `package_config.json` file of the project. /// /// This is the replacement for .packages which contains language /// version information. File get packageConfigFile => directory.childDirectory('.dart_tool').childFile('package_config.json'); /// The `.metadata` file of this project. File get metadataFile => directory.childFile('.metadata'); /// The `.flutter-plugins` file of this project. File get flutterPluginsFile => directory.childFile('.flutter-plugins'); /// The `.flutter-plugins-dependencies` file of this project, /// which contains the dependencies each plugin depends on. File get flutterPluginsDependenciesFile => directory.childFile('.flutter-plugins-dependencies'); /// The `.dart-tool` directory of this project. Directory get dartTool => directory.childDirectory('.dart_tool'); /// The directory containing the generated code for this project. Directory get generated => directory .absolute .childDirectory('.dart_tool') .childDirectory('build') .childDirectory('generated') .childDirectory(manifest.appName); /// The example sub-project of this project. FlutterProject get example => FlutterProject( _exampleDirectory(directory), _exampleManifest, FlutterManifest.empty(logger: globals.logger), ); /// True if this project is a Flutter module project. bool get isModule => manifest.isModule; /// True if this project is a Flutter plugin project. bool get isPlugin => manifest.isPlugin; /// True if the Flutter project is using the AndroidX support library. bool get usesAndroidX => manifest.usesAndroidX; /// True if this project has an example application. bool get hasExampleApp => _exampleDirectory(directory).existsSync(); /// The directory that will contain the example if an example exists. static Directory _exampleDirectory(Directory directory) => directory.childDirectory('example'); /// Reads and validates the `pubspec.yaml` file at [path], asynchronously /// returning a [FlutterManifest] representation of the contents. /// /// Completes with an empty [FlutterManifest], if the file does not exist. /// Completes with a ToolExit on validation error. static FlutterManifest _readManifest(String path, { @required Logger logger, @required FileSystem fileSystem, }) { FlutterManifest manifest; try { manifest = FlutterManifest.createFromPath( path, logger: logger, fileSystem: fileSystem, ); } on YamlException catch (e) { logger.printStatus('Error detected in pubspec.yaml:', emphasis: true); logger.printError('$e'); } on FormatException catch (e) { logger.printError('Error detected while parsing pubspec.yaml:', emphasis: true); logger.printError('$e'); } on FileSystemException catch (e) { logger.printError('Error detected while reading pubspec.yaml:', emphasis: true); logger.printError('$e'); } if (manifest == null) { throwToolExit('Please correct the pubspec.yaml file at $path'); } return manifest; } /// Reapplies template files and regenerates project files and plugin /// registrants for app and module projects only. /// /// Will not create project platform directories if they do not already exist. Future<void> regeneratePlatformSpecificTooling() async { return ensureReadyForPlatformSpecificTooling( androidPlatform: android.existsSync(), iosPlatform: ios.existsSync(), // TODO(stuartmorgan): Revisit the conditions here once the plans for handling // desktop in existing projects are in place. linuxPlatform: featureFlags.isLinuxEnabled && linux.existsSync(), macOSPlatform: featureFlags.isMacOSEnabled && macos.existsSync(), windowsPlatform: featureFlags.isWindowsEnabled && windows.existsSync(), webPlatform: featureFlags.isWebEnabled && web.existsSync(), ); } /// Applies template files and generates project files and plugin /// registrants for app and module projects only for the specified platforms. Future<void> ensureReadyForPlatformSpecificTooling({ bool androidPlatform = false, bool iosPlatform = false, bool linuxPlatform = false, bool macOSPlatform = false, bool windowsPlatform = false, bool webPlatform = false, bool windowsUwpPlatform = false, }) async { if (!directory.existsSync() || hasExampleApp || isPlugin) { return; } await refreshPluginsList(this, iosPlatform: iosPlatform, macOSPlatform: macOSPlatform); if (androidPlatform) { await android.ensureReadyForPlatformSpecificTooling(); } if (iosPlatform) { await ios.ensureReadyForPlatformSpecificTooling(); } if (linuxPlatform) { await linux.ensureReadyForPlatformSpecificTooling(); } if (macOSPlatform) { await macos.ensureReadyForPlatformSpecificTooling(); } if (windowsPlatform) { await windows.ensureReadyForPlatformSpecificTooling(); } if (webPlatform) { await web.ensureReadyForPlatformSpecificTooling(); } if (windowsUwpPlatform) { await windowsUwp.ensureReadyForPlatformSpecificTooling(); } await injectPlugins( this, androidPlatform: androidPlatform, iosPlatform: iosPlatform, linuxPlatform: linuxPlatform, macOSPlatform: macOSPlatform, windowsPlatform: windowsPlatform, webPlatform: webPlatform, ); } /// Returns a json encoded string containing the [appName], [version], and [buildNumber] that is used to generate version.json String getVersionInfo() { final Map<String, String> versionFileJson = <String, String>{ 'app_name': manifest.appName, 'version': manifest.buildName, 'build_number': manifest.buildNumber }; return jsonEncode(versionFileJson); } } /// Base class for projects per platform. abstract class FlutterProjectPlatform { /// Plugin's platform config key, e.g., "macos", "ios". String get pluginConfigKey; /// Whether the platform exists in the project. bool existsSync(); } /// Represents an Xcode-based sub-project. /// /// This defines interfaces common to iOS and macOS projects. abstract class XcodeBasedProject { /// The parent of this project. FlutterProject get parent; /// Whether the subproject (either iOS or macOS) exists in the Flutter project. bool existsSync(); /// The Xcode project (.xcodeproj directory) of the host app. Directory get xcodeProject; /// The 'project.pbxproj' file of [xcodeProject]. File get xcodeProjectInfoFile; /// The Xcode workspace (.xcworkspace directory) of the host app. Directory get xcodeWorkspace; /// Contains definitions for FLUTTER_ROOT, LOCAL_ENGINE, and more flags for /// the Xcode build. File get generatedXcodePropertiesFile; /// The Flutter-managed Xcode config file for [mode]. File xcodeConfigFor(String mode); /// The script that exports environment variables needed for Flutter tools. /// Can be run first in a Xcode Script build phase to make FLUTTER_ROOT, /// LOCAL_ENGINE, and other Flutter variables available to any flutter /// tooling (`flutter build`, etc) to convert into flags. File get generatedEnvironmentVariableExportScript; /// The CocoaPods 'Podfile'. File get podfile; /// The CocoaPods 'Podfile.lock'. File get podfileLock; /// The CocoaPods 'Manifest.lock'. File get podManifestLock; } /// Represents a CMake-based sub-project. /// /// This defines interfaces common to Windows and Linux projects. abstract class CmakeBasedProject { /// The parent of this project. FlutterProject get parent; /// Whether the subproject (either Windows or Linux) exists in the Flutter project. bool existsSync(); /// The native project CMake specification. File get cmakeFile; /// Contains definitions for the Flutter library and the tool. File get managedCmakeFile; /// Contains definitions for FLUTTER_ROOT, LOCAL_ENGINE, and more flags for /// the build. File get generatedCmakeConfigFile; /// Included CMake with rules and variables for plugin builds. File get generatedPluginCmakeFile; /// The directory to write plugin symlinks. Directory get pluginSymlinkDirectory; } /// Represents the iOS sub-project of a Flutter project. /// /// Instances will reflect the contents of the `ios/` sub-folder of /// Flutter applications and the `.ios/` sub-folder of Flutter module projects. class IosProject extends FlutterProjectPlatform implements XcodeBasedProject { IosProject.fromFlutter(this.parent); @override final FlutterProject parent; @override String get pluginConfigKey => IOSPlugin.kConfigKey; static final RegExp _productBundleIdPattern = RegExp(r'''^\s*PRODUCT_BUNDLE_IDENTIFIER\s*=\s*(["']?)(.*?)\1;\s*$'''); static const String _productBundleIdVariable = r'$(PRODUCT_BUNDLE_IDENTIFIER)'; static const String _hostAppProjectName = 'Runner'; Directory get ephemeralModuleDirectory => parent.directory.childDirectory('.ios'); Directory get _editableDirectory => parent.directory.childDirectory('ios'); /// This parent folder of `Runner.xcodeproj`. Directory get hostAppRoot { if (!isModule || _editableDirectory.existsSync()) { return _editableDirectory; } return ephemeralModuleDirectory; } /// The root directory of the iOS wrapping of Flutter and plugins. This is the /// parent of the `Flutter/` folder into which Flutter artifacts are written /// during build. /// /// This is the same as [hostAppRoot] except when the project is /// a Flutter module with an editable host app. Directory get _flutterLibRoot => isModule ? ephemeralModuleDirectory : _editableDirectory; /// True, if the parent Flutter project is a module project. bool get isModule => parent.isModule; /// Whether the flutter application has an iOS project. bool get exists => hostAppRoot.existsSync(); /// Put generated files here. Directory get ephemeralDirectory => _flutterLibRoot.childDirectory('Flutter').childDirectory('ephemeral'); @override File xcodeConfigFor(String mode) => _flutterLibRoot.childDirectory('Flutter').childFile('$mode.xcconfig'); @override File get generatedEnvironmentVariableExportScript => _flutterLibRoot.childDirectory('Flutter').childFile('flutter_export_environment.sh'); @override File get podfile => hostAppRoot.childFile('Podfile'); @override File get podfileLock => hostAppRoot.childFile('Podfile.lock'); @override File get podManifestLock => hostAppRoot.childDirectory('Pods').childFile('Manifest.lock'); /// The default 'Info.plist' file of the host app. The developer can change this location in Xcode. File get defaultHostInfoPlist => hostAppRoot.childDirectory(_hostAppProjectName).childFile('Info.plist'); Directory get symlinks => _flutterLibRoot.childDirectory('.symlinks'); @override Directory get xcodeProject => hostAppRoot.childDirectory('$_hostAppProjectName.xcodeproj'); @override File get xcodeProjectInfoFile => xcodeProject.childFile('project.pbxproj'); File get xcodeProjectWorkspaceData => xcodeProject .childDirectory('project.xcworkspace') .childFile('contents.xcworkspacedata'); @override Directory get xcodeWorkspace => hostAppRoot.childDirectory('$_hostAppProjectName.xcworkspace'); /// Xcode workspace shared data directory for the host app. Directory get xcodeWorkspaceSharedData => xcodeWorkspace.childDirectory('xcshareddata'); /// Xcode workspace shared workspace settings file for the host app. File get xcodeWorkspaceSharedSettings => xcodeWorkspaceSharedData.childFile('WorkspaceSettings.xcsettings'); @override bool existsSync() { return parent.isModule || _editableDirectory.existsSync(); } /// The product bundle identifier of the host app, or null if not set or if /// iOS tooling needed to read it is not installed. Future<String> productBundleIdentifier(BuildInfo buildInfo) async { if (!existsSync()) { return null; } return _productBundleIdentifier ??= await _parseProductBundleIdentifier(buildInfo); } String _productBundleIdentifier; Future<String> _parseProductBundleIdentifier(BuildInfo buildInfo) async { String fromPlist; final File defaultInfoPlist = defaultHostInfoPlist; // Users can change the location of the Info.plist. // Try parsing the default, first. if (defaultInfoPlist.existsSync()) { try { fromPlist = globals.plistParser.getValueFromFile( defaultHostInfoPlist.path, PlistParser.kCFBundleIdentifierKey, ); } on FileNotFoundException { // iOS tooling not found; likely not running OSX; let [fromPlist] be null } if (fromPlist != null && !fromPlist.contains(r'$')) { // Info.plist has no build variables in product bundle ID. return fromPlist; } } final Map<String, String> allBuildSettings = await buildSettingsForBuildInfo(buildInfo); if (allBuildSettings != null) { if (fromPlist != null) { // Perform variable substitution using build settings. return xcode.substituteXcodeVariables(fromPlist, allBuildSettings); } return allBuildSettings['PRODUCT_BUNDLE_IDENTIFIER']; } // On non-macOS platforms, parse the first PRODUCT_BUNDLE_IDENTIFIER from // the project file. This can return the wrong bundle identifier if additional // bundles have been added to the project and are found first, like frameworks // or companion watchOS projects. However, on non-macOS platforms this is // only used for display purposes and to regenerate organization names, so // best-effort is probably fine. final String fromPbxproj = _firstMatchInFile(xcodeProjectInfoFile, _productBundleIdPattern)?.group(2); if (fromPbxproj != null && (fromPlist == null || fromPlist == _productBundleIdVariable)) { return fromPbxproj; } return null; } /// The bundle name of the host app, `My App.app`. Future<String> hostAppBundleName(BuildInfo buildInfo) async { if (!existsSync()) { return null; } return _hostAppBundleName ??= await _parseHostAppBundleName(buildInfo); } String _hostAppBundleName; Future<String> _parseHostAppBundleName(BuildInfo buildInfo) async { // The product name and bundle name are derived from the display name, which the user // is instructed to change in Xcode as part of deploying to the App Store. // https://flutter.dev/docs/deployment/ios#review-xcode-project-settings // The only source of truth for the name is Xcode's interpretation of the build settings. String productName; if (globals.xcodeProjectInterpreter.isInstalled) { final Map<String, String> xcodeBuildSettings = await buildSettingsForBuildInfo(buildInfo); if (xcodeBuildSettings != null) { productName = xcodeBuildSettings['FULL_PRODUCT_NAME']; } } if (productName == null) { globals.printTrace('FULL_PRODUCT_NAME not present, defaulting to $_hostAppProjectName'); } return productName ?? '$_hostAppProjectName.app'; } /// The build settings for the host app of this project, as a detached map. /// /// Returns null, if iOS tooling is unavailable. Future<Map<String, String>> buildSettingsForBuildInfo(BuildInfo buildInfo) async { if (!existsSync()) { return null; } _buildSettingsByScheme ??= <String, Map<String, String>>{}; final XcodeProjectInfo info = await projectInfo(); if (info == null) { return null; } final String scheme = info.schemeFor(buildInfo); if (scheme == null) { info.reportFlavorNotFoundAndExit(); } return _buildSettingsByScheme[scheme] ??= await _xcodeProjectBuildSettings(scheme); } Map<String, Map<String, String>> _buildSettingsByScheme; Future<XcodeProjectInfo> projectInfo() async { if (!xcodeProject.existsSync() || !globals.xcodeProjectInterpreter.isInstalled) { return null; } return _projectInfo ??= await globals.xcodeProjectInterpreter.getInfo(hostAppRoot.path); } XcodeProjectInfo _projectInfo; Future<Map<String, String>> _xcodeProjectBuildSettings(String scheme) async { if (!globals.xcodeProjectInterpreter.isInstalled) { return null; } final Map<String, String> buildSettings = await globals.xcodeProjectInterpreter.getBuildSettings( xcodeProject.path, scheme: scheme, ); if (buildSettings != null && buildSettings.isNotEmpty) { // No timeouts, flakes, or errors. return buildSettings; } return null; } Future<void> ensureReadyForPlatformSpecificTooling() async { await _regenerateFromTemplateIfNeeded(); if (!_flutterLibRoot.existsSync()) { return; } await _updateGeneratedXcodeConfigIfNeeded(); } /// Check if one the [targets] of the project is a watchOS companion app target. Future<bool> containsWatchCompanion(List<String> targets, BuildInfo buildInfo) async { final String bundleIdentifier = await productBundleIdentifier(buildInfo); // A bundle identifier is required for a companion app. if (bundleIdentifier == null) { return false; } for (final String target in targets) { // Create Info.plist file of the target. final File infoFile = hostAppRoot.childDirectory(target).childFile('Info.plist'); // The Info.plist file of a target contains the key WKCompanionAppBundleIdentifier, // if it is a watchOS companion app. if (infoFile.existsSync() && globals.plistParser.getValueFromFile(infoFile.path, 'WKCompanionAppBundleIdentifier') == bundleIdentifier) { return true; } } return false; } Future<void> _updateGeneratedXcodeConfigIfNeeded() async { if (globals.cache.isOlderThanToolsStamp(generatedXcodePropertiesFile)) { await xcode.updateGeneratedXcodeProperties( project: parent, buildInfo: BuildInfo.debug, targetOverride: bundle.defaultMainPath, ); } } Future<void> _regenerateFromTemplateIfNeeded() async { if (!isModule) { return; } final bool pubspecChanged = globals.fsUtils.isOlderThanReference( entity: ephemeralModuleDirectory, referenceFile: parent.pubspecFile, ); final bool toolingChanged = globals.cache.isOlderThanToolsStamp(ephemeralModuleDirectory); if (!pubspecChanged && !toolingChanged) { return; } _deleteIfExistsSync(ephemeralModuleDirectory); await _overwriteFromTemplate( globals.fs.path.join('module', 'ios', 'library'), ephemeralModuleDirectory, ); // Add ephemeral host app, if a editable host app does not already exist. if (!_editableDirectory.existsSync()) { await _overwriteFromTemplate( globals.fs.path.join('module', 'ios', 'host_app_ephemeral'), ephemeralModuleDirectory, ); if (hasPlugins(parent)) { await _overwriteFromTemplate( globals.fs.path.join('module', 'ios', 'host_app_ephemeral_cocoapods'), ephemeralModuleDirectory, ); } // Use release mode so host project can link on bitcode variant. _copyEngineArtifactToProject(BuildMode.release, EnvironmentType.physical); } } void _copyEngineArtifactToProject(BuildMode mode, EnvironmentType environmentType) { // Copy framework from engine cache. The actual build mode // doesn't actually matter as it will be overwritten by xcode_backend.sh. // However, cocoapods will run before that script and requires something // to be in this location. final Directory framework = globals.fs.directory( globals.artifacts.getArtifactPath( Artifact.flutterXcframework, platform: TargetPlatform.ios, mode: mode, environmentType: environmentType, ) ); if (framework.existsSync()) { copyDirectory( framework, engineCopyDirectory.childDirectory('Flutter.xcframework'), ); } } @override File get generatedXcodePropertiesFile => _flutterLibRoot .childDirectory('Flutter') .childFile('Generated.xcconfig'); /// No longer compiled to this location. /// /// Used only for "flutter clean" to remove old references. Directory get deprecatedCompiledDartFramework => _flutterLibRoot .childDirectory('Flutter') .childDirectory('App.framework'); /// No longer copied to this location. /// /// Used only for "flutter clean" to remove old references. Directory get deprecatedProjectFlutterFramework => _flutterLibRoot .childDirectory('Flutter') .childDirectory('Flutter.framework'); /// Used only for "flutter clean" to remove old references. File get flutterPodspec => _flutterLibRoot .childDirectory('Flutter') .childFile('Flutter.podspec'); Directory get pluginRegistrantHost { return isModule ? _flutterLibRoot .childDirectory('Flutter') .childDirectory('FlutterPluginRegistrant') : hostAppRoot.childDirectory(_hostAppProjectName); } File get pluginRegistrantHeader { final Directory registryDirectory = isModule ? pluginRegistrantHost.childDirectory('Classes') : pluginRegistrantHost; return registryDirectory.childFile('GeneratedPluginRegistrant.h'); } File get pluginRegistrantImplementation { final Directory registryDirectory = isModule ? pluginRegistrantHost.childDirectory('Classes') : pluginRegistrantHost; return registryDirectory.childFile('GeneratedPluginRegistrant.m'); } Directory get engineCopyDirectory { return isModule ? ephemeralModuleDirectory.childDirectory('Flutter').childDirectory('engine') : hostAppRoot.childDirectory('Flutter'); } Future<void> _overwriteFromTemplate(String path, Directory target) async { final Template template = await Template.fromName( path, fileSystem: globals.fs, templateManifest: null, logger: globals.logger, templateRenderer: globals.templateRenderer, ); template.render( target, <String, dynamic>{ 'ios': true, 'projectName': parent.manifest.appName, 'iosIdentifier': parent.manifest.iosBundleIdentifier, }, printStatusWhenWriting: false, overwriteExisting: true, ); } } /// Represents the Android sub-project of a Flutter project. /// /// Instances will reflect the contents of the `android/` sub-folder of /// Flutter applications and the `.android/` sub-folder of Flutter module projects. class AndroidProject extends FlutterProjectPlatform { AndroidProject._(this.parent); /// The parent of this project. final FlutterProject parent; @override String get pluginConfigKey => AndroidPlugin.kConfigKey; static final RegExp _applicationIdPattern = RegExp('^\\s*applicationId\\s+[\'"](.*)[\'"]\\s*\$'); static final RegExp _kotlinPluginPattern = RegExp('^\\s*apply plugin\\:\\s+[\'"]kotlin-android[\'"]\\s*\$'); static final RegExp _groupPattern = RegExp('^\\s*group\\s+[\'"](.*)[\'"]\\s*\$'); /// The Gradle root directory of the Android host app. This is the directory /// containing the `app/` subdirectory and the `settings.gradle` file that /// includes it in the overall Gradle project. Directory get hostAppGradleRoot { if (!isModule || _editableHostAppDirectory.existsSync()) { return _editableHostAppDirectory; } return ephemeralDirectory; } /// The Gradle root directory of the Android wrapping of Flutter and plugins. /// This is the same as [hostAppGradleRoot] except when the project is /// a Flutter module with an editable host app. Directory get _flutterLibGradleRoot => isModule ? ephemeralDirectory : _editableHostAppDirectory; Directory get ephemeralDirectory => parent.directory.childDirectory('.android'); Directory get _editableHostAppDirectory => parent.directory.childDirectory('android'); /// True if the parent Flutter project is a module. bool get isModule => parent.isModule; /// True if the Flutter project is using the AndroidX support library. bool get usesAndroidX => parent.usesAndroidX; /// Returns true if the current version of the Gradle plugin is supported. bool get isSupportedVersion => _isSupportedVersion ??= _computeSupportedVersion(); bool _isSupportedVersion; bool _computeSupportedVersion() { final FileSystem fileSystem = hostAppGradleRoot.fileSystem; final File plugin = hostAppGradleRoot.childFile( fileSystem.path.join('buildSrc', 'src', 'main', 'groovy', 'FlutterPlugin.groovy')); if (plugin.existsSync()) { return false; } final File appGradle = hostAppGradleRoot.childFile( fileSystem.path.join('app', 'build.gradle')); if (!appGradle.existsSync()) { return false; } for (final String line in appGradle.readAsLinesSync()) { if (line.contains(RegExp(r'apply from: .*/flutter.gradle')) || line.contains("def flutterPluginVersion = 'managed'")) { return true; } } return false; } /// True, if the app project is using Kotlin. bool get isKotlin { final File gradleFile = hostAppGradleRoot.childDirectory('app').childFile('build.gradle'); return _firstMatchInFile(gradleFile, _kotlinPluginPattern) != null; } File get appManifestFile { return isUsingGradle ? globals.fs.file(globals.fs.path.join(hostAppGradleRoot.path, 'app', 'src', 'main', 'AndroidManifest.xml')) : hostAppGradleRoot.childFile('AndroidManifest.xml'); } File get gradleAppOutV1File => gradleAppOutV1Directory.childFile('app-debug.apk'); Directory get gradleAppOutV1Directory { return globals.fs.directory(globals.fs.path.join(hostAppGradleRoot.path, 'app', 'build', 'outputs', 'apk')); } /// Whether the current flutter project has an Android sub-project. @override bool existsSync() { return parent.isModule || _editableHostAppDirectory.existsSync(); } bool get isUsingGradle { return hostAppGradleRoot.childFile('build.gradle').existsSync(); } String get applicationId { final File gradleFile = hostAppGradleRoot.childDirectory('app').childFile('build.gradle'); return _firstMatchInFile(gradleFile, _applicationIdPattern)?.group(1); } String get group { final File gradleFile = hostAppGradleRoot.childFile('build.gradle'); return _firstMatchInFile(gradleFile, _groupPattern)?.group(1); } /// The build directory where the Android artifacts are placed. Directory get buildDirectory { return parent.directory.childDirectory('build'); } Future<void> ensureReadyForPlatformSpecificTooling() async { if (getEmbeddingVersion() == AndroidEmbeddingVersion.v1) { globals.printStatus( """ ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ Warning ────────────────────────────────────────────────────────────────────────────── Your Flutter application is created using an older version of the Android embedding. It's being deprecated in favor of Android embedding v2. Follow the steps at https://flutter.dev/go/android-project-migration to migrate your project. ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ """ ); } if (isModule && _shouldRegenerateFromTemplate()) { await _regenerateLibrary(); // Add ephemeral host app, if an editable host app does not already exist. if (!_editableHostAppDirectory.existsSync()) { await _overwriteFromTemplate(globals.fs.path.join('module', 'android', 'host_app_common'), ephemeralDirectory); await _overwriteFromTemplate(globals.fs.path.join('module', 'android', 'host_app_ephemeral'), ephemeralDirectory); } } if (!hostAppGradleRoot.existsSync()) { return; } gradle.updateLocalProperties(project: parent, requireAndroidSdk: false); } bool _shouldRegenerateFromTemplate() { return globals.fsUtils.isOlderThanReference( entity: ephemeralDirectory, referenceFile: parent.pubspecFile, ) || globals.cache.isOlderThanToolsStamp(ephemeralDirectory); } File get localPropertiesFile => _flutterLibGradleRoot.childFile('local.properties'); Directory get pluginRegistrantHost => _flutterLibGradleRoot.childDirectory(isModule ? 'Flutter' : 'app'); Future<void> _regenerateLibrary() async { _deleteIfExistsSync(ephemeralDirectory); await _overwriteFromTemplate(globals.fs.path.join( 'module', 'android', 'library_new_embedding', ), ephemeralDirectory); await _overwriteFromTemplate(globals.fs.path.join('module', 'android', 'gradle'), ephemeralDirectory); globals.gradleUtils.injectGradleWrapperIfNeeded(ephemeralDirectory); } Future<void> _overwriteFromTemplate(String path, Directory target) async { final Template template = await Template.fromName( path, fileSystem: globals.fs, templateManifest: null, logger: globals.logger, templateRenderer: globals.templateRenderer, ); template.render( target, <String, dynamic>{ 'android': true, 'projectName': parent.manifest.appName, 'androidIdentifier': parent.manifest.androidPackage, 'androidX': usesAndroidX, }, printStatusWhenWriting: false, overwriteExisting: true, ); } AndroidEmbeddingVersion getEmbeddingVersion() { if (isModule) { // A module type's Android project is used in add-to-app scenarios and // only supports the V2 embedding. return AndroidEmbeddingVersion.v2; } if (appManifestFile == null || !appManifestFile.existsSync()) { return AndroidEmbeddingVersion.v1; } XmlDocument document; try { document = XmlDocument.parse(appManifestFile.readAsStringSync()); } on XmlParserException { throwToolExit('Error parsing $appManifestFile ' 'Please ensure that the android manifest is a valid XML document and try again.'); } on FileSystemException { throwToolExit('Error reading $appManifestFile even though it exists. ' 'Please ensure that you have read permission to this file and try again.'); } for (final XmlElement metaData in document.findAllElements('meta-data')) { final String name = metaData.getAttribute('android:name'); if (name == 'flutterEmbedding') { final String embeddingVersionString = metaData.getAttribute('android:value'); if (embeddingVersionString == '1') { return AndroidEmbeddingVersion.v1; } if (embeddingVersionString == '2') { return AndroidEmbeddingVersion.v2; } } } return AndroidEmbeddingVersion.v1; } } /// Iteration of the embedding Java API in the engine used by the Android project. enum AndroidEmbeddingVersion { /// V1 APIs based on io.flutter.app.FlutterActivity. v1, /// V2 APIs based on io.flutter.embedding.android.FlutterActivity. v2, } /// Represents the web sub-project of a Flutter project. class WebProject extends FlutterProjectPlatform { WebProject._(this.parent); final FlutterProject parent; @override String get pluginConfigKey => WebPlugin.kConfigKey; /// Whether this flutter project has a web sub-project. @override bool existsSync() { return parent.directory.childDirectory('web').existsSync() && indexFile.existsSync(); } /// The 'lib' directory for the application. Directory get libDirectory => parent.directory.childDirectory('lib'); /// The directory containing additional files for the application. Directory get directory => parent.directory.childDirectory('web'); /// The html file used to host the flutter web application. File get indexFile => parent.directory .childDirectory('web') .childFile('index.html'); Future<void> ensureReadyForPlatformSpecificTooling() async {} } /// Deletes [directory] with all content. void _deleteIfExistsSync(Directory directory) { if (directory.existsSync()) { directory.deleteSync(recursive: true); } } /// Returns the first line-based match for [regExp] in [file]. /// /// Assumes UTF8 encoding. Match _firstMatchInFile(File file, RegExp regExp) { if (!file.existsSync()) { return null; } for (final String line in file.readAsLinesSync()) { final Match match = regExp.firstMatch(line); if (match != null) { return match; } } return null; } /// The macOS sub project. class MacOSProject extends FlutterProjectPlatform implements XcodeBasedProject { MacOSProject._(this.parent); @override final FlutterProject parent; @override String get pluginConfigKey => MacOSPlugin.kConfigKey; static const String _hostAppProjectName = 'Runner'; @override bool existsSync() => _macOSDirectory.existsSync(); Directory get _macOSDirectory => parent.directory.childDirectory('macos'); /// The directory in the project that is managed by Flutter. As much as /// possible, files that are edited by Flutter tooling after initial project /// creation should live here. Directory get managedDirectory => _macOSDirectory.childDirectory('Flutter'); /// The subdirectory of [managedDirectory] that contains files that are /// generated on the fly. All generated files that are not intended to be /// checked in should live here. Directory get ephemeralDirectory => managedDirectory.childDirectory('ephemeral'); /// The xcfilelist used to track the inputs for the Flutter script phase in /// the Xcode build. File get inputFileList => ephemeralDirectory.childFile('FlutterInputs.xcfilelist'); /// The xcfilelist used to track the outputs for the Flutter script phase in /// the Xcode build. File get outputFileList => ephemeralDirectory.childFile('FlutterOutputs.xcfilelist'); @override File get generatedXcodePropertiesFile => ephemeralDirectory.childFile('Flutter-Generated.xcconfig'); @override File xcodeConfigFor(String mode) => managedDirectory.childFile('Flutter-$mode.xcconfig'); @override File get generatedEnvironmentVariableExportScript => ephemeralDirectory.childFile('flutter_export_environment.sh'); @override File get podfile => _macOSDirectory.childFile('Podfile'); @override File get podfileLock => _macOSDirectory.childFile('Podfile.lock'); @override File get podManifestLock => _macOSDirectory.childDirectory('Pods').childFile('Manifest.lock'); @override Directory get xcodeProject => _macOSDirectory.childDirectory('$_hostAppProjectName.xcodeproj'); @override File get xcodeProjectInfoFile => xcodeProject.childFile('project.pbxproj'); @override Directory get xcodeWorkspace => _macOSDirectory.childDirectory('$_hostAppProjectName.xcworkspace'); /// The file where the Xcode build will write the name of the built app. /// /// Ideally this will be replaced in the future with inspection of the Runner /// scheme's target. File get nameFile => ephemeralDirectory.childFile('.app_filename'); Future<void> ensureReadyForPlatformSpecificTooling() async { // TODO(stuartmorgan): Add create-from-template logic here. await _updateGeneratedXcodeConfigIfNeeded(); } Future<void> _updateGeneratedXcodeConfigIfNeeded() async { if (globals.cache.isOlderThanToolsStamp(generatedXcodePropertiesFile)) { await xcode.updateGeneratedXcodeProperties( project: parent, buildInfo: BuildInfo.debug, useMacOSConfig: true, setSymroot: false, ); } } } /// The Windows sub project. class WindowsProject extends FlutterProjectPlatform implements CmakeBasedProject { WindowsProject._(this.parent); @override final FlutterProject parent; @override String get pluginConfigKey => WindowsPlugin.kConfigKey; String get _childDirectory => 'windows'; @override bool existsSync() => _editableDirectory.existsSync() && cmakeFile.existsSync(); @override File get cmakeFile => _editableDirectory.childFile('CMakeLists.txt'); @override File get managedCmakeFile => managedDirectory.childFile('CMakeLists.txt'); @override File get generatedCmakeConfigFile => ephemeralDirectory.childFile('generated_config.cmake'); @override File get generatedPluginCmakeFile => managedDirectory.childFile('generated_plugins.cmake'); @override Directory get pluginSymlinkDirectory => ephemeralDirectory.childDirectory('.plugin_symlinks'); Directory get _editableDirectory => parent.directory.childDirectory(_childDirectory); /// The directory in the project that is managed by Flutter. As much as /// possible, files that are edited by Flutter tooling after initial project /// creation should live here. Directory get managedDirectory => _editableDirectory.childDirectory('flutter'); /// The subdirectory of [managedDirectory] that contains files that are /// generated on the fly. All generated files that are not intended to be /// checked in should live here. Directory get ephemeralDirectory => managedDirectory.childDirectory('ephemeral'); Future<void> ensureReadyForPlatformSpecificTooling() async {} } /// The Windows UWP version of the Windows project. class WindowsUwpProject extends WindowsProject { WindowsUwpProject._(FlutterProject parent) : super._(parent); @override String get _childDirectory => 'windows-uwp'; /// Eventually this will be used to check if the user's unstable project needs to be regenerated. int get projectVersion => int.tryParse(_editableDirectory.childFile('project_version').readAsStringSync()); } /// The Linux sub project. class LinuxProject extends FlutterProjectPlatform implements CmakeBasedProject { LinuxProject._(this.parent); @override final FlutterProject parent; @override String get pluginConfigKey => LinuxPlugin.kConfigKey; static final RegExp _applicationIdPattern = RegExp(r'''^\s*set\s*\(\s*APPLICATION_ID\s*"(.*)"\s*\)\s*$'''); Directory get _editableDirectory => parent.directory.childDirectory('linux'); /// The directory in the project that is managed by Flutter. As much as /// possible, files that are edited by Flutter tooling after initial project /// creation should live here. Directory get managedDirectory => _editableDirectory.childDirectory('flutter'); /// The subdirectory of [managedDirectory] that contains files that are /// generated on the fly. All generated files that are not intended to be /// checked in should live here. Directory get ephemeralDirectory => managedDirectory.childDirectory('ephemeral'); @override bool existsSync() => _editableDirectory.existsSync(); @override File get cmakeFile => _editableDirectory.childFile('CMakeLists.txt'); @override File get managedCmakeFile => managedDirectory.childFile('CMakeLists.txt'); @override File get generatedCmakeConfigFile => ephemeralDirectory.childFile('generated_config.cmake'); @override File get generatedPluginCmakeFile => managedDirectory.childFile('generated_plugins.cmake'); @override Directory get pluginSymlinkDirectory => ephemeralDirectory.childDirectory('.plugin_symlinks'); Future<void> ensureReadyForPlatformSpecificTooling() async {} String get applicationId { return _firstMatchInFile(cmakeFile, _applicationIdPattern)?.group(1); } } /// The Fuchsia sub project. class FuchsiaProject { FuchsiaProject._(this.project); final FlutterProject project; Directory _editableHostAppDirectory; Directory get editableHostAppDirectory => _editableHostAppDirectory ??= project.directory.childDirectory('fuchsia'); bool existsSync() => editableHostAppDirectory.existsSync(); Directory _meta; Directory get meta => _meta ??= editableHostAppDirectory.childDirectory('meta'); }