| // 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 '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]. |
| class FlutterProject { |
| FlutterProject(this.directory); |
| |
| /// The location of this project. |
| final Directory directory; |
| |
| /// 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')); |
| |
| /// Returns true if this project has an example application |
| bool get hasExampleApp => directory.childDirectory('example').childFile('pubspec.yaml').existsSync(); |
| |
| /// The example sub project of this (package or plugin) project. |
| FlutterProject get example => new FlutterProject(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. |
| /// |
| /// Returns the number of files written. |
| Future<void> ensureReadyForPlatformSpecificTooling() async { |
| if (!directory.existsSync() || hasExampleApp) { |
| return 0; |
| } |
| final FlutterManifest manifest = await FlutterManifest.createFromPath(directory.childFile('pubspec.yaml').path); |
| if (manifest.isModule) { |
| await androidModule.ensureReadyForPlatformSpecificTooling(manifest); |
| } |
| injectPlugins(directory: directory.path); |
| await xcode.generateXcodeProperties(directory.path); |
| } |
| } |
| |
| /// 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; |
| |
| 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 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); |
| |
| 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-generated/ folder of a Flutter module |
| /// project. |
| class AndroidModuleProject { |
| AndroidModuleProject(this.directory); |
| |
| final Directory directory; |
| |
| Future<void> ensureReadyForPlatformSpecificTooling(FlutterManifest manifest) async { |
| if (_shouldRegenerate()) { |
| final Template template = new Template.fromName(fs.path.join('module', 'android')); |
| template.render(directory, <String, dynamic>{ |
| 'androidIdentifier': manifest.moduleDescriptor['androidPackage'], |
| }, printStatusWhenWriting: false); |
| gradle.injectGradleWrapper(directory); |
| } |
| gradle.updateLocalPropertiesSync(directory, manifest); |
| } |
| |
| bool _shouldRegenerate() { |
| final File flutterToolsStamp = Cache.instance.getStampFileFor('flutter_tools'); |
| final File buildDotGradleFile = directory.childFile('build.gradle'); |
| if (!buildDotGradleFile.existsSync()) |
| return true; |
| return flutterToolsStamp.existsSync() && |
| flutterToolsStamp |
| .lastModifiedSync() |
| .isAfter(buildDotGradleFile.lastModifiedSync()); |
| } |
| } |
| |
| /// 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); |
| } |