build_apk.dart 4.35 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
2 3 4
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5
import '../android/android_builder.dart';
6
import '../android/build_validation.dart';
7
import '../android/gradle_utils.dart';
8
import '../build_info.dart';
9
import '../cache.dart';
10
import '../globals.dart' as globals;
11
import '../project.dart';
12
import '../reporting/reporting.dart';
13
import '../runner/flutter_command.dart' show FlutterCommandResult;
14
import 'build.dart';
15

16
class BuildApkCommand extends BuildSubCommand {
17 18 19
  BuildApkCommand({
    required super.logger, bool verboseHelp = false
  }) : super(verboseHelp: verboseHelp) {
20
    addTreeShakeIconsFlag();
21
    usesTargetOption();
22
    addBuildModeFlags(verboseHelp: verboseHelp);
23
    usesFlavorOption();
24
    usesPubOption();
25 26
    usesBuildNumberOption();
    usesBuildNameOption();
27
    addShrinkingFlag(verboseHelp: verboseHelp);
28
    addSplitDebugInfoOption();
29
    addDartObfuscationOption();
30
    usesDartDefineOption();
31
    usesExtraDartFlagOptions(verboseHelp: verboseHelp);
32
    addBundleSkSLPathOption(hide: !verboseHelp);
33
    addEnableExperimentation(hide: !verboseHelp);
34
    addBuildPerformanceFile(hide: !verboseHelp);
35
    addNullSafetyModeOptions(hide: !verboseHelp);
36
    usesAnalyzeSizeFlag();
37
    addAndroidSpecificBuildOptions(hide: !verboseHelp);
38
    addMultidexOption();
39
    addIgnoreDeprecationOption();
40
    argParser
41
      ..addFlag('split-per-abi',
42
        negatable: false,
43
        help: 'Whether to split the APKs per ABIs. '
44
              'To learn more, see: https://developer.android.com/studio/build/configure-apk-splits#configure-abi-split',
45
      )
46
      ..addMultiOption('target-platform',
47
        defaultsTo: <String>['android-arm', 'android-arm64', 'android-x64'],
48 49 50
        allowed: <String>['android-arm', 'android-arm64', 'android-x86', 'android-x64'],
        help: 'The target platform for which the app is compiled.',
      );
51
    usesTrackWidgetCreation(verboseHelp: verboseHelp);
52 53
  }

54 55 56
  @override
  final String name = 'apk';

57
  @override
58
  DeprecationBehavior get deprecationBehavior => boolArgDeprecated('ignore-deprecation') ? DeprecationBehavior.ignore : DeprecationBehavior.exit;
59

60 61 62 63 64
  @override
  Future<Set<DevelopmentArtifact>> get requiredArtifacts async => <DevelopmentArtifact>{
    DevelopmentArtifact.androidGenSnapshot,
  };

65
  @override
66
  final String description = 'Build an Android APK file from your app.\n\n'
67 68
    "This command can build debug and release versions of your application. 'debug' builds support "
    "debugging and a quick development cycle. 'release' builds don't support debugging and are "
69
    'suitable for deploying to app stores. If you are deploying the app to the Play Store, '
70
    "it's recommended to use app bundles or split the APK to reduce the APK size. Learn more at:\n\n"
71 72
    ' * https://developer.android.com/guide/app-bundle\n'
    ' * https://developer.android.com/studio/build/configure-apk-splits#configure-abi-split';
73

74
  @override
75 76
  Future<CustomDimensions> get usageValues async {
    String buildMode;
77

78
    if (boolArgDeprecated('release')) {
79
      buildMode = 'release';
80
    } else if (boolArgDeprecated('debug')) {
81
      buildMode = 'debug';
82
    } else if (boolArgDeprecated('profile')) {
83
      buildMode = 'profile';
84 85
    } else {
      // The build defaults to release.
86
      buildMode = 'release';
87
    }
88 89 90 91

    return CustomDimensions(
      commandBuildApkTargetPlatform: stringsArg('target-platform').join(','),
      commandBuildApkBuildMode: buildMode,
92
      commandBuildApkSplitPerAbi: boolArgDeprecated('split-per-abi'),
93
    );
94 95
  }

96
  @override
97
  Future<FlutterCommandResult> runCommand() async {
98
    if (globals.androidSdk == null) {
99 100
      exitWithNoSdkMessage();
    }
101
    final BuildInfo buildInfo = await getBuildInfo();
Emmanuel Garcia's avatar
Emmanuel Garcia committed
102 103
    final AndroidBuildInfo androidBuildInfo = AndroidBuildInfo(
      buildInfo,
104
      splitPerAbi: boolArgDeprecated('split-per-abi'),
105
      targetArchs: stringsArg('target-platform').map<AndroidArch>(getAndroidArchForName),
106
      multidexEnabled: boolArgDeprecated('multidex'),
107
    );
108
    validateBuild(androidBuildInfo);
109
    displayNullSafetyMode(androidBuildInfo.buildInfo);
110
    globals.terminal.usesTerminalUi = true;
111
    await androidBuilder?.buildApk(
112
      project: FlutterProject.current(),
113
      target: targetFile,
114
      androidBuildInfo: androidBuildInfo,
115
    );
116
    return FlutterCommandResult.success();
117
  }
118
}