项目地址:https://github.com/tianzhilan0/OCFlutter
注:项目比较大600多兆
最终效果图如下
data:image/s3,"s3://crabby-images/d330d/d330da244ee46ee6b50ed0471f191b96c3a5c4a7" alt=""
1、创建OC项目
data:image/s3,"s3://crabby-images/0ba37/0ba376eaf0321a84a6790e46910ce989a4b3ba38" alt=""
data:image/s3,"s3://crabby-images/bcde0/bcde00c417a5c5f98d4b95af622a039a388e619a" alt=""
2、添加cocoapods
cd /Users/lc/Desktop/Test/OCFlutter
pod init
pod install
data:image/s3,"s3://crabby-images/df2c6/df2c6ada8d70234bda2edc20e94fabb1ae8319c7" alt=""
3、创建Flutter 工程
在OCFlutter项目的相同路径上创建module
cd /Users/lc/Desktop/Test
flutter create -t module flutter_module
data:image/s3,"s3://crabby-images/0a7a2/0a7a2f9e4254b01f98846839c97b1c7e08f0aba7" alt=""
4、在OCFlutter项目新建Config目录,在这个目录下新建3个配置文件
-
Flutter.xccofig,内容如下:
#include "../flutter_module/.ios/Flutter/Generated.xcconfig"
ENABLE_BITCODE=NO
-
Debug.xccofig,内容如下:
#include "Flutter.xcconfig"
#include "Pods/Target Support Files/Pods-iOS项目名称/Pods-iOS项目名称.debug.xcconfig"
-
Release.xccofig,内容如下:
#include "Flutter.xcconfig"
#include "Pods/Target Support Files/Pods-iOS项目名称/Pods-iOS项目名称.debug.xcconfig"
FLUTTER_BUILD_MODE=release
data:image/s3,"s3://crabby-images/b585c/b585c02718248f07c93dec44dcb30289d839820d" alt=""
data:image/s3,"s3://crabby-images/7e7d0/7e7d0f3d7635c8c333302158bf5daec6c2758c38" alt=""
data:image/s3,"s3://crabby-images/3e4d1/3e4d18895dec53e7fcf51aaab55fb9b065512bbd" alt=""
data:image/s3,"s3://crabby-images/9a171/9a171c2a1c67e593844cc0490e02c51e1c0f410b" alt=""
5、修改OCFlutter配置
-
关闭Bitcode
data:image/s3,"s3://crabby-images/14cbf/14cbf8769960086c11aba58fc27afdc9d01a3673" alt=""
-
修改运行环境
data:image/s3,"s3://crabby-images/db328/db328119075b8c9e1d91cb389ddd3a9f008af706" alt=""
-
添加Run Script
"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh" build
"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh" embed
data:image/s3,"s3://crabby-images/2b2a4/2b2a4b0bc9d3025aecaf46639b2da8050557a3fa" alt=""
data:image/s3,"s3://crabby-images/fde73/fde73549f92dca6165f9661e6db87842ac5fb09b" alt=""
-
修改Podfile内容,添加如下内容,添加完成之后再执行一次
pod install
flutter_application_path = '../flutter_module'
eval(File.read(File.join(flutter_application_path, '.ios', 'Flutter', 'podhelper.rb')), binding)
data:image/s3,"s3://crabby-images/3a3e4/3a3e4076e9f28a662244ff28b9d95a5a49b359d7" alt=""
data:image/s3,"s3://crabby-images/3080e/3080ed85a882f5ee300a8b9d0fbd37971bbb0bb6" alt=""
6、运行一下flutter_module,生成一些文件,运行成功之后,关闭就行
data:image/s3,"s3://crabby-images/4ca5d/4ca5ddaa3835cb0ca1e51d3c0cca81535d6d1bc5" alt=""
7、在OCFlutter项目下添加Flutter
目录
-
command+shift+。
显示隐藏文件
data:image/s3,"s3://crabby-images/1811d/1811da719b503f82160d9933434d6055e63da757" alt=""
-
在Flutter文件下添加
flutter_assets
将此处的flutter_assets
拷贝至Flutter
目录下,导入到项目
data:image/s3,"s3://crabby-images/8c24d/8c24d7842eb7c21daf4c5776af396bdd8361b9ae" alt=""
data:image/s3,"s3://crabby-images/d30d6/d30d6753677e0c589091d937fd9c0ce300771a61" alt=""
-
在Flutter文件下添加
App.framework
将此处的App.framework
拷贝至Flutter
目录下,导入到项目
data:image/s3,"s3://crabby-images/8e493/8e493e59e66bf75a2d0d2bc573cde9d6a57f924b" alt=""
data:image/s3,"s3://crabby-images/b127b/b127b3d95e71a9b4650963257e24baf76e1d19ed" alt=""
-
在Flutter文件下添加
engine
将此处的engine
拷贝至Flutter
目录下,导入到项目
data:image/s3,"s3://crabby-images/9a9cd/9a9cd6dafaba09a5a667d15a3df85e59b3aafe64" alt=""
data:image/s3,"s3://crabby-images/b127b/b127b3d95e71a9b4650963257e24baf76e1d19ed" alt=""
-
最终显示如下样式
data:image/s3,"s3://crabby-images/1f88c/1f88c35a4fae9df76f5b6636408c6e128401aa09" alt=""
在此处编译一下OCFlutter项目,如不报错,则已准备完成,接下来就是撸代码了。
8、撸代码
-
修改
AppDelegate.h
代码
#import <UIKit/UIKit.h>
#import <Flutter/Flutter.h>
@interface AppDelegate : FlutterAppDelegate <UIApplicationDelegate, FlutterAppLifeCycleProvider>
@property (strong, nonatomic) UIWindow *window;
@end
-
修改
AppDelegate.m
代码
#import "AppDelegate.h"
@interface AppDelegate ()
@end
@implementation AppDelegate
{
FlutterPluginAppLifeCycleDelegate *_lifeCycleDelegate;
}
- (instancetype)init {
if (self = [super init]) {
_lifeCycleDelegate = [[FlutterPluginAppLifeCycleDelegate alloc] init];
}
return self;
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// Override point for customization after application launch.
return [_lifeCycleDelegate application:application didFinishLaunchingWithOptions:launchOptions];
}
- (void)application:(UIApplication*)application
didRegisterUserNotificationSettings:(UIUserNotificationSettings*)notificationSettings {
[_lifeCycleDelegate application:application
didRegisterUserNotificationSettings:notificationSettings];
}
- (void)application:(UIApplication*)application
didRegisterForRemoteNotificationsWithDeviceToken:(NSData*)deviceToken {
[_lifeCycleDelegate application:application
didRegisterForRemoteNotificationsWithDeviceToken:deviceToken];
}
- (void)application:(UIApplication*)application
didReceiveRemoteNotification:(NSDictionary*)userInfo
fetchCompletionHandler:(void (^)(UIBackgroundFetchResult result))completionHandler {
[_lifeCycleDelegate application:application
didReceiveRemoteNotification:userInfo
fetchCompletionHandler:completionHandler];
}
- (BOOL)application:(UIApplication*)application
openURL:(NSURL*)url
options:(NSDictionary<UIApplicationOpenURLOptionsKey, id>*)options {
return [_lifeCycleDelegate application:application openURL:url options:options];
}
- (BOOL)application:(UIApplication*)application handleOpenURL:(NSURL*)url {
return [_lifeCycleDelegate application:application handleOpenURL:url];
}
- (BOOL)application:(UIApplication*)application
openURL:(NSURL*)url
sourceApplication:(NSString*)sourceApplication
annotation:(id)annotation {
return [_lifeCycleDelegate application:application
openURL:url
sourceApplication:sourceApplication
annotation:annotation];
}
- (void)application:(UIApplication*)application
performActionForShortcutItem:(UIApplicationShortcutItem*)shortcutItem
completionHandler:(void (^)(BOOL succeeded))completionHandler NS_AVAILABLE_IOS(9_0) {
[_lifeCycleDelegate application:application
performActionForShortcutItem:shortcutItem
completionHandler:completionHandler];
}
- (void)application:(UIApplication*)application
handleEventsForBackgroundURLSession:(nonnull NSString*)identifier
completionHandler:(nonnull void (^)(void))completionHandler {
[_lifeCycleDelegate application:application
handleEventsForBackgroundURLSession:identifier
completionHandler:completionHandler];
}
- (void)application:(UIApplication*)application
performFetchWithCompletionHandler:(void (^)(UIBackgroundFetchResult result))completionHandler {
[_lifeCycleDelegate application:application performFetchWithCompletionHandler:completionHandler];
}
- (void)addApplicationLifeCycleDelegate:(NSObject<FlutterPlugin>*)delegate {
[_lifeCycleDelegate addDelegate:delegate];
}
#pragma mark - UISceneSession lifecycle
- (UISceneConfiguration *)application:(UIApplication *)application configurationForConnectingSceneSession:(UISceneSession *)connectingSceneSession options:(UISceneConnectionOptions *)options API_AVAILABLE(ios(13.0)){
// Called when a new scene session is being created.
// Use this method to select a configuration to create the new scene with.
return [[UISceneConfiguration alloc] initWithName:@"Default Configuration" sessionRole:connectingSceneSession.role];
}
- (void)application:(UIApplication *)application didDiscardSceneSessions:(NSSet<UISceneSession *> *)sceneSessions API_AVAILABLE(ios(13.0)){
// Called when the user discards a scene session.
// If any sessions were discarded while the application was not running, this will be called shortly after application:didFinishLaunchingWithOptions.
// Use this method to release any resources that were specific to the discarded scenes, as they will not return.
}
@end
-
ViewController.m
中代码如下
#import "ViewController.h"
#import <Flutter/FlutterViewController.h>
#import "LoginViewController.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
}
- (IBAction)jumpToFlutter:(id)sender {
FlutterViewController* flutterViewController = [[FlutterViewController alloc]init];
[flutterViewController setInitialRoute:@"pag1"];
FlutterMethodChannel *channel = [FlutterMethodChannel methodChannelWithName:@"com.pages.your/native_get" binaryMessenger:(NSObject<FlutterBinaryMessenger> *)flutterViewController];
[channel setMethodCallHandler:^(FlutterMethodCall * _Nonnull call, FlutterResult _Nonnull result) {
NSString *method = [call method];
if ([method isEqualToString:@"FlutterPopIOS"]) {
NSLog(@"FlutterPopIOS:获取Flutter返回时传的值 == %@",[call arguments]);
[self.navigationController popViewControllerAnimated:true];
}
if ([method isEqualToString:@"FlutterCickedActionPushIOSNewVC"]) {
NSLog(@"FlutterCickedActionPushIOSNewVC:返回来传的参数是 == %@",[call arguments]);
LoginViewController *loginCV = [[LoginViewController alloc] init];
[self.navigationController pushViewController:loginCV animated:true];
}
if ([method isEqualToString:@"FlutterGetIOSArguments"]) {
NSDictionary *dic = @{@"name":@"名字"};
result(dic);
}
}];
[self.navigationController pushViewController:flutterViewController animated:YES];
}
@end
-
flutter_module
中main.dart
代码如下
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(title: 'Flutter Demo Home Page'),
);
}
}
class MyHomePage extends StatefulWidget {
MyHomePage({Key key, this.title}) : super(key: key);
final String title;
@override
_MyHomePageState createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
// 创建一个给native的channel (类似iOS的通知)
static const methodChannel = const MethodChannel('com.pages.your/native_get');
String _textString = "null";
void _backAction() {
_iOSPushToVC();
}
_iOSPushToVC() async {
await methodChannel.invokeMethod('FlutterPopIOS', '我是返回数据');
}
void _pushIOSNewVC() async {
Map<String, dynamic> map = {
"code": "200",
"data": [1, 2, 3]
};
await methodChannel.invokeMethod('FlutterCickedActionPushIOSNewVC', map);
}
// 给客户端发送一些东东 , 并且拿到一些东东
Future<Null> _flutterGetIOSArguments(para) async {
dynamic result;
try {
result = await methodChannel.invokeMethod('FlutterGetIOSArguments', para);
_textString = result["name"];
} on PlatformException {
result = 100000;
}
setState(() {});
}
@override
Widget build(BuildContext context) {
return new Scaffold(
body: new Center(
child: new Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
RaisedButton(
onPressed: _backAction,
child: Text("返回原生"),
),
RaisedButton(
onPressed: _pushIOSNewVC,
child: Text("跳转进入一个新的原生页面"),
),
RaisedButton(
onPressed: () {
_flutterGetIOSArguments("flutter传值");
},
child: Text("从原生页面获取数据"),
),
Text("从原生获取的内容:$_textString"),
],
),
),
);
}
}
网友评论