把flutter作为framework添加到已存在的iOS中

文章目录

之前写了一篇关于如何将 flutter 直接打包成 android aar 的文章, 本篇写一写如何将 flutter 打包成 framework 以便于直接让没有 flutter 环境的 iOS 开发者使用, 因为国内很多项目都有这样的要求

本篇并不会做完全的工程集成化, 只是做一下如何将 flutter 的 framework 打出来, 并且置入到 iOS 原生工程中, 因为各项目一定户会有自己的特殊性, 不可能完全一样

本篇打包脚本部分参考了 https://www.jianshu.com/p/700bd7d2122b 的内容,但是又有一些针对 flutter 版本的变化和 flutter type 不同的情况进行的修改, 不观看连接中的内容并不会影响观看

开发环境

MacOS
XCode 10
git
flutter 及 flutter 的相关工具链
cocoapods

创建几个工程

iOS 原生工程

使用 xcode 创建

img

img

这个原生工程就是模拟你的原有工程

Flutter 工程

这里我只使用 flutter module 的方式, 如果你 flutter 是 app 的方式创建的,则打包脚本的内容需要根据应用结构有所调整

$ flutter create -t module flutter_module_for_ios

这次直接在里面添加一个带有原生功能的插件, 和 android 篇相同依然选择 shared_preferences 那个插件

修改 pubspec.yaml:

1dependencies:
2  shared_preferences: ^0.5.3+1

创建脚本

观察结构

打包 iOS,然后观察 build 文件夹

1flutter packages get
2flutter build ios --release --no-codesign
 1tree build/ios/Release-iphoneos
 2build/ios/Release-iphoneos
 3├── FlutterPluginRegistrant
 4│   └── libFlutterPluginRegistrant.a
 5├── Runner.app
 6│   ├── AppIcon20x20@2x.png
 7│   ├── AppIcon20x20@2x~ipad.png
 8│   ├── AppIcon20x20@3x.png
 9│   ├── AppIcon20x20~ipad.png
10│   ├── AppIcon29x29.png
11│   ├── AppIcon29x29@2x.png
12│   ├── AppIcon29x29@2x~ipad.png
13│   ├── AppIcon29x29@3x.png
14│   ├── AppIcon29x29~ipad.png
15│   ├── AppIcon40x40@2x.png
16│   ├── AppIcon40x40@2x~ipad.png
17│   ├── AppIcon40x40@3x.png
18│   ├── AppIcon40x40~ipad.png
19│   ├── AppIcon60x60@2x.png
20│   ├── AppIcon60x60@3x.png
21│   ├── AppIcon76x76@2x~ipad.png
22│   ├── AppIcon76x76~ipad.png
23│   ├── AppIcon83.5x83.5@2x~ipad.png
24│   ├── Assets.car
25│   ├── Base.lproj
26│   │   ├── LaunchScreen.storyboardc
27│   │   │   ├── 01J-lp-oVM-view-Ze5-6b-2t3.nib
28│   │   │   ├── Info.plist
29│   │   │   └── UIViewController-01J-lp-oVM.nib
30│   │   └── Main.storyboardc
31│   │       ├── BYZ-38-t0r-view-8bC-Xf-vdC.nib
32│   │       ├── Info.plist
33│   │       └── UIViewController-BYZ-38-t0r.nib
34│   ├── Debug.xcconfig
35│   ├── Flutter.xcconfig
36│   ├── Frameworks
37│   │   ├── App.framework
38│   │   │   ├── App
39│   │   │   ├── Info.plist
40│   │   │   └── flutter_assets
41│   │   │       ├── AssetManifest.json
42│   │   │       ├── FontManifest.json
43│   │   │       ├── LICENSE
44│   │   │       ├── fonts
45│   │   │       │   └── MaterialIcons-Regular.ttf
46│   │   │       └── packages
47│   │   │           └── cupertino_icons
48│   │   │               └── assets
49│   │   │                   └── CupertinoIcons.ttf
50│   │   └── Flutter.framework
51│   │       ├── Flutter
52│   │       ├── Info.plist
53│   │       └── icudtl.dat
54│   ├── Info.plist
55│   ├── PkgInfo
56│   ├── Release.xcconfig
57│   └── Runner
58├── Runner.app.dSYM
59│   └── Contents
60│       ├── Info.plist
61│       └── Resources
62│           └── DWARF
63│               └── Runner
64├── libPods-Runner.a
65└── shared_preferences
66    └── libshared_preferences.a
67
6818 directories, 46 files

发现打包出来的是.a 文件, 这里我们需要修改一下 ios 目录下的文件, 以便于打包出来 framework 文件, 因为 framework 是 apple 提供的一种打包方案, 直接将所有需要的资源包括头文件,库文件都聚集到了一起,方便引用, 而.a 文件就不一样了, 还需要包含对应的头文件

修改.ios 下的 podfile

在第一行添加这个

1use_frameworks!

然后先清除一下刚刚打包的内容 $ flutter clean

1flutter build ios --release --no-codesign
  1tree build/ios/Release-iphoneos
  2build/ios/Release-iphoneos
  3├── FlutterPluginRegistrant
  4│   ├── FlutterPluginRegistrant.framework
  5│   │   ├── FlutterPluginRegistrant
  6│   │   ├── Headers
  7│   │   │   ├── FlutterPluginRegistrant-umbrella.h
  8│   │   │   └── GeneratedPluginRegistrant.h
  9│   │   ├── Info.plist
 10│   │   └── Modules
 11│   │       └── module.modulemap
 12│   └── FlutterPluginRegistrant.framework.dSYM
 13│       └── Contents
 14│           ├── Info.plist
 15│           └── Resources
 16│               └── DWARF
 17│                   └── FlutterPluginRegistrant
 18├── Pods_Runner.framework
 19│   ├── Headers
 20│   │   └── Pods-Runner-umbrella.h
 21│   ├── Info.plist
 22│   ├── Modules
 23│   │   └── module.modulemap
 24│   └── Pods_Runner
 25├── Runner.app
 26│   ├── AppIcon20x20@2x.png
 27│   ├── AppIcon20x20@2x~ipad.png
 28│   ├── AppIcon20x20@3x.png
 29│   ├── AppIcon20x20~ipad.png
 30│   ├── AppIcon29x29.png
 31│   ├── AppIcon29x29@2x.png
 32│   ├── AppIcon29x29@2x~ipad.png
 33│   ├── AppIcon29x29@3x.png
 34│   ├── AppIcon29x29~ipad.png
 35│   ├── AppIcon40x40@2x.png
 36│   ├── AppIcon40x40@2x~ipad.png
 37│   ├── AppIcon40x40@3x.png
 38│   ├── AppIcon40x40~ipad.png
 39│   ├── AppIcon60x60@2x.png
 40│   ├── AppIcon60x60@3x.png
 41│   ├── AppIcon76x76@2x~ipad.png
 42│   ├── AppIcon76x76~ipad.png
 43│   ├── AppIcon83.5x83.5@2x~ipad.png
 44│   ├── Assets.car
 45│   ├── Base.lproj
 46│   │   ├── LaunchScreen.storyboardc
 47│   │   │   ├── 01J-lp-oVM-view-Ze5-6b-2t3.nib
 48│   │   │   ├── Info.plist
 49│   │   │   └── UIViewController-01J-lp-oVM.nib
 50│   │   └── Main.storyboardc
 51│   │       ├── BYZ-38-t0r-view-8bC-Xf-vdC.nib
 52│   │       ├── Info.plist
 53│   │       └── UIViewController-BYZ-38-t0r.nib
 54│   ├── Debug.xcconfig
 55│   ├── Flutter.xcconfig
 56│   ├── Frameworks
 57│   │   ├── App.framework
 58│   │   │   ├── App
 59│   │   │   ├── Info.plist
 60│   │   │   └── flutter_assets
 61│   │   │       ├── AssetManifest.json
 62│   │   │       ├── FontManifest.json
 63│   │   │       ├── LICENSE
 64│   │   │       ├── fonts
 65│   │   │       │   └── MaterialIcons-Regular.ttf
 66│   │   │       └── packages
 67│   │   │           └── cupertino_icons
 68│   │   │               └── assets
 69│   │   │                   └── CupertinoIcons.ttf
 70│   │   ├── Flutter.framework
 71│   │   │   ├── Flutter
 72│   │   │   ├── Info.plist
 73│   │   │   └── icudtl.dat
 74│   │   ├── FlutterPluginRegistrant.framework
 75│   │   │   ├── FlutterPluginRegistrant
 76│   │   │   └── Info.plist
 77│   │   └── shared_preferences.framework
 78│   │       ├── Info.plist
 79│   │       └── shared_preferences
 80│   ├── Info.plist
 81│   ├── PkgInfo
 82│   ├── Release.xcconfig
 83│   └── Runner
 84├── Runner.app.dSYM
 85│   └── Contents
 86│       ├── Info.plist
 87│       └── Resources
 88│           └── DWARF
 89│               └── Runner
 90└── shared_preferences
 91    ├── shared_preferences.framework
 92    │   ├── Headers
 93    │   │   ├── SharedPreferencesPlugin.h
 94    │   │   └── shared_preferences-umbrella.h
 95    │   ├── Info.plist
 96    │   ├── Modules
 97    │   │   └── module.modulemap
 98    │   └── shared_preferences
 99    └── shared_preferences.framework.dSYM
100        └── Contents
101            ├── Info.plist
102            └── Resources
103                └── DWARF
104                    └── shared_preferences
105
10637 directories, 65 files

Runner.app 这个熟悉的名字下虽然看似包含所有的 framework,但是仔细观察, 这些 framework 内不包含头文件, 所以其实是用不了的

可以使用的是如图所示的几个

img

根据上面的 tree 图可以看到,其中包含了全部的库文件, 但除此以外,还需要两个库:

  1. App.framework: 这个包含了 flutter 的资源
  2. Flutter.framework: 这个是 Flutter 的 engine 运行库, 每个 flutter 应用必然有的东西

这两个东西需要从.ios 中找到

  1. App.framework: .ios/Flutter/App.framework
  2. Flutter.framework: .ios/Flutter/engine/Flutter.framework

这样就集齐了所有的库文件, 问题是, 这样是手动做的, 我们需要一个自动化的方案把重复的工作都完成

所以我根据开头的文章内的脚本编写了一个脚本:

 1if [ -z $out ]; then
 2    out='ios_frameworks'
 3fi
 4
 5echo "准备输出所有文件到目录: $out"
 6
 7find . -d -name build | xargs rm -rf
 8flutter clean
 9rm -rf $out
10rm -rf build
11
12flutter packages get
13flutter build ios --release --no-codesign
14
15mkdir $out
16
17cp -r build/ios/Release-iphoneos/*/*.framework $out
18cp -r .ios/Flutter/App.framework $out
19cp -r .ios/Flutter/engine/Flutter.framework $out

接着可以通过sh build_ios.sh 来自动打包,并且将所有文件复制到 ios_frameword 目录下, 嗯 这个目录可以通过 out=ios_out sh build_ios.sh来改变

关联库文件和工程

原生工程的管理方式有很多种, 比如只使用 xcode 来管理

cocoapod(支持 oc 和 swift)

单独针对 swift 还有 Carthage, Swift Package Manager

当然这里我还是用 cocoapod 的方案, 其他的方案请自行研究

使用 cocoapod 管理原生工程

为原生工程添加一个 Podfile

1cd top.kikt.existsapp
2touch Podfile
3code Podfile # 这一步是用 vscode 打开 你也可以用 vim 或其他任何文本编辑器

修改 Podfile

1platform :ios, '8.0'
2use_frameworks!
3
4target 'top.kikt.existsapp' do
5
6end

然后继续在命令行里 $ pod install

如果没有报错, 则说明原生项目现在已经被 pod 管理了

将 framework 作为一个 pod 库

因为找遍了 podfile 的相关文档, 没有找到可以直接引用 framework 的方式

所以需要一个 pod 库作为"中转"

新建一个库 $ pod lib create flutter-lib, 然后按顺序回答问题, 这目录根据你自己的实际情况来

我的是和 iOS 原生同级目录

 1$ pod lib create flutter-lib
 2Cloning `https://github.com/CocoaPods/pod-template.git` into `flutter-lib`.
 3Configuring flutter-lib template.
 4
 5------------------------------
 6
 7To get you started we need to ask a few questions, this should only take a minute.
 8
 9If this is your first time we recommend running through with the guide:
10 - https://guides.cocoapods.org/making/using-pod-lib-create.html
11 ( hold cmd and double click links to open in a browser. )
12
13
14What platform do you want to use?? [ iOS / macOS ]
15 >
16ios
17What language do you want to use?? [ Swift / ObjC ]
18 > objc
19
20Would you like to include a demo application with your library? [ Yes / No ]
21 > no
22
23Which testing frameworks will you use? [ Specta / Kiwi / None ]
24 > none
25
26Would you like to do view based testing? [ Yes / No ]
27 > no
28
29What is your class prefix?
30 >
31
32You need to provide an answer.
33What is your class prefix?
34 > FFF
35
36Running pod install on your new library.
37
38Analyzing dependencies
39Fetching podspec for `flutter-lib` from `../`
40Downloading dependencies
41Installing flutter-lib (0.1.0)
42Generating Pods project
43Integrating client project
44
45[!] Please close any current Xcode sessions and use `flutter-lib.xcworkspace` for this project from now on.
46Sending stats
47Pod installation complete! There is 1 dependency from the Podfile and 1 total pod installed.
48
49[!] Automatically assigning platform `ios` with version `9.3` on target `flutter-lib_Tests` because no platform was specified. Please specify a platform for this target in your Podfile. See `https://guides.cocoapods.org/syntax/podfile.html#platform`.
50
51 Ace! you're ready to go!
52 We will start you off by opening your project in Xcode
53  open 'flutter-lib/Example/flutter-lib.xcworkspace'
54
55To learn more about the template see `https://github.com/CocoaPods/pod-template.git`.
56To learn more about creating a new pod, see `https://guides.cocoapods.org/making/making-a-cocoapod`.

创建完成后, 修改 flutter-lib.podspec 文件

这里需要先把刚刚的 framework 复制到项目中

默认有一堆注释都可以删掉

然后添加这么一行

1s.ios.vendored_frameworks = 'ios_frameworks/App.framework', 'ios_frameworks/Flutter.framework', 'ios_frameworks/FlutterPluginRegistrant.framework', 'ios_frameworks/shared_preferences.framework'

这个方案是临时的, 后面肯定会写成动态查找的方案, 不可能有一个库写一次

现在我的 podspec 文件是这样的

 1Pod::Spec.new do |s|
 2  s.name             = 'flutter-lib'
 3  s.version          = '0.1.0'
 4  s.summary          = 'A short description of flutter-lib.'
 5  s.description      = <<-DESC
 6TODO: Add long description of the pod here.
 7                       DESC
 8
 9  s.homepage         = 'https://github.com/cjl_spy@163.com/flutter-lib'
10  s.license          = { :type => 'MIT', :file => 'LICENSE' }
11  s.author           = { 'cjl_spy@163.com' => 'cjl_spy@163.com' }
12  s.source           = { :git => 'https://github.com/cjl_spy@163.com/flutter-lib.git', :tag => s.version.to_s }
13  s.ios.deployment_target = '8.0'
14  s.static_framework = true
15  # s.source_files = 'flutter-lib/Classes/**/*'
16  s.ios.vendored_frameworks = 'ios_frameworks/App.framework', 'ios_frameworks/Flutter.framework', 'ios_frameworks/FlutterPluginRegistrant.framework', 'ios_frameworks/shared_preferences.framework'
17end

在原生项目中引用这个库

1platform :ios, '8.0'
2use_frameworks!
3
4target 'top.kikt.existsapp' do
5   pod 'flutter-lib', :path => '../flutter-lib'
6end

接着在原生项目中$ pod install

运行项目可能会有两种错误:

信息包含什么 System 之类的, 这个是需要修改构建系统

img

运行项目还可能会报一个 bitcode 的错误, 这里需要修改一个选项

img

将这里修改为 NO

我的项目目前为止能跑起来不报错了, 但是目前项目还没有启动 Flutter 的页面, 到这一步后我是按照 flutter 官方 wiki 进行修改的

我给的例子仅有 oc 的,swift 的请自行参考 wiki

修改AppDelegate.h

1#import <UIKit/UIKit.h>
2#import <Flutter/Flutter.h>
3
4@interface AppDelegate : FlutterAppDelegate
5@property (nonatomic,strong) FlutterEngine *flutterEngine;
6@end

AppDelegate.m

 1#import <FlutterPluginRegistrant/GeneratedPluginRegistrant.h> // Only if you have Flutter Plugins
 2
 3#include "AppDelegate.h"
 4
 5@implementation AppDelegate
 6
 7// This override can be omitted if you do not have any Flutter Plugins.
 8- (BOOL)application:(UIApplication *)application
 9    didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
10  self.flutterEngine = [[FlutterEngine alloc] initWithName:@"io.flutter" project:nil];
11  [self.flutterEngine runWithEntrypoint:nil];
12  [GeneratedPluginRegistrant registerWithRegistry:self.flutterEngine];
13  return [super application:application didFinishLaunchingWithOptions:launchOptions];
14}
15
16@end

接着就是启动 Flutter 的事情了, 这个根据你业务逻辑自行实现, 核心代码是如下几行

 1#import <Flutter/Flutter.h>
 2#import "AppDelegate.h"
 3#import "ViewController.h"
 4
 5@implementation ViewController
 6- (void)viewDidLoad {
 7    [super viewDidLoad];
 8    UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
 9    [button addTarget:self
10               action:@selector(handleButtonAction)
11     forControlEvents:UIControlEventTouchUpInside];
12    [button setTitle:@"Press me" forState:UIControlStateNormal];
13    [button setBackgroundColor:[UIColor blueColor]];
14    button.frame = CGRectMake(80.0, 210.0, 160.0, 40.0);
15    [self.view addSubview:button];
16}
17
18- (void)handleButtonAction {
19    FlutterEngine *flutterEngine = [(AppDelegate *)[[UIApplication sharedApplication] delegate] flutterEngine];
20    FlutterViewController *flutterViewController = [[FlutterViewController alloc] initWithEngine:flutterEngine nibName:nil bundle:nil];
21    [self presentViewController:flutterViewController animated:false completion:nil];
22}
23@end

优化脚本

优化 podspec 文件

这一步的目的是, 遍历文件夹, 不再单独的每个引用 framework 文件

podspec 使用的是 ruby 语法, 也就是说本身就支持编程, 这时候如果再用 shell 或者 dart 就有点增加复杂度了

如果不了解 ruby 语法, 你也可以用自己的方法解决(shell,python,dart 什么都随你), 我也是边写边看的 ruby 语法, 按照如下方式修改

 1Pod::Spec.new do |s|
 2  s.name             = 'flutter-lib'
 3  s.version          = '0.1.0'
 4  s.summary          = 'A short description of flutter-lib.'
 5  s.description      = <<-DESC
 6TODO: Add long description of the pod here.
 7                       DESC
 8
 9  s.homepage         = 'https://github.com/cjl_spy@163.com/flutter-lib'
10  s.license          = { :type => 'MIT', :file => 'LICENSE' }
11  s.author           = { 'cjl_spy@163.com' => 'cjl_spy@163.com' }
12  s.source           = { :git => 'https://github.com/cjl_spy@163.com/flutter-lib.git', :tag => s.version.to_s }
13  s.ios.deployment_target = '8.0'
14  s.static_framework = true
15  # s.source_files = 'flutter-lib/Classes/**/*'
16  p = Dir::open("ios_frameworks")
17  arr = Array.new
18  p.each do |f|
19    if f == '.' || f == '..'
20    else
21        arr.push('ios_frameworks/'+f)
22    end
23  end
24
25  s.ios.vendored_frameworks = arr
26end

然后执行 pod install

优化 shell 脚本

 1if [ -z $out ]; then
 2    out='ios_frameworks'
 3fi
 4
 5echo "准备输出所有文件到目录: $out"
 6
 7find . -d -name build | xargs rm -rf
 8flutter clean
 9rm -rf $out
10rm -rf build
11
12flutter packages get
13flutter build ios --release --no-codesign
14
15mkdir $out
16
17cp -r build/ios/Release-iphoneos/*/*.framework $out
18cp -r .ios/Flutter/App.framework $out
19cp -r .ios/Flutter/engine/Flutter.framework $out
20
21cp -r $out ../flutter-lib/ios_frameworks # 添加这步

验证下别的插件

添加插件到 pubspec.yaml

1dependencies:
2  path_provider: ^1.1.0

$ flutter pub get

这里有个小坑,当添加了 pubspec 插件后, flutter/.ios/Podfile 的内容会被恢复成默认状态

所以需要再修改一下 shell 脚本:

 1if [ -z $out ]; then
 2    out='ios_frameworks'
 3fi
 4
 5echo "准备输出所有文件到目录: $out"
 6
 7echo "清除所有已编译文件"
 8find . -d -name build | xargs rm -rf
 9flutter clean
10rm -rf $out
11rm -rf build
12
13flutter packages get
14
15addFlag(){
16    cat .ios/Podfile > tmp1.txt
17    echo "use_frameworks!" >> tmp2.txt
18    cat tmp1.txt >> tmp2.txt
19    cat tmp2.txt > .ios/Podfile
20    rm tmp1.txt tmp2.txt
21}
22
23echo "检查 .ios/Podfile文件状态"
24a=$(cat .ios/Podfile)
25if [[ $a == use* ]]; then
26    echo '已经添加use_frameworks, 不再添加'
27else
28    echo '未添加use_frameworks,准备添加'
29    addFlag
30    echo "添加use_frameworks 完成"
31fi
32
33echo "编译flutter"
34flutter build ios --release --no-codesign
35
36echo "编译flutter完成"
37mkdir $out
38
39cp -r build/ios/Release-iphoneos/*/*.framework $out
40cp -r .ios/Flutter/App.framework $out
41cp -r .ios/Flutter/engine/Flutter.framework $out
42
43echo "复制framework库到临时文件夹: $out"
44
45libpath='../flutter-lib'
46
47rm -rf "$libpath/ios_frameworks"
48mkdir $libpath
49cp -r $out $libpath
50
51echo "复制库文件到: $libpath"

顺便润色了一下,加了一点日志

运行下项目, 跑了一下, 果然没有问题了

这里有一个插曲请注意,我脚本里用了 gsed 命令,这个工具是 GNU-sed 的缩写, 因为 mac 的 sed 命令有一些问题比较难用
请使用$ brew install gnu-sed安装一下, 否则这个命令可能失败

备注: 因为有反馈, gsed 在不同的 mac 上会出现把文本变成一行的问题, 现在采用另一种方案,不再需要 gsed

总结

建议: 请将 flutter,原生和 flutter 的 framework 所在的 pod 库作为 3 个单独的库,便于管理(有 pod 的私服请按照自己的方案实现 framework 的共享)

然后脚本中的目录什么的需要根据你的实际情况进行调整

打包的使用总结

flutter 方:

  1. 创建一个 pod 库
  2. 使用脚本打包并把所有的文件复制到 pod 库内
  3. git commit + push

ios 方:

  1. git pull
  2. pod install
  3. run

后记

到目前为止, iOS 也算是完成了加入到已有工程, 原生方只使用 pod+git 理论上就达到了可以不用 flutter 依然可运行的目的

仓库看这里: https://github.com/CaiJingLong/add_flutter_to_exists_ios_example

以上