RN原生模塊的調(diào)用基礎(chǔ)(五)

假設(shè)現(xiàn)在我們要畫一個(gè)圓纺阔,當(dāng)然,你可以用js自己去實(shí)現(xiàn)即寡,但是今天要講的并不是js去實(shí)現(xiàn)圓徊哑,而是在Native端去寫一個(gè)View去暴露給js調(diào)用,ok聪富,我們廢話少說(shuō)莺丑,進(jìn)入正題

暴露一個(gè)簡(jiǎn)單控件給js調(diào)用分為以下幾個(gè)步驟:
1.自定義一個(gè)View控件
2.自定義一個(gè)Manager實(shí)現(xiàn)SimpleViewManager
3.將自定義的Manager注冊(cè)進(jìn)Package
4.js調(diào)用

自定義View控件相信android老司機(jī)們?cè)缫疡{輕就熟,不再啰嗦墩蔓,直接看代碼

public class CircleView extends View {
    private static final String TAG = "CircleView";
    private Paint mPaint;
    private float mRadius;

    public CircleView(Context context) {
        super(context);
        mPaint = new Paint();
    }

    /**
     * 設(shè)置圓的背景色
     * @param color
     */
    public void setColor(Integer color) {
        mPaint.setColor(color); // 設(shè)置畫筆顏色
        invalidate();   // 更新畫板
    }

    /**
     * 設(shè)置圓的半徑
     * @param radius
     */
    public void setRadius(Integer radius) {
        /**
         * 由于JS傳過(guò)的數(shù)字是dip單位,需要轉(zhuǎn)換為實(shí)際像素
         * 使用com.facebook.react.uimanager包中的PixelUtil,進(jìn)行轉(zhuǎn)換
         */
        mRadius = PixelUtil.toPixelFromDIP(radius);
        invalidate();
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        canvas.drawCircle(mRadius, mRadius, mRadius, mPaint);;
    }

下面開始自定義Manager:

public class CircleManager extends SimpleViewManager<CircleView> {
    @Override
    public String getName() {
        return "MCircle";
    }

    /**
     * 創(chuàng)建UI組件實(shí)例
     * @param reactContext
     * @return CircleView
     */
    @Override
    protected CircleView createViewInstance(ThemedReactContext reactContext) {
        return new CircleView(reactContext);
    }

    /**
     * 傳輸背景色參數(shù)
     * @param view
     * @param color
     */
    @ReactProp(name = "color")
    public void setColor(CircleView view, Integer color) {
        view.setColor(color);
    }

    /**
     * 傳輸半徑參數(shù)
     * @param view
     * @param radius
     */
    @ReactProp(name = "radius")
    public void setRadius(CircleView view, Integer radius) {
        view.setRadius(radius);
    }
}

注意這里的getName方法的返回值要與js里一致梢莽,這個(gè)文章后面再詳細(xì)說(shuō)明,先有個(gè)大致的概念奸披,同時(shí)還需要注意的是@ReactProp這個(gè)注解

OK昏名,我們有了Manager,現(xiàn)在去注冊(cè)到package阵面,很簡(jiǎn)單

public class CommPackage implements ReactPackage {
    @Override
    public List<NativeModule> createNativeModules(ReactApplicationContext reactContext) {
        List<NativeModule> modules = new ArrayList<>();
        modules.add(new CommonModule(reactContext));
        return modules;
    }

    @Override
    public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) {
        List<ViewManager> views = new ArrayList<>();
        views.add(new CircleManager());
        return views;
    }
}

注意到這里的createViewManagers不再返回一個(gè)emptyList轻局,而是把CircleManager給add進(jìn)去了

最后,我們寫一下js代碼
首先我們需要寫一個(gè)Circle.js

import React, { Component } from 'react';
import { requireNativeComponent, View, processColor } from 'react-native';  //processColor=>字符Color轉(zhuǎn)換為數(shù)字
import PropTypes from 'prop-types';

const MCircle = requireNativeComponent('MCircle', {
  propTypes: {
      color: PropTypes.number,
      radius: PropTypes.number,
      ...View.propTypes // 包含默認(rèn)的View的屬性
  },
});

class Circle extends Component {
      static propTypes = {
        radius: PropTypes.number,
        color: PropTypes.string, // 這里傳過(guò)來(lái)的是string
        ...View.propTypes // 包含默認(rèn)的View的屬性
      }

      render() {
        const { style, radius, color } = this.props;

        return (
          <MCircle
            style={style}
            radius={radius}
            color={processColor(color)}
          />
        );
      }
}

module.exports = Circle;

注意這邊的requireNativeComponent有兩個(gè)參數(shù):第一個(gè)參數(shù)是MCircle膜钓,這個(gè)名字要與上面的CircleManager的getName返回值保持一致嗽交;第二個(gè)參數(shù)約定了屬性的一些類型卿嘲,可以看到有一個(gè)number類型的color屬性颂斜,一個(gè)number類型的radius屬性,這兩個(gè)屬性是怎么對(duì)應(yīng)到Native端的拾枣,這就要說(shuō)到上面的@ReactProp注解沃疮,我們截取下之前的代碼

/**
     * 傳輸背景色參數(shù)
     * @param view
     * @param color
     */
    @ReactProp(name = "color")
    public void setColor(CircleView view, Integer color) {
        view.setColor(color);
    }

    /**
     * 傳輸半徑參數(shù)
     * @param view
     * @param radius
     */
    @ReactProp(name = "radius")
    public void setRadius(CircleView view, Integer radius) {
        view.setRadius(radius);
    }

可以看到@ReactProp注解有個(gè)name的key,他的value值對(duì)應(yīng)的就是Circle.js里propTypes聲明的屬性梅肤,每當(dāng)js那邊設(shè)置color屬性司蔬,這個(gè)setColor的方法就會(huì)自動(dòng)調(diào)用,這樣就把js的參數(shù)傳遞到了native

最后我們改下index.js

'use strict'
import React, { Component} from 'react';
import { AsyncStorage,NativeModules,ToastAndroid } from 'react-native';
import {
  AppRegistry,
  StyleSheet,
  Text,
  Image,
  View
} from 'react-native';

import Circle from './Circle';

let title = 'React Native界面';

export default class YRNTest extends Component {
    /**
    * Callback 通信方式
    */
    callbackComm(msg) {
        NativeModules.CommonModule.rnCallNativeFromCallback(msg,(result) => {
             ToastAndroid.show("CallBack收到消息:" + result, ToastAndroid.SHORT);
        })
    }

    /**
    * Promise 通信方式
    */
    promiseComm(msg) {
        NativeModules.CommonModule.rnCallNativeFromPromise(msg).then(
            (result) =>{
                ToastAndroid.show("Promise收到消息:" + result, ToastAndroid.SHORT)
            }
        ).catch((error) =>{console.log(error)});
    }

  render() {
    return (
        <View style={styles.container}>
            <Circle
                style={{width: 100, height: 100}}
                color="#25c5f7"
                radius={50}
            />
        </View>
//      <View style={styles.container}>
//        <Text style={styles.welcome} >
//            {title}
//        </Text>
//        <Text style={styles.welcome} onPress={this.callbackComm.bind(this,'你好啊姨蝴,android')}>
//             Callback通信方式
//        </Text>
//        <Text style={styles.welcome} onPress={this.promiseComm.bind(this,'你好啊俊啼,android')}>
//             Promise通信方式
//        </Text>
//      </View>
    );
  }
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'center',
    alignItems: 'center',
    backgroundColor: '#FFFFFF',
  },
  welcome: {
    fontSize: 20,
    textAlign: 'center',
    margin: 10,
  }
});

AppRegistry.registerComponent('YRNTest', () => YRNTest);

添加import Circle from './Circle';
render里改為

<View style={styles.container}>
            <Circle
                style={{width: 100, height: 100}}
                color="#25c5f7"
                radius={50}
            />
</View>

運(yùn)行工程,大功告成左医,附上截圖
1525331564339.jpg
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末授帕,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子浮梢,更是在濱河造成了極大的恐慌跛十,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,590評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件秕硝,死亡現(xiàn)場(chǎng)離奇詭異芥映,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,157評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門奈偏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)坞嘀,“玉大人,你說(shuō)我怎么就攤上這事霎苗∧房裕” “怎么了?”我有些...
    開封第一講書人閱讀 169,301評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵唁盏,是天一觀的道長(zhǎng)内狸。 經(jīng)常有香客問(wèn)我,道長(zhǎng)厘擂,這世上最難降的妖魔是什么昆淡? 我笑而不...
    開封第一講書人閱讀 60,078評(píng)論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮刽严,結(jié)果婚禮上昂灵,老公的妹妹穿的比我還像新娘。我一直安慰自己舞萄,他們只是感情好眨补,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,082評(píng)論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著倒脓,像睡著了一般撑螺。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上崎弃,一...
    開封第一講書人閱讀 52,682評(píng)論 1 312
  • 那天甘晤,我揣著相機(jī)與錄音,去河邊找鬼饲做。 笑死线婚,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的盆均。 我是一名探鬼主播塞弊,決...
    沈念sama閱讀 41,155評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼泪姨!你這毒婦竟也來(lái)了游沿?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 40,098評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤驴娃,失蹤者是張志新(化名)和其女友劉穎奏候,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體唇敞,經(jīng)...
    沈念sama閱讀 46,638評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蔗草,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,701評(píng)論 3 342
  • 正文 我和宋清朗相戀三年咒彤,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片咒精。...
    茶點(diǎn)故事閱讀 40,852評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡镶柱,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出模叙,到底是詐尸還是另有隱情歇拆,我是刑警寧澤,帶...
    沈念sama閱讀 36,520評(píng)論 5 351
  • 正文 年R本政府宣布范咨,位于F島的核電站故觅,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏渠啊。R本人自食惡果不足惜输吏,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,181評(píng)論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望替蛉。 院中可真熱鬧贯溅,春花似錦、人聲如沸躲查。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,674評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)镣煮。三九已至姐霍,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間怎静,已是汗流浹背邮弹。 一陣腳步聲響...
    開封第一講書人閱讀 33,788評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工黔衡, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蚓聘,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,279評(píng)論 3 379
  • 正文 我出身青樓盟劫,卻偏偏與公主長(zhǎng)得像夜牡,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子侣签,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,851評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容