Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
477530f3
Commit
477530f3
authored
Jan 19, 2016
by
Chinmay Garde
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1249 from chinmaygarde/master
First take on `flutter start` for iOS devices
parents
32078052
c8672a40
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
165 additions
and
71 deletions
+165
-71
application_package.dart
packages/flutter_tools/lib/src/application_package.dart
+5
-7
start.dart
packages/flutter_tools/lib/src/commands/start.dart
+24
-15
device.dart
packages/flutter_tools/lib/src/device.dart
+124
-49
flutter_command_runner.dart
.../flutter_tools/lib/src/runner/flutter_command_runner.dart
+12
-0
No files found.
packages/flutter_tools/lib/src/application_package.dart
View file @
477530f3
...
@@ -47,11 +47,11 @@ class AndroidApk extends ApplicationPackage {
...
@@ -47,11 +47,11 @@ class AndroidApk extends ApplicationPackage {
}
}
class
IOSApp
extends
ApplicationPackage
{
class
IOSApp
extends
ApplicationPackage
{
static
const
String
_default
Name
=
'SkyShell.app
'
;
static
const
String
_default
Id
=
'io.flutter.runner.Runner
'
;
static
const
String
_default
Id
=
'com.google.SkyShell
'
;
static
const
String
_default
Path
=
'ios
'
;
IOSApp
({
IOSApp
({
String
localPath
,
String
localPath
:
_defaultPath
,
String
id:
_defaultId
String
id:
_defaultId
})
:
super
(
localPath:
localPath
,
id:
id
);
})
:
super
(
localPath:
localPath
,
id:
id
);
}
}
...
@@ -98,14 +98,12 @@ class ApplicationPackageStore {
...
@@ -98,14 +98,12 @@ class ApplicationPackageStore {
case
TargetPlatform
.
iOS
:
case
TargetPlatform
.
iOS
:
assert
(
iOS
==
null
);
assert
(
iOS
==
null
);
assert
(
config
.
type
!=
BuildType
.
prebuilt
);
iOS
=
new
IOSApp
();
iOS
=
new
IOSApp
(
localPath:
path
.
join
(
config
.
buildDir
,
IOSApp
.
_defaultName
));
break
;
break
;
case
TargetPlatform
.
iOSSimulator
:
case
TargetPlatform
.
iOSSimulator
:
assert
(
iOSSimulator
==
null
);
assert
(
iOSSimulator
==
null
);
assert
(
config
.
type
!=
BuildType
.
prebuilt
);
iOSSimulator
=
new
IOSApp
();
iOSSimulator
=
new
IOSApp
(
localPath:
path
.
join
(
config
.
buildDir
,
IOSApp
.
_defaultName
));
break
;
break
;
case
TargetPlatform
.
mac
:
case
TargetPlatform
.
mac
:
...
...
packages/flutter_tools/lib/src/commands/start.dart
View file @
477530f3
...
@@ -9,6 +9,7 @@ import 'package:path/path.dart' as path;
...
@@ -9,6 +9,7 @@ import 'package:path/path.dart' as path;
import
'../application_package.dart'
;
import
'../application_package.dart'
;
import
'../base/logging.dart'
;
import
'../base/logging.dart'
;
import
'../build_configuration.dart'
;
import
'../device.dart'
;
import
'../device.dart'
;
import
'../runner/flutter_command.dart'
;
import
'../runner/flutter_command.dart'
;
import
'build.dart'
;
import
'build.dart'
;
...
@@ -84,6 +85,8 @@ abstract class StartCommandBase extends FlutterCommand {
...
@@ -84,6 +85,8 @@ abstract class StartCommandBase extends FlutterCommand {
continue
;
continue
;
logging
.
fine
(
'Running build command for
$device
.'
);
logging
.
fine
(
'Running build command for
$device
.'
);
if
(
device
.
platform
==
TargetPlatform
.
android
)
{
BuildCommand
builder
=
new
BuildCommand
();
BuildCommand
builder
=
new
BuildCommand
();
builder
.
inheritFromParent
(
this
);
builder
.
inheritFromParent
(
this
);
await
builder
.
buildInTempDir
(
await
builder
.
buildInTempDir
(
...
@@ -100,6 +103,12 @@ abstract class StartCommandBase extends FlutterCommand {
...
@@ -100,6 +103,12 @@ abstract class StartCommandBase extends FlutterCommand {
startedSomething
=
true
;
startedSomething
=
true
;
}
}
);
);
}
else
{
bool
result
=
await
device
.
startApp
(
package
);
if
(!
result
)
{
logging
.
severe
(
'Could not start
\'
${package.name}
\'
on
\'
${device.id}
\'
'
);
}
}
}
}
if
(!
startedSomething
)
{
if
(!
startedSomething
)
{
...
...
packages/flutter_tools/lib/src/device.dart
View file @
477530f3
...
@@ -95,8 +95,7 @@ class IOSDevice extends Device {
...
@@ -95,8 +95,7 @@ class IOSDevice extends Device {
'To copy files to iOS devices, please install ios-deploy. '
'To copy files to iOS devices, please install ios-deploy. '
'You can do this using homebrew as follows:
\n
'
'You can do this using homebrew as follows:
\n
'
'
\$
brew tap flutter/flutter
\n
'
'
\$
brew tap flutter/flutter
\n
'
'
\$
brew install ios-deploy'
,
'
\$
brew install ios-deploy'
);
'Copying files to iOS devices is not currently supported on Linux.'
);
}
}
static
List
<
IOSDevice
>
getAttachedDevices
([
IOSDevice
mockIOS
])
{
static
List
<
IOSDevice
>
getAttachedDevices
([
IOSDevice
mockIOS
])
{
...
@@ -178,7 +177,7 @@ class IOSDevice extends Device {
...
@@ -178,7 +177,7 @@ class IOSDevice extends Device {
@override
@override
bool
isAppInstalled
(
ApplicationPackage
app
)
{
bool
isAppInstalled
(
ApplicationPackage
app
)
{
try
{
try
{
String
apps
=
runCheckedSync
([
installerPath
,
'-
l
'
]);
String
apps
=
runCheckedSync
([
installerPath
,
'-
-list-apps
'
]);
if
(
new
RegExp
(
app
.
id
,
multiLine:
true
).
hasMatch
(
apps
))
{
if
(
new
RegExp
(
app
.
id
,
multiLine:
true
).
hasMatch
(
apps
))
{
return
true
;
return
true
;
}
}
...
@@ -190,22 +189,42 @@ class IOSDevice extends Device {
...
@@ -190,22 +189,42 @@ class IOSDevice extends Device {
@override
@override
Future
<
bool
>
startApp
(
ApplicationPackage
app
)
async
{
Future
<
bool
>
startApp
(
ApplicationPackage
app
)
async
{
if
(!
isAppInstalled
(
app
))
{
logging
.
fine
(
"Attempting to build and install
${app.name}
on
$id
"
);
// Step 1: Install the precompiled application if necessary
bool
buildResult
=
await
_buildIOSXcodeProject
(
app
,
true
);
if
(!
buildResult
)
{
logging
.
severe
(
'Could not build the precompiled application for the device'
);
return
false
;
return
false
;
}
}
// idevicedebug hangs forever after launching the app, so kill it after
// giving it plenty of time to send the launch command.
// Step 2: Check that the application exists at the specified path
return
await
runAndKill
(
Directory
bundle
=
new
Directory
(
path
.
join
(
app
.
localPath
,
'build'
,
'Release-iphoneos'
,
'Runner.app'
));
[
debuggerPath
,
'run'
,
app
.
id
],
new
Duration
(
seconds:
3
)
bool
bundleExists
=
await
bundle
.
exists
();
).
then
(
if
(!
bundleExists
)
{
(
_
)
{
logging
.
severe
(
'Could not find the built application bundle at
${bundle.path}
'
);
return
true
;
},
onError:
(
e
)
{
logging
.
info
(
'Failure running
$debuggerPath
: '
,
e
);
return
false
;
return
false
;
}
}
);
// Step 3: Attempt to install the application on the device
int
installationResult
=
await
runCommandAndStreamOutput
([
'/usr/bin/env'
,
'ios-deploy'
,
'--id'
,
id
,
'--bundle'
,
bundle
.
path
,
]);
if
(
installationResult
!=
0
)
{
logging
.
severe
(
'Could not install
${bundle.path}
on
$id
'
);
return
false
;
}
logging
.
fine
(
'Installation successful'
);
return
true
;
}
}
@override
@override
...
@@ -230,11 +249,6 @@ class IOSDevice extends Device {
...
@@ -230,11 +249,6 @@ class IOSDevice extends Device {
]);
]);
return
true
;
return
true
;
}
else
{
}
else
{
// TODO(iansf): It may be possible to make this work on Linux. Since this
// functionality appears to be the only that prevents us from
// supporting iOS on Linux, it may be worth putting some time
// into investigating this.
// See https://bbs.archlinux.org/viewtopic.php?id=192655
return
false
;
return
false
;
}
}
return
false
;
return
false
;
...
@@ -337,8 +351,6 @@ class IOSSimulator extends Device {
...
@@ -337,8 +351,6 @@ class IOSSimulator extends Device {
List
<
IOSSimulator
>
devices
=
[];
List
<
IOSSimulator
>
devices
=
[];
String
id
=
_getRunningSimulatorID
(
mockIOS
);
String
id
=
_getRunningSimulatorID
(
mockIOS
);
if
(
id
!=
null
)
{
if
(
id
!=
null
)
{
// TODO(iansf): get the simulator's name
// String name = _getDeviceName(id, mockIOS);
devices
.
add
(
new
IOSSimulator
(
id:
id
));
devices
.
add
(
new
IOSSimulator
(
id:
id
));
}
}
return
devices
;
return
devices
;
...
@@ -421,20 +433,53 @@ class IOSSimulator extends Device {
...
@@ -421,20 +433,53 @@ class IOSSimulator extends Device {
@override
@override
Future
<
bool
>
startApp
(
ApplicationPackage
app
)
async
{
Future
<
bool
>
startApp
(
ApplicationPackage
app
)
async
{
if
(!
isAppInstalled
(
app
))
{
logging
.
fine
(
'Building
${app.name}
for
$id
'
);
// Step 1: Build the Xcode project
bool
buildResult
=
await
_buildIOSXcodeProject
(
app
,
false
);
if
(!
buildResult
)
{
logging
.
severe
(
'Could not build the application for the simulator'
);
return
false
;
return
false
;
}
}
try
{
if
(
id
==
defaultDeviceID
)
{
// Step 2: Assert that the Xcode project was successfully built
runCheckedSync
(
Directory
bundle
=
new
Directory
(
path
.
join
(
app
.
localPath
,
'build'
,
'Release-iphonesimulator'
,
'Runner.app'
));
[
xcrunPath
,
'simctl'
,
'launch'
,
'booted'
,
app
.
id
]);
bool
bundleExists
=
await
bundle
.
exists
();
}
else
{
if
(!
bundleExists
)
{
runCheckedSync
([
xcrunPath
,
'simctl'
,
'launch'
,
id
,
app
.
id
]);
logging
.
severe
(
'Could not find the built application bundle at
${bundle.path}
'
);
return
false
;
}
}
return
true
;
}
catch
(
e
)
{
// Step 3: Install the updated bundle to the simulator
int
installResult
=
await
runCommandAndStreamOutput
([
xcrunPath
,
'simctl'
,
'install'
,
id
==
defaultDeviceID
?
'booted'
:
id
,
path
.
absolute
(
bundle
.
path
)
]);
if
(
installResult
!=
0
)
{
logging
.
severe
(
'Could not install the application bundle on the simulator'
);
return
false
;
return
false
;
}
}
// Step 4: Launch the updated application in the simulator
int
launchResult
=
await
runCommandAndStreamOutput
([
xcrunPath
,
'simctl'
,
'launch'
,
id
==
defaultDeviceID
?
'booted'
:
id
,
app
.
id
]);
if
(
launchResult
!=
0
)
{
logging
.
severe
(
'Could not launch the freshly installed application on the simulator'
);
return
false
;
}
logging
.
fine
(
'Successfully started
${app.name}
on
$id
'
);
return
true
;
}
}
@override
@override
...
@@ -973,6 +1018,30 @@ class DeviceStore {
...
@@ -973,6 +1018,30 @@ class DeviceStore {
this
.
iOSSimulator
this
.
iOSSimulator
});
});
static
Device
_deviceForConfig
(
BuildConfiguration
config
,
List
<
Device
>
devices
)
{
Device
device
=
null
;
if
(
config
.
deviceId
!=
null
)
{
// Step 1: If a device identifier is specified, try to find a device
// matching that specific identifier
device
=
devices
.
firstWhere
(
(
Device
dev
)
=>
(
dev
.
id
==
config
.
deviceId
),
orElse:
()
=>
null
);
if
(
device
==
null
)
{
logging
.
severe
(
'Warning: Device ID
${config.deviceId}
not found'
);
}
}
else
if
(
devices
.
length
==
1
)
{
// Step 2: If no identifier is specified and there is only one connected
// device, pick that one.
device
=
devices
[
0
];
}
else
if
(
devices
.
length
>
1
)
{
// Step 3: D:
logging
.
severe
(
'Warning: Multiple devices are connected, but no device ID was specified.'
);
}
return
device
;
}
factory
DeviceStore
.
forConfigs
(
List
<
BuildConfiguration
>
configs
)
{
factory
DeviceStore
.
forConfigs
(
List
<
BuildConfiguration
>
configs
)
{
AndroidDevice
android
;
AndroidDevice
android
;
IOSDevice
iOS
;
IOSDevice
iOS
;
...
@@ -982,27 +1051,19 @@ class DeviceStore {
...
@@ -982,27 +1051,19 @@ class DeviceStore {
switch
(
config
.
targetPlatform
)
{
switch
(
config
.
targetPlatform
)
{
case
TargetPlatform
.
android
:
case
TargetPlatform
.
android
:
assert
(
android
==
null
);
assert
(
android
==
null
);
List
<
AndroidDevice
>
androidDevices
=
AndroidDevice
.
getAttachedDevices
();
android
=
_deviceForConfig
(
config
,
AndroidDevice
.
getAttachedDevices
());
if
(
config
.
deviceId
!=
null
)
{
android
=
androidDevices
.
firstWhere
(
(
AndroidDevice
dev
)
=>
(
dev
.
id
==
config
.
deviceId
),
orElse:
()
=>
null
);
if
(
android
==
null
)
{
print
(
'Warning: Device ID
${config.deviceId}
not found'
);
}
}
else
if
(
androidDevices
.
length
==
1
)
{
android
=
androidDevices
[
0
];
}
else
if
(
androidDevices
.
length
>
1
)
{
print
(
'Warning: Multiple Android devices are connected, but no device ID was specified.'
);
}
break
;
break
;
case
TargetPlatform
.
iOS
:
case
TargetPlatform
.
iOS
:
assert
(
iOS
==
null
);
assert
(
iOS
==
null
);
iOS
=
new
IOSDevice
(
);
iOS
=
_deviceForConfig
(
config
,
IOSDevice
.
getAttachedDevices
()
);
break
;
break
;
case
TargetPlatform
.
iOSSimulator
:
case
TargetPlatform
.
iOSSimulator
:
assert
(
iOSSimulator
==
null
);
assert
(
iOSSimulator
==
null
);
iOSSimulator
=
_deviceForConfig
(
config
,
IOSSimulator
.
getAttachedDevices
());
if
(
iOSSimulator
==
null
)
{
// Creates a simulator with the default identifier
iOSSimulator
=
new
IOSSimulator
();
iOSSimulator
=
new
IOSSimulator
();
}
break
;
break
;
case
TargetPlatform
.
mac
:
case
TargetPlatform
.
mac
:
case
TargetPlatform
.
linux
:
case
TargetPlatform
.
linux
:
...
@@ -1013,3 +1074,17 @@ class DeviceStore {
...
@@ -1013,3 +1074,17 @@ class DeviceStore {
return
new
DeviceStore
(
android:
android
,
iOS:
iOS
,
iOSSimulator:
iOSSimulator
);
return
new
DeviceStore
(
android:
android
,
iOS:
iOS
,
iOSSimulator:
iOSSimulator
);
}
}
}
}
Future
<
bool
>
_buildIOSXcodeProject
(
ApplicationPackage
app
,
bool
isDevice
)
async
{
List
<
String
>
command
=
[
'/usr/bin/env'
,
'xcrun'
,
'xcodebuild'
,
'-target'
,
'Runner'
,
'-configuration'
,
'Release'
];
if
(!
isDevice
)
{
command
.
addAll
([
'-sdk'
,
'iphonesimulator'
]);
}
int
result
=
await
runCommandAndStreamOutput
(
command
,
workingDirectory:
app
.
localPath
);
return
result
==
0
;
}
packages/flutter_tools/lib/src/runner/flutter_command_runner.dart
View file @
477530f3
...
@@ -226,6 +226,18 @@ class FlutterCommandRunner extends CommandRunner {
...
@@ -226,6 +226,18 @@ class FlutterCommandRunner extends CommandRunner {
testable:
true
testable:
true
));
));
}
}
if
(
hostPlatform
==
HostPlatform
.
mac
)
{
configs
.
add
(
new
BuildConfiguration
.
prebuilt
(
hostPlatform:
HostPlatform
.
mac
,
targetPlatform:
TargetPlatform
.
iOS
));
configs
.
add
(
new
BuildConfiguration
.
prebuilt
(
hostPlatform:
HostPlatform
.
mac
,
targetPlatform:
TargetPlatform
.
iOSSimulator
));
}
}
else
{
}
else
{
if
(!
FileSystemEntity
.
isDirectorySync
(
enginePath
))
if
(!
FileSystemEntity
.
isDirectorySync
(
enginePath
))
logging
.
warning
(
'
$enginePath
is not a valid directory'
);
logging
.
warning
(
'
$enginePath
is not a valid directory'
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment