flutter_boost集成与使用(iOS )

背景

具有一定混合方案的App通常有一套通用的基础库,一般需要依赖很多基础库。使用flutter重新开发App的成本和风险都比较高。在Native App进行渐进式迁移是比较稳健的方式,所以就有了使用咸鱼flutter_boost混合解决方案。

准备工作

创建flutter_module

在使用flutter_boost之前,首先要生成flutter_module包。如果你是使用最新版的Andrider Studio新建工程直接选择flutter_module就可以创建成功,除此之外也可以使用终端命令创建:

  • 新建文件夹flutter_boost_test
  • 终端创建flutter_module, cd到刚才创建的文件夹
    cd /Users/mac/Desktop/flutter_boost_test
    执行命令flutter create -t module flutter_module

创建iOS 工程

1:使用Xcode创建iOS工程,工程名为iOSflutter,集成cocoaPods,工程位置放在刚才创建的flutter_boost_test文件夹里面,它与flutter_module是同一级别的。
2:将flutter_moduleiOSflutter关联,打开iOS工程中Podfile文件,填写如下代码:

# Uncomment the next line to define a global platform for your project
# platform :ios, '9.0'

# 配置
flutter_application_path = '../flutter_module/'
load File.join(flutter_application_path, '.ios', 'Flutter', 'podhelper.rb')

target 'iOSflutter' do
  # Comment the next line if you don't want to use dynamic frameworks
  use_frameworks!

  # Pods for iOSflutter
# 配置
  install_all_flutter_pods(flutter_application_path)

end

终端cdiOS工程,执行pod install命令,如下所示即成功:

企业微信截图_bbbf7aac-cbd5-4be2-af0a-f6fbdf3fef26.png

集成flutter_boost

使用工具打开flutter_module找到pubspec.yaml 添加依赖

dependencies:
  flutter_boost: ^1.17.1

点击查看flutter_boost具体依赖版本和介绍,get package之后,再次打开终端cdiOS工程,执行pod install命令,如图所示即成功:

企业微信截图_6bdbbfaa-5bbc-485b-8fff-5aa975148d5a.png

flutter_boost 使用

iOS端

首先创建 router类,这个可以直接按照咸鱼的demo来写


#import <Foundation/Foundation.h>
#import <FlutterBoost.h>

NS_ASSUME_NONNULL_BEGIN

//@protocol FLBPlatform;

/**
 * 实现平台侧的页面打开和关闭,不建议直接使用用于页面打开,建议使用FlutterBoostPlugin中的open和close方法来打开或关闭页面;
 * FlutterBoostPlugin带有页面返回数据的能力
 */
@interface PlatformRouterImp : NSObject<FLBPlatform>
@property (nonatomic,strong) UINavigationController *navigationController;
@end

NS_ASSUME_NONNULL_END
#import "PlatformRouterImp.h"
#import "UIViewControllerDemo.h"

@implementation PlatformRouterImp

- (void)openNativeVC:(NSString *)name urlParams:(NSDictionary *)params exts:(NSDictionary *)exts{

    UIViewControllerDemo *vc = [[UIViewControllerDemo alloc] initWithNibName:@"UIViewControllerDemo" bundle:[NSBundle mainBundle]];
    BOOL animated = [exts[@"animated"] boolValue];
    if ([params[@"present"] boolValue]) {
        [self.navigationController presentViewController:vc animated:animated completion:^{

        }];
    }else{
        [self.navigationController pushViewController:vc animated:animated];
    }
}
#pragma mark - boost 1.5
/**
 可以根据 url 进行 页面跳转
 */
- (void)open:(NSString *)url urlParams:(NSDictionary *)urlParams exts:(NSDictionary *)exts completion:(void (^)(BOOL))completion{
    
    NSLog(@"flutter跳转native的url %@",url);
    
    if ([url isEqualToString:@"native"]) {
        [self openNativeVC:url urlParams:urlParams exts:exts];
        
        NSLog(@"platformRouterImp");
        return;
    }
    BOOL animated = [exts[@"animated"] boolValue];
    FLBFlutterViewContainer *vc = FLBFlutterViewContainer.new;
    [vc setName:url params:urlParams];
    [urlParams enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
        
        NSLog(@"key:%@",key);
        NSLog(@"obj:%@",obj);
    }];
    [self.navigationController pushViewController:vc animated:animated];
    if (completion) {
        completion(YES);
    }
}
- (void)present:(NSString *)url urlParams:(NSDictionary *)urlParams exts:(NSDictionary *)exts completion:(void (^)(BOOL))completion {
    
    BOOL animated = [exts[@"animated"] boolValue];
       FLBFlutterViewContainer *vc = FLBFlutterViewContainer.new;
       [vc setName:url params:urlParams];
       [self.navigationController presentViewController:vc animated:animated completion:^{
           if(completion) completion(YES);
       }];
}

- (void)close:(NSString *)uid result:(NSDictionary *)result exts:(NSDictionary *)exts completion:(void (^)(BOOL))completion {
    
    BOOL animated = [exts[@"animated"] boolValue];
       animated = YES;
       FLBFlutterViewContainer *vc = (id)self.navigationController.presentedViewController;
       if([vc isKindOfClass:FLBFlutterViewContainer.class] && [vc.uniqueIDString isEqual: uid]){
           [vc dismissViewControllerAnimated:animated completion:^{}];
       }else{
           [self.navigationController popViewControllerAnimated:animated];
       }
}
@end

AppDelegate的里面的代码

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
    
    PlatformRouterImp *router = [PlatformRouterImp new];
    
    [FlutterBoostPlugin.sharedInstance startFlutterWithPlatform:router onStart:^(FlutterEngine * _Nonnull engine) {
        
    }];
    return YES;
}

在原生页面可以使用一下代码进行跳转

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view from its nib.
    self.navigationItem.title = @"我是原生类 UIViewControllerDemo";
}

- (IBAction)pushFlutterPage:(id)sender {
    [FlutterBoostPlugin open:@"first" urlParams:@{kPageCallBackId:@"MycallbackId#1"} exts:@{@"animated":@(YES)} onPageFinished:^(NSDictionary *result) {
        NSLog(@"call me when page finished, and your result is:%@", result);
    } completion:^(BOOL f) {
        NSLog(@"page is opened");
    }];
}

- (IBAction)present:(id)sender {
    [FlutterBoostPlugin open:@"second" urlParams:@{@"present":@(YES),kPageCallBackId:@"MycallbackId#2"} exts:@{@"animated":@(YES)} onPageFinished:^(NSDictionary *result) {
        
        
        NSLog(@"call me when page finished, and your result is:%@", result);
        
    } completion:^(BOOL f) {
        NSLog(@"page is presented");
    }];
}

flutter端:
首先在main.dart进行跳转注册代码如下:

class _MyHomePageState extends State<MyHomePage> {
  int _counter = 0;

  void _incrementCounter() {
    setState(() {
      // This call to setState tells the Flutter framework that something has
      // changed in this State, which causes it to rerun the build method below
      // so that the display can reflect the updated values. If we changed
      // _counter without calling setState(), then the build method would not be
      // called again, and so nothing would appear to happen.
      _counter++;
    });
  }
@override
  void initState() {
    // TODO: implement initState
    super.initState();
    FlutterBoost.singleton.registerPageBuilders(<String,PageBuilder>{
      'embeded': (String pageName, Map<String, dynamic> params, String _) =>
          EmbeddedFirstRouteWidget(),
      'first': (String pageName, Map<String, dynamic> params, String _) => FirstRouteWidget(),
      'firstFirst': (String pageName, Map<String, dynamic> params, String _) =>
          FirstFirstRouteWidget(),
      'second': (String pageName, Map<String, dynamic> params, String _) => SecondRouteWidget(),
      'secondStateful': (String pageName, Map<String, dynamic> params, String _) =>
          SecondStatefulRouteWidget(),
      'tab': (String pageName, Map<String, dynamic> params, String _) => TabRouteWidget(),
      'platformView': (String pageName, Map<String, dynamic> params, String _) =>
          PlatformRouteWidget(),
      'flutterFragment': (String pageName, Map<String, dynamic> params, String _) =>
          FragmentRouteWidget(params),

      ///可以在native层通过 getContainerParams 来传递参数
      'flutterPage': (String pageName, Map<String, dynamic> params, String _) {
        print('flutterPage params:$params');

        return FlutterRouteWidget(params: params);
      },
    });
    FlutterBoost.singleton.addBoostNavigatorObserver(TestBoostNavigatorObserver());
  }
  @override
  Widget build(BuildContext context) {
    // TODO: implement build
    return MaterialApp(
      title: 'Flutter boost example',
      builder: FlutterBoost.init(postPush: _onRoutePushed),
      home: Container(color: Colors.white,),
    );
  }
  void _onRoutePushed(String pageName,String uniqueId,Map<String,dynamic> params,Route<dynamic> route,Future<dynamic>_){}
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352