Commit 62fb32d4 authored by Chinmay Garde's avatar Chinmay Garde

Merge pull request #1792 from chinmaygarde/master

Allow users to upgrade their iOS engines without removing their edits to the Xcode project
parents 2a1b0f6c 23c632ab
...@@ -82,7 +82,7 @@ class AndroidApk extends ApplicationPackage { ...@@ -82,7 +82,7 @@ class AndroidApk extends ApplicationPackage {
class IOSApp extends ApplicationPackage { class IOSApp extends ApplicationPackage {
static const String _defaultId = 'io.flutter.runner.Runner'; static const String _defaultId = 'io.flutter.runner.Runner';
static const String _defaultPath = 'ios'; static const String _defaultPath = 'ios/Generated';
IOSApp({ IOSApp({
String localPath: _defaultPath, String localPath: _defaultPath,
......
...@@ -59,6 +59,11 @@ class IOSCommand extends FlutterCommand { ...@@ -59,6 +59,11 @@ class IOSCommand extends FlutterCommand {
tempFile.writeAsBytesSync(archiveBytes); tempFile.writeAsBytesSync(archiveBytes);
try { try {
// Remove the old generated project if one is present
runCheckedSync(['/bin/rm', '-rf', directory]);
// Create the directory so unzip can write to it
runCheckedSync(['/bin/mkdir', '-p', directory]);
// Unzip the Xcode project into the new empty directory
runCheckedSync(['/usr/bin/unzip', tempFile.path, '-d', directory]); runCheckedSync(['/usr/bin/unzip', tempFile.path, '-d', directory]);
} catch (error) { } catch (error) {
return false; return false;
...@@ -88,6 +93,41 @@ class IOSCommand extends FlutterCommand { ...@@ -88,6 +93,41 @@ class IOSCommand extends FlutterCommand {
return true; return true;
} }
void _writeUserEditableFilesIfNecessary(String directory) {
printStatus("Checking if user editable files need updates");
// Step 1: Check if the Info.plist exists and write one if not
File infoPlist = new File(path.join(directory, "Info.plist"));
if (!infoPlist.existsSync()) {
printStatus("Did not find an existing Info.plist. Creating one.");
infoPlist.writeAsStringSync(_infoPlistInitialContents);
} else {
printStatus("Info.plist present. Using existing.");
}
// Step 2: Check if the LaunchScreen.storyboard exists and write one if not
File launchScreen = new File(path.join(directory, "LaunchScreen.storyboard"));
if (!launchScreen.existsSync()) {
printStatus("Did not find an existing LaunchScreen.storyboard. Creating one.");
launchScreen.writeAsStringSync(_launchScreenInitialContents);
} else {
printStatus("LaunchScreen.storyboard present. Using existing.");
}
// Step 3: Check if the Assets.xcassets exists and write one if not
Directory xcassets = new Directory(path.join(directory, "Assets.xcassets"));
if (!xcassets.existsSync()) {
printStatus("Did not find an existing Assets.xcassets. Creating one.");
Directory iconsAssetsDir = new Directory(path.join(xcassets.path, "AppIcon.appiconset"));
iconsAssetsDir.createSync(recursive: true);
File iconContents = new File(path.join(iconsAssetsDir.path, "Contents.json"));
iconContents.writeAsStringSync(_iconAssetInitialContents);
} else {
printStatus("Assets.xcassets present. Using existing.");
}
}
void _setupXcodeProjXcconfig(String filePath) { void _setupXcodeProjXcconfig(String filePath) {
StringBuffer localsBuffer = new StringBuffer(); StringBuffer localsBuffer = new StringBuffer();
...@@ -111,7 +151,8 @@ class IOSCommand extends FlutterCommand { ...@@ -111,7 +151,8 @@ class IOSCommand extends FlutterCommand {
Future<int> _runInitCommand() async { Future<int> _runInitCommand() async {
// Step 1: Fetch the archive from the cloud // Step 1: Fetch the archive from the cloud
String xcodeprojPath = path.join(Directory.current.path, "ios"); String iosFilesPath = path.join(Directory.current.path, "ios");
String xcodeprojPath = path.join(iosFilesPath, "Generated");
List<int> archiveBytes = await _fetchXcodeArchive(); List<int> archiveBytes = await _fetchXcodeArchive();
if (archiveBytes.isEmpty) { if (archiveBytes.isEmpty) {
...@@ -122,21 +163,31 @@ class IOSCommand extends FlutterCommand { ...@@ -122,21 +163,31 @@ class IOSCommand extends FlutterCommand {
// Step 2: Inflate the archive into the user project directory // Step 2: Inflate the archive into the user project directory
bool result = await _inflateXcodeArchive(xcodeprojPath, archiveBytes); bool result = await _inflateXcodeArchive(xcodeprojPath, archiveBytes);
if (!result) { if (!result) {
printError("Error: Could not init the Xcode project: the 'ios' directory already exists."); printError("Could not inflate the Xcode project archive.");
printError("To proceed, remove the 'ios' directory and try again.");
printError("Warning: You may have made manual changes to files in the 'ios' directory.");
return 1; return 1;
} }
// Step 3: Populate the Local.xcconfig with project specific paths // Step 3: The generated project should NOT be checked into the users
// version control system. Be nice and write a gitignore for them if
// one does not exist.
File generatedGitignore = new File(path.join(iosFilesPath, ".gitignore"));
if (!generatedGitignore.existsSync()) {
generatedGitignore.writeAsStringSync("Generated/\n");
}
// Step 4: Setup default user editable files if this is the first run of
// the init command.
_writeUserEditableFilesIfNecessary(iosFilesPath);
// Step 5: Populate the Local.xcconfig with project specific paths
_setupXcodeProjXcconfig(path.join(xcodeprojPath, "Local.xcconfig")); _setupXcodeProjXcconfig(path.join(xcodeprojPath, "Local.xcconfig"));
// Step 4: Write the REVISION file // Step 6: Write the REVISION file
File revisionFile = new File(path.join(xcodeprojPath, "REVISION")); File revisionFile = new File(path.join(xcodeprojPath, "REVISION"));
revisionFile.createSync(); revisionFile.createSync();
revisionFile.writeAsStringSync(ArtifactStore.engineRevision); revisionFile.writeAsStringSync(ArtifactStore.engineRevision);
// Step 5: Tell the user the location of the generated project. // Step 7: Tell the user the location of the generated project.
printStatus("An Xcode project has been placed in 'ios/'."); printStatus("An Xcode project has been placed in 'ios/'.");
printStatus("You may edit it to modify iOS specific configuration."); printStatus("You may edit it to modify iOS specific configuration.");
return 0; return 0;
...@@ -156,3 +207,159 @@ class IOSCommand extends FlutterCommand { ...@@ -156,3 +207,159 @@ class IOSCommand extends FlutterCommand {
return 1; return 1;
} }
} }
final String _infoPlistInitialContents = '''
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>CFBundleDevelopmentRegion</key>
<string>en</string>
<key>CFBundleExecutable</key>
<string>Runner</string>
<key>CFBundleIdentifier</key>
<string>io.flutter.runner.Runner</string>
<key>CFBundleInfoDictionaryVersion</key>
<string>6.0</string>
<key>CFBundleName</key>
<string>Flutter</string>
<key>CFBundlePackageType</key>
<string>APPL</string>
<key>CFBundleShortVersionString</key>
<string>1.0</string>
<key>CFBundleSignature</key>
<string>????</string>
<key>CFBundleVersion</key>
<string>1</string>
<key>LSRequiresIPhoneOS</key>
<true/>
<key>UILaunchStoryboardName</key>
<string>LaunchScreen</string>
<key>UIRequiredDeviceCapabilities</key>
<array>
<string>arm64</string>
</array>
<key>UISupportedInterfaceOrientations</key>
<array>
<string>UIInterfaceOrientationPortrait</string>
<string>UIInterfaceOrientationLandscapeLeft</string>
<string>UIInterfaceOrientationLandscapeRight</string>
</array>
<key>UISupportedInterfaceOrientations~ipad</key>
<array>
<string>UIInterfaceOrientationPortrait</string>
<string>UIInterfaceOrientationPortraitUpsideDown</string>
<string>UIInterfaceOrientationLandscapeLeft</string>
<string>UIInterfaceOrientationLandscapeRight</string>
</array>
<key>UIViewControllerBasedStatusBarAppearance</key>
<false/>
</dict>
</plist>
''';
final String _launchScreenInitialContents = '''
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="9531" systemVersion="15C50" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" useTraitCollections="YES" initialViewController="01J-lp-oVM">
<dependencies>
<deployment identifier="iOS"/>
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="9529"/>
</dependencies>
<scenes>
<!--View Controller-->
<scene sceneID="EHf-IW-A2E">
<objects>
<viewController id="01J-lp-oVM" sceneMemberID="viewController">
<layoutGuides>
<viewControllerLayoutGuide type="top" id="Llm-lL-Icb"/>
<viewControllerLayoutGuide type="bottom" id="xb3-aO-Qok"/>
</layoutGuides>
<view key="view" contentMode="scaleToFill" id="Ze5-6b-2t3">
<rect key="frame" x="0.0" y="0.0" width="600" height="600"/>
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
<color key="backgroundColor" white="1" alpha="1" colorSpace="custom" customColorSpace="calibratedWhite"/>
</view>
</viewController>
<placeholder placeholderIdentifier="IBFirstResponder" id="iYj-Kq-Ea1" userLabel="First Responder" sceneMemberID="firstResponder"/>
</objects>
<point key="canvasLocation" x="53" y="375"/>
</scene>
</scenes>
</document>
''';
final String _iconAssetInitialContents = '''
{
"images" : [
{
"idiom" : "iphone",
"size" : "29x29",
"scale" : "2x"
},
{
"idiom" : "iphone",
"size" : "29x29",
"scale" : "3x"
},
{
"idiom" : "iphone",
"size" : "40x40",
"scale" : "2x"
},
{
"idiom" : "iphone",
"size" : "40x40",
"scale" : "3x"
},
{
"idiom" : "iphone",
"size" : "60x60",
"scale" : "2x"
},
{
"idiom" : "iphone",
"size" : "60x60",
"scale" : "3x"
},
{
"idiom" : "ipad",
"size" : "29x29",
"scale" : "1x"
},
{
"idiom" : "ipad",
"size" : "29x29",
"scale" : "2x"
},
{
"idiom" : "ipad",
"size" : "40x40",
"scale" : "1x"
},
{
"idiom" : "ipad",
"size" : "40x40",
"scale" : "2x"
},
{
"idiom" : "ipad",
"size" : "76x76",
"scale" : "1x"
},
{
"idiom" : "ipad",
"size" : "76x76",
"scale" : "2x"
},
{
"idiom" : "ipad",
"size" : "83.5x83.5",
"scale" : "2x"
}
],
"info" : {
"version" : 1,
"author" : "xcode"
}
}
''';
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment