iOS.ReactNative-3-about-viewmanager-uimanager-and-bridgemodule
RCTViewManager and RCTUIManager
1. RCTViewManager
1.1 RCTViewManager 实现了接口RCTBridgeModule
@interface RCTViewManager : NSObject <RCTBridgeModule>
1.2 宏RCT_EXPORT_VIEW_PROPERTY
from file RCTViewManager.m
RCT_EXPORT_VIEW_PROPERTY(accessibilityLabel, NSString)
以上宏展开为:
// RCT_EXPORT_VIEW_PROPERTY(accessibilityLabel, NSString)
+ (NSString *)getPropConfigView_accessibilityLabel { return @"NSString"; }
- (void)set_accessibilityLabel:(id)json forView:(UIView *)view withDefaultView:(UIView *)defaultView
{ if ((json && !RCTSetProperty(view, @"accessibilityLabel", @selector(NSString:), json))
|| (!json && !RCTCopyProperty(view, defaultView, @"accessibilityLabel")))
{
do {
if (RCTLogLevelError >= RCTLogLevelMustFix) {
do {
BOOL pass = ((__objc_no) != );
if ( && !pass){
[[NSAssertionHandler currentHandler] handleFailureInFunction:@(__func__)
file:@("/Users/XiaoKL/react-native/AwesomeProject/node_modules/react-native/React/Views/RCTViewManager.m")
lineNumber:
description:@"%@ does not have setter for `%s` property", [view class], "accessibilityLabel"];
}
_RCTAssertFormat(pass, "/Users/XiaoKL/react-native/AwesomeProject/node_modules/react-native/React/Views/RCTViewManager.m",
, __func__,
@"%@ does not have setter for `%s` property",
[view class], "accessibilityLabel");
} while ();
}
_RCTLogFormat(RCTLogLevelError, "/Users/XiaoKL/react-native/AwesomeProject/node_modules/react-native/React/Views/RCTViewManager.m", , @"%@ does not have setter for `%s` property",
[view class], "accessibilityLabel");
} while ();
}
}
3. RCTBridgeModule接口
/**
* Provides the interface needed to register a bridge module.
*/
@protocol RCTBridgeModule <NSObject>
RCTBridgeModule: 定义注册一个“bridge module”所需要的接口。
bridge module: 实现接口RCTBridgeModule的类被称为“bridge module”, 这些模块可以被注册到 RCTBridge 中。
3.1 RCTBridgeModule.h中提供的宏定义
| 宏 | 宏定义以及描述 |
| RCT_EXPORT_MODULE(js_name) |
#define RCT_EXPORT_MODULE(js_name) \ RCT_EXTERN void RCTRegisterModule(Class); \ + (NSString *)moduleName { return @#js_name; } \ + (void)load { RCTRegisterModule([self class]); } |
| RCT_EXPORT_METHOD(method) |
#define RCT_EXPORT_METHOD(method) \ RCT_REMAP_METHOD(, method) |
| RCT_REMAP_METHOD(js_name, method) |
#define RCT_REMAP_METHOD(js_name, method) \ RCT_EXTERN_REMAP_METHOD(js_name, method) \ - (void)method |
| RCT_EXTERN_MODULE(objc_name, objc_supername) |
#define RCT_EXTERN_MODULE(objc_name, objc_supername) \ RCT_EXTERN_REMAP_MODULE(, objc_name, objc_supername) |
| RCT_EXTERN_REMAP_MODULE(js_name, objc_name, objc_supername) |
#define RCT_EXTERN_REMAP_MODULE(js_name, objc_name, objc_supername) \ objc_name : objc_supername \ @end \ @interface objc_name (RCTExternModule) <RCTBridgeModule> \ @end \ @implementation objc_name (RCTExternModule) \ RCT_EXPORT_MODULE(js_name) |
| RCT_EXTERN_METHOD(method) |
#define RCT_EXTERN_METHOD(method) \ RCT_EXTERN_REMAP_METHOD(, method) |
| RCT_EXTERN_REMAP_METHOD(js_name, method) |
#define RCT_EXTERN_REMAP_METHOD(js_name, method) \ + (NSArray *)RCT_CONCAT(__rct_export__, RCT_CONCAT(js_name, RCT_CONCAT(__LINE__, __COUNTER__))) { \ return @[@#js_name, @#method]; \ } \ |
A): RCT_EXPORT_MODULE(js_name): 将BridgeModule注册到bridge中。该宏依赖函数:
RCTRegisterModule(), 下面看一下该函数的实现: (RCTBridge.m)
void RCTRegisterModule(Class moduleClass)
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
RCTModuleClasses = [[NSMutableArray alloc] init];
}); RCTAssert([moduleClass conformsToProtocol:@protocol(RCTBridgeModule)], // 1
@"%@ does not conform to the RCTBridgeModule protocol",
NSStringFromClass(moduleClass)); // Register module
[RCTModuleClasses addObject:moduleClass];
}
RCTRegisterModule()仅仅是将Class对象添加到了一个数组类型的静态变量RCTModuleClasses中.
1: “bridge module” 必须实现接口 RCTBridgeModule 。
RCT_EXPORT_MODULE(js_name)
"The optional js_name argument will be used as the JS module name. If omitted, the JS module name will
match the Objective-C class name. "
B): RCT_EXPORT_METHOD(method) 宏
Wrap the parameter line of your method implementation with this macro to
expose it to JS.
用宏 RCT_EXPORT_METHOD() 来处理要导出到JS中的接口:
RCT_EXPORT_METHOD(createTimer:(NSNumber *)callbackID
duration:(NSTimeInterval)jsDuration
jsSchedulingTime:(NSDate *)jsSchedulingTime
repeats:(BOOL)repeats)
以上宏 RCT_EXPORT_METHOD()展开为:
+ (NSArray *)__rct_export__1720 { return @[@"", @"createTimer:(NSNumber *)callbackID duration:(NSTimeInterval)jsDuration jsSchedulingTime:(NSDate *)jsSchedulingTime repeats:(BOOL)repeats"]; }
- (void)createTimer:(NSNumber *)callbackID duration:(NSTimeInterval)jsDuration jsSchedulingTime:(NSDate *)jsSchedulingTime repeats:(BOOL)repeats
导入到JS的接口为: "NativeModules.ModuleName.createTimer(number)", 只有Selector第一个:以前的部分。
TODO: JavaScript Promise Javascript.Promise
C): RCT_REMAP_METHOD(js_name, method) 宏
该宏和RCT_EXPORT_METHOD(method)类似, 只不过通过该宏可以指定导入到JS的接口名称。
D): 宏 RCT_EXTERN_MODULE 和 RCT_EXTERN_REMAP_MODULE
RCT_EXTERN_MODULE(objc_name, objc_supername)
RCT_EXTERN_REMAP_MODULE(js_name, objc_name, objc_supername)
"Use this macro in a private Objective-C implementation file to automatically register an
external module with the bridge when it loads." 'private Object-C implementation'是指Objective-C中的类别(category)。
/* MyModuleExport.m:
*
* #import "RCTBridgeModule.h"
*
* @interface RCT_EXTERN_MODULE(MyModule, NSObject)
*
* RCT_EXTERN_METHOD(doSomething:(NSString *)string withFoo:(NSInteger)a bar:(NSInteger)b)
*
* @end
*
* This will now expose MyModule and the method to JavaScript via
* `NativeModules.MyModule.doSomething`
*/
#define RCT_EXTERN_MODULE(objc_name, objc_supername) \
RCT_EXTERN_REMAP_MODULE(, objc_name, objc_supername) /**
* Like RCT_EXTERN_MODULE, but allows setting a custom JavaScript name.
*/
#define RCT_EXTERN_REMAP_MODULE(js_name, objc_name, objc_supername) \
objc_name : objc_supername \
@end \
@interface objc_name (RCTExternModule) <RCTBridgeModule> \
@end \
@implementation objc_name (RCTExternModule) \
RCT_EXPORT_MODULE(js_name)
E: 宏RCT_EXTERN_METHOD 和 RCT_EXTERN_REMAP_METHOD
略
3.2 RCTBridgeModule的接口方法
RCTBridgeModule提供的接口用来注册"Bridge Module" 模块。
3.3 实现RCTBridgeModule接口的类
实现接口RCTBridgeModule的类都是 "Bridge Module" 模块类,这些类的实例称为"Bridge Module"。
3.4 "Bridge Module" 初始化
RCTBatchedBridge类的方法initModules 对Native "Bridge Module" 进行了初始化。
Reference
1. React-Native Source Code
iOS.ReactNative-3-about-viewmanager-uimanager-and-bridgemodule的更多相关文章
- iOS React-Native入门指南之HelloWorld
React-native 作为facebook开源项目,最近是火的一塌糊涂,它采用node.js能够写ios和android的native界面代码,简直是太酷了.支持动态更新,而且appstore 提 ...
- 新闻头条应用源码ios版
<ignore_js_op> 源码下载:http://code.662p.com/view/13343.html 作者ymcao,源码TopNewsIOS,新闻头条IOS ...
- React-Native入门指南之HelloWorld
iOS React-Native入门指南之HelloWorld React-native 作为facebook开源项目,最近是火的一塌糊涂,它采用node.js能够写ios和android的nativ ...
- React Native之code-push的热更新(ios android)
React Native之code-push的热更新(ios android) React Native支持大家用React Native技术开发APP,并打包生成一个APP.在动态更新方面React ...
- react-native开发总结
项目地址:http://liu12fei08fei.github.io/blog/41react-native.html 说明 • 项目总结代码地址 • 从项目开始启动(2018-07-02)到项目进 ...
- react-native 常用命令
创建项目 react-native init AwesomeProject //AwesomeProject是项目名 启动 Node.js web server react-native start ...
- React Native环境配置
React Native环境配置 史上最全Windows版本搭建安装React Native环境配置 配置过React Native 环境的都知道,在Windows React Native环境配置有 ...
- window环境下搭建react native及相关插件
可以先浏览一下中文翻译的开发文档具体了解一下关于React Native,想要查看官方文档可以点http://facebook.github.io/react-native/docs/getting- ...
- React Native获取组件位置和大小
RN页面中定位或滚动操作时,需要获取元素的大小和位置信息,有几种常用的方法 获取设备屏幕的宽高 import {Dimensions} from 'react-native'; var {height ...
- CodePush热更新组件详细接入教程
CodePush热更新组件详细接入教程 什么是CodePush CodePush是一个微软开发的云服务器.通过它,开发者可以直接在用户的设备上部署手机应用更新.CodePush相当于一个中心仓库,开发 ...
随机推荐
- WS+MQ+WCF+EF(Code First)
前言 有段时间没有更新博文了,一直在忙工作很少有时间静下心来继续研究点东西,说来也惭愧,归咎原因最主要的还是因为懒惰.空想也是不管用的,有时候很多想法被扼杀到了摇篮里,还没开始做就放弃了,这是多数人会 ...
- cs107
基本类型:bool,char,short,int,long,float,double 对于char,short,int,long: 多字节类型赋值给少字节类型,对低字节的细节感兴趣,位模式拷贝. 少字 ...
- Maximo子表中增加附件功能
附件功能的实现(详见ewell.webclient.beans.warranty.WarrantysDateBean ,ewell.webclient.beans.doclinks.custom.Ad ...
- synchronized锁重入
package synLockIn_1; /* synchronized锁重入,当一个线程得到一个对象锁且还未释放锁时,再次请求此对象锁时可以再次得到该对象的锁 * 此例中线程1进入Service类的 ...
- 使用二分法查找mobile文件中区号归属地
#!/usr/bin/env python #coding:utf-8 ''' Created on 2015年12月8日 @author: DL @Description: 使用二分法查找mobil ...
- ROW_NUMBER() OVER函数的用法
语法:ROW_NUMBER() OVER(PARTITION BY COLUMN ORDER BY COLUMN) partition 划分,分割 --ROW_NUMBER() 就是生成一个有顺序的行 ...
- CocoaPods安装及使用《转》
http://www.cnblogs.com/eagley/p/5407721.html
- Nginx系列一:信号与配置
一.Nginx与信号 Nginx支持平滑重启,相比于Apache,修改了配置文件后可以不需要先停止程序,再重新启动. 1.启动 nginx –c nginx.conf 其中,-c nginx.conf ...
- 机器学习PR:感知机模型
感知机是二类分类的线性分类模型,所谓二分类指的是输出的类别只有-1或1两种,所谓线性指的是输入的特征向量集合在特征空间中被超平面划分为相互分离的正负两类.感知机学习的目的正是为了求出将训练数据进行线性 ...
- 考查SQLite 3索引对整数排序的性能影响
做个实验,想了解SQLite3索引对整数排序的性能影响. 用这个测试表,考查绿色那列: id name date 自增型主键 字符串型,随机生成 整数型 随机生成,范围0到54354354 1 bMz ...