| // Copyright 2018 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| import 'dart:async'; |
| import 'dart:convert'; |
| |
| import 'android/gradle.dart' as gradle; |
| import 'base/file_system.dart'; |
| import 'bundle.dart' as bundle; |
| import 'cache.dart'; |
| import 'flutter_manifest.dart'; |
| import 'ios/xcodeproj.dart' as xcode; |
| import 'plugins.dart'; |
| import 'template.dart'; |
| |
| /// Represents the contents of a Flutter project at the specified [directory]. |
| /// |
| /// Instances should be treated as immutable snapshots, to be replaced by new |
| /// instances on changes to `pubspec.yaml` files. |
| class FlutterProject { |
| FlutterProject._(this.directory, this.manifest, this._exampleManifest); |
| |
| /// Returns a future that completes with a FlutterProject view of the given directory. |
| static Future<FlutterProject> fromDirectory(Directory directory) async { |
| final FlutterManifest manifest = await FlutterManifest.createFromPath( |
| directory.childFile(bundle.defaultManifestPath).path, |
| ); |
| final Directory exampleDirectory = directory.childDirectory('example'); |
| final FlutterManifest exampleManifest = await FlutterManifest.createFromPath( |
| exampleDirectory.childFile(bundle.defaultManifestPath).path, |
| ); |
| return new FlutterProject._(directory, manifest, exampleManifest); |
| } |
| |
| /// Returns a future that completes with a FlutterProject view of the current directory. |
| static Future<FlutterProject> current() => fromDirectory(fs.currentDirectory); |
| |
| /// Returns a future that completes with a FlutterProject view of the given directory. |
| static Future<FlutterProject> fromPath(String path) => fromDirectory(fs.directory(path)); |
| |
| /// The location of this project. |
| final Directory directory; |
| |
| /// The manifest of this project, or null, if `pubspec.yaml` is invalid. |
| final FlutterManifest manifest; |
| |
| /// The manifest of the example sub-project of this project, or null, if |
| /// `example/pubspec.yaml` is invalid. |
| final FlutterManifest _exampleManifest; |
| |
| /// Asynchronously returns the organization names found in this project as |
| /// part of iOS product bundle identifier, Android application ID, or |
| /// Gradle group ID. |
| Future<Set<String>> organizationNames() async { |
| final List<String> candidates = await Future.wait(<Future<String>>[ |
| ios.productBundleIdentifier(), |
| android.applicationId(), |
| android.group(), |
| example.android.applicationId(), |
| example.ios.productBundleIdentifier(), |
| ]); |
| return new Set<String>.from(candidates |
| .map(_organizationNameFromPackageName) |
| .where((String name) => name != null)); |
| } |
| |
| String _organizationNameFromPackageName(String packageName) { |
| if (packageName != null && 0 <= packageName.lastIndexOf('.')) |
| return packageName.substring(0, packageName.lastIndexOf('.')); |
| else |
| return null; |
| } |
| |
| /// The iOS sub project of this project. |
| IosProject get ios => new IosProject(directory.childDirectory('ios')); |
| |
| /// The Android sub project of this project. |
| AndroidProject get android => new AndroidProject(directory.childDirectory('android')); |
| |
| /// The generated AndroidModule sub project of this module project. |
| AndroidModuleProject get androidModule => new AndroidModuleProject(directory.childDirectory('.android')); |
| |
| /// The generated IosModule sub project of this module project. |
| IosModuleProject get iosModule => new IosModuleProject(directory.childDirectory('.ios')); |
| |
| File get androidLocalPropertiesFile { |
| return directory |
| .childDirectory(manifest.isModule ? '.android' : 'android') |
| .childFile('local.properties'); |
| } |
| |
| File get generatedXcodePropertiesFile { |
| return directory |
| .childDirectory(manifest.isModule ? '.ios' : 'ios') |
| .childDirectory('Flutter') |
| .childFile('Generated.xcconfig'); |
| } |
| |
| File get flutterPluginsFile => directory.childFile('.flutter-plugins'); |
| |
| Directory get androidPluginRegistrantHost { |
| return manifest.isModule |
| ? directory.childDirectory('.android').childDirectory('Flutter') |
| : directory.childDirectory('android').childDirectory('app'); |
| } |
| |
| Directory get iosPluginRegistrantHost { |
| // In a module create the GeneratedPluginRegistrant as a pod to be included |
| // from a hosting app. |
| // For a non-module create the GeneratedPluginRegistrant as source files |
| // directly in the iOS project. |
| return manifest.isModule |
| ? directory.childDirectory('.ios').childDirectory('Flutter').childDirectory('FlutterPluginRegistrant') |
| : directory.childDirectory('ios').childDirectory('Runner'); |
| } |
| |
| /// The example sub-project of this project. |
| FlutterProject get example => new FlutterProject._(_exampleDirectory, _exampleManifest, FlutterManifest.empty()); |
| |
| /// True, if this project has an example application |
| bool get hasExampleApp => _exampleDirectory.childFile('pubspec.yaml').existsSync(); |
| |
| /// The directory that will contain the example if an example exists. |
| Directory get _exampleDirectory => directory.childDirectory('example'); |
| |
| /// Generates project files necessary to make Gradle builds work on Android |
| /// and CocoaPods+Xcode work on iOS, for app and module projects only. |
| Future<void> ensureReadyForPlatformSpecificTooling() async { |
| if (!directory.existsSync() || hasExampleApp) { |
| return; |
| } |
| if (manifest.isModule) { |
| await androidModule.ensureReadyForPlatformSpecificTooling(this); |
| await iosModule.ensureReadyForPlatformSpecificTooling(); |
| } |
| await xcode.generateXcodeProperties(project: this); |
| await injectPlugins(this); |
| } |
| } |
| |
| /// Represents the contents of the ios/ folder of a Flutter project. |
| class IosProject { |
| static final RegExp _productBundleIdPattern = new RegExp(r'^\s*PRODUCT_BUNDLE_IDENTIFIER\s*=\s*(.*);\s*$'); |
| IosProject(this.directory); |
| |
| final Directory directory; |
| |
| /// The xcode config file for [mode]. |
| File xcodeConfigFor(String mode) => directory.childDirectory('Flutter').childFile('$mode.xcconfig'); |
| |
| /// The 'Podfile'. |
| File get podfile => directory.childFile('Podfile'); |
| |
| /// The 'Podfile.lock'. |
| File get podfileLock => directory.childFile('Podfile.lock'); |
| |
| /// The 'Manifest.lock'. |
| File get podManifestLock => directory.childDirectory('Pods').childFile('Manifest.lock'); |
| |
| Future<String> productBundleIdentifier() { |
| final File projectFile = directory.childDirectory('Runner.xcodeproj').childFile('project.pbxproj'); |
| return _firstMatchInFile(projectFile, _productBundleIdPattern).then((Match match) => match?.group(1)); |
| } |
| } |
| |
| /// Represents the contents of the .ios/ folder of a Flutter module |
| /// project. |
| class IosModuleProject { |
| IosModuleProject(this.directory); |
| |
| final Directory directory; |
| |
| Future<void> ensureReadyForPlatformSpecificTooling() async { |
| if (_shouldRegenerate()) { |
| final Template template = new Template.fromName(fs.path.join('module', 'ios')); |
| template.render(directory, <String, dynamic>{}, printStatusWhenWriting: false); |
| } |
| } |
| |
| bool _shouldRegenerate() { |
| return Cache.instance.fileOlderThanToolsStamp(directory.childFile('podhelper.rb')); |
| } |
| } |
| |
| /// Represents the contents of the android/ folder of a Flutter project. |
| class AndroidProject { |
| static final RegExp _applicationIdPattern = new RegExp('^\\s*applicationId\\s+[\'\"](.*)[\'\"]\\s*\$'); |
| static final RegExp _groupPattern = new RegExp('^\\s*group\\s+[\'\"](.*)[\'\"]\\s*\$'); |
| |
| AndroidProject(this.directory); |
| |
| File get gradleManifestFile { |
| return isUsingGradle() |
| ? fs.file(fs.path.join(directory.path, 'app', 'src', 'main', 'AndroidManifest.xml')) |
| : directory.childFile('AndroidManifest.xml'); |
| } |
| |
| File get gradleAppOutV1File => gradleAppOutV1Directory.childFile('app-debug.apk'); |
| |
| Directory get gradleAppOutV1Directory { |
| return fs.directory(fs.path.join(directory.path, 'app', 'build', 'outputs', 'apk')); |
| } |
| |
| bool isUsingGradle() { |
| return directory.childFile('build.gradle').existsSync(); |
| } |
| |
| final Directory directory; |
| |
| Future<String> applicationId() { |
| final File gradleFile = directory.childDirectory('app').childFile('build.gradle'); |
| return _firstMatchInFile(gradleFile, _applicationIdPattern).then((Match match) => match?.group(1)); |
| } |
| |
| Future<String> group() { |
| final File gradleFile = directory.childFile('build.gradle'); |
| return _firstMatchInFile(gradleFile, _groupPattern).then((Match match) => match?.group(1)); |
| } |
| } |
| |
| /// Represents the contents of the .android/ folder of a Flutter module project. |
| class AndroidModuleProject { |
| AndroidModuleProject(this.directory); |
| |
| final Directory directory; |
| |
| Future<void> ensureReadyForPlatformSpecificTooling(FlutterProject project) async { |
| if (_shouldRegenerate()) { |
| final Template template = new Template.fromName(fs.path.join('module', 'android')); |
| template.render( |
| directory, |
| <String, dynamic>{ |
| 'androidIdentifier': project.manifest.androidPackage, |
| }, |
| printStatusWhenWriting: false, |
| ); |
| gradle.injectGradleWrapper(directory); |
| } |
| await gradle.updateLocalProperties(project: project, requireAndroidSdk: false); |
| } |
| |
| bool _shouldRegenerate() { |
| return Cache.instance.fileOlderThanToolsStamp(directory.childFile('build.gradle')); |
| } |
| } |
| |
| /// Asynchronously returns the first line-based match for [regExp] in [file]. |
| /// |
| /// Assumes UTF8 encoding. |
| Future<Match> _firstMatchInFile(File file, RegExp regExp) async { |
| if (!await file.exists()) { |
| return null; |
| } |
| return file |
| .openRead() |
| .transform(utf8.decoder) |
| .transform(const LineSplitter()) |
| .map(regExp.firstMatch) |
| .firstWhere((Match match) => match != null, orElse: () => null); |
| } |