android原生组件,RN原生的安卓UI组件

这里有一大堆的原生组件可以用,一些是平台自带的,另一些是第三方库的,而更多的正在其他项目中被使用。RN包装了大部分核心平台组件,但不是所有。幸运的是可以很简单地完美集成现有的原生组件到RN中

ImageView案例

这个案例的目标是实现在JS中使用imageView。RN中对安卓原生组件的创建和管理都是通过继承ViewManager或SimpleViewManager。SimpleViewManager在这里用起来更加方便,因为它可以把应用公共的属性如背景色、透明度和弹性布局。

以上RN中创建出来的这些子类本质上都是单例的,每个子类的单个实例都是通过RN bridge创建的。这些子类把原生组件交给NativeViewHierarchyManager,NativeViewHierarchyManager再通知这些原生组件去设置和更新自己的属性。而ViewManager则让这些原生组件通过RN bridge把事件传递回给JS。

1.创建ViewManager子类。

public class ReactImageManager extends SimpleViewManager{public static final String REACT_CLASS = "RCTImageView";//JS通过这里返回的name值来获取当前子类

@OverridepublicString getName() {returnREACT_CLASS;

}//视图都是在这个方法中创建的

@OverrideprotectedReactImageView createViewInstance(ThemedReactContext reactContext) {return newReactImageView(context, Fresco.newDraweeControllerBuilder(), mCallerContext);

}

}

3.通过ReactProps或ReactPropsGroup注解来暴露视图属性的setter

setter方法的第一个参数应该是要更新的视图,第二个参数是属性值。要求是public void。

ReactProps注解拥有一个必填的字符串name属性。这个name值用于在JS端引用视图的属性。除了name之外,还有其他可选的defaultBoolean、defaultInt和defaultFloat。这些属性的值会被传递到setter函数中(当setter引用的属性被移除时会选用这些值)

在RN中更新视图的属性会导致setter执行。一种更新组件的方式就是移除之前设置的属性,这样setter也会执行,而且采用defaultXX设置的默认值

@ReactProp(name = "src")public voidsetSrc(ReactImageView view, @Nullable ReadableArray sources) {

view.setSource(sources);

}

@ReactProp(name= "borderRadius", defaultFloat =0f)public void setBorderRadius(ReactImageView view, floatborderRadius) {

view.setBorderRadius(borderRadius);

}

@ReactProp(name=ViewProps.RESIZE_MODE)public voidsetResizeMode(ReactImageView view, @Nullable String resizeMode) {

view.setScaleType(ImageResizeMode.toScaleType(resizeMode));

}

4.注册viewManager

在java中的最后一步就是把ViewManager注册到application中。

@Overridepublic ListcreateViewManagers(

ReactApplicationContext reactContext) {return Arrays.asList(newReactImageManager()

);

}

步骤:首先创建一个package(一个子类实现了ReactPackage,实现以上方法),接着在Application中重新getPackages方法,将刚刚的package实例添加到数组中返回即可。过程可以参照这里https://facebook.github.io/react-native/docs/native-modules-android.html

5.实现JS模块

最后创建一个模块,模块中为用户定义了组件的接口

//ImageView.js

import PropTypes from'prop-types';

import {requireNativeComponent, ViewPropTypes} from'react-native';var iface ={

name:'ImageView',

propTypes: {

src: PropTypes.string,

borderRadius: PropTypes.number,

resizeMode: PropTypes.oneOf(['cover', 'contain', 'stretch']),

...ViewPropTypes,//include the default view properties

},

};

module.exports= requireNativeComponent('RCTImageView', iface);

以上接口的name值用于debug,而propTypes用于让RN去反射,来检测原生组件是否有效。

事件

就以上例子不做其他任何处理,原生组件可以处理对应的触摸事件,如setOnClickListener等。现在需要把这个事件通知给JS,当事件发生的时候(下面那个是某事件的回调方法),开始往JS传递一个topChange事件

class MyCustomView extendsView {

...public voidonReceiveNativeEvent() {

WritableMap event=Arguments.createMap();

event.putString("message", "MyMessage");

ReactContext reactContext=(ReactContext)getContext();

reactContext.getJSModule(RCTEventEmitter.class).receiveEvent(

getId(),"topChange",

event);

}

}

为了建立toChange消息与JS回调函数onChange的映射,需要在ViewManager中重写以下方法。这样一来,当以上消息发出之后,RN原生组件上的onChange属性回调函数就会被执行

public class ReactImageManager extends SimpleViewManager{

...publicMap getExportedCustomBubblingEventTypeConstants() {returnMapBuilder.builder()

.put("topChange",

MapBuilder.of("phasedRegistrationNames",

MapBuilder.of("bubbled", "onChange")))

.build();

}

}

在RN中定义一个组件进行包装,设置原生组件的onChange事件

//MyCustomView.js

class MyCustomView extends React.Component {

constructor(props) {

super(props);this._onChange = this._onChange.bind(this);

}

_onChange(event: Event) {if (!this.props.onChangeMessage) {return;

}this.props.onChangeMessage(event.nativeEvent.message);

}

render() {return ;

}

}

MyCustomView.propTypes={/**

* Callback that is called continuously when the user is dragging the map.*/onChangeMessage: PropTypes.func,

...

};var RCTMyCustomView =requireNativeComponent(`RCTMyCustomView`, MyCustomView, {

nativeOnly: {onChange:true}

});

文档中对于nativeOnly的解释我不是很懂....

实践

坑坑坑。。。在RN使用这个原生组件时,如果不设置尺寸,则这个组件不会被渲染出来,通过android studio monitor分析如下,少了一个节点:

63bcf424365e16e6c4dda8ddc31304b9b39.jpg

1、Android显示GIF动画 GifView GifView 是一个为了解决android中现在没有直接显示gif的view,只能通过mediaplay来显示这个问题的项目,其用法和 ImageView一样,支持gif图片 使用方法:1-把GifView.jar加入你的项目。2-在xml中配置GifView的基本属性,GifView继承自View类,和Button、ImageView一样是一个UI控件。 如: 3-在代码中配置常用属性: // 从xml中得到GifView的句柄 gf1 = (GifView) findViewById(R.id.gif1); // 设置Gif图片源 gf1.setGifImage(R.drawable.gif1); // 添加监听器 gf1.setOnClickListener(this); // 设置显示的大小,拉伸或者压缩 gf1.setShowDimension(300, 300); // 设置加载方式:先加载后显示、边加载边显示、只显示第一帧再显示 gf1.setGifImageType(GifImageType.COVER); GifView的Jar包共有四个类: GifAction.java 观察者类,监视GIF是否加载成功 GifFrame.java 里面三个成员:当前图片、延时、下张Frame的链接。 GifDecoder.java 解码线程类 GifView.java 主类,包括常用方法,如GifView构造方法、设置图片源、延迟、绘制等。 2、Calendar.v0.5.0 是 Android 平台的一个日历显示组件。 3、CWAC EndlessAdapter 是 Android 上一个可以无限往下滑进行列表数据加载的控件。 4、Android Horizontal ListView 是 Android 上一个水平滑动的 ListView 组件。 5、Android ViewBadger 视图布局。 6、滑动刷新的ListView Android PullToRefresh 为 Android 应用提供一个向下滑动即刷新列表的功能,就两个目标文件。 7、pakerfeldt-android-viewflow 是 Android 平台上一个视图切换的效果库。ViewFlow 相当于 Android UI 部件提供水平滚动的 ViewGroup,使用 Adapter 进行条目绑定。 8、Android 导航菜单 RibbonMenu 是 Android 上的一个导航菜单组件。就三个目标文件,菜单项直接在 XML 中定义,可添加文本和图标。 9、AndroidUI工具包 android-ui-utils 是一个工具包用来帮助设计和开发 Android 用户界面,包含三个单独的工具:Android Asset Studio用户界面原型模具,Android 设计预览,时常需要重复确认程序版面设计状况的 Android App 开发者,应该会爱上这个轻量级的 Java 程序:Andorid Design Preview 工具,通过 USB 连接之后,只要简单的在计算机中选取您想要显示的程序版面范围,就可将镜像结果直接显示于手机装置之上。 10、Androidui开发类库 GreenDroid 是一个Androidui开发类库,能够使你的Android开发更加简便和快捷。 11、Android滑动式菜单 SlidingMenu 是 Android 上实现类似 Facebook 和 Path 2.0 滑动式菜单的组件。 12、AsyncImageView 是 Android 上的一个异步从网络上获取图片并进行浏览的开源组件,可自动在本地进行缓存。该项目是 GreenDroid 的一部分。 13、仿Path按钮动画效果 PathButton 仿照Path应用首页左下角的Button动画效果写了个简单的Demo,由于数学不好,坐标总是和理想有出入,只是大致实现了动画效果,若果有人能把坐标算对,那么修改我的demo就能做成类似Path的那种动画效果!希望大家出点力帮着优化一下,并分享出来! 14、Android Intent开发包 OpenIntents Ope
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值