React-Native 与 React-Web 的融合

关于

对于react-native在实际中的应用, facebook官方的说法是react-native是为多平台提供共同的开发方式,而不是说一份代码,多处使用。 而后一份代码可以多处使用仍是颇有意义的,我所了解到的已经在尝试作这件事情的:javascript

  1. modularise-css-the-react-way
  2. react-style
  3. native-css

现阶段你们都是在摸索中,且react-native 还不够成熟,为此我也想经过一个实际的例子提早探究一下共享代码的可行性。css

下面是我以SampleApp作的一个简单demo, 先奉献上截图:html

web 版本:java

react-native版本:node

初步想法

组件

react-native基本上是View套上Text这样来布局,为了作web和native的兼容,咱们得提供继承版本的View ,针对不一样的平台返回不一样作兼容,咱们将提供:react

  1. Share.View -> View (reac-native = View , web = div)
  2. Share.P + Share.Span -> Text (Text在react-native中分为块级别和inline级别因此得用两个元素来区分)

样式

咱们知道react-native的样式是css很小的一个子集,大概支持50种属性,为了作到web和native使用一样地样式,那么个人想法是:android

  1. 使用css文件来编写样式,经过编译的方式生产不一样平台须要的样式
  2. 对于web,使用auto-prefixel处理,生产web兼容的css代码
  3. 对于react-native,生成对应的styles.js
  4. css的写法用OOCSS的方式

这样作的另一个缘由是,由于css是全集,react-native是子集,全集到子集能够经过删减来处理,可是若是想经过子集到全集就会很麻烦(react-style就是经过react-native来生成css)。 且这样作还有不少好处,例如咱们能够支持react-native里边不支持的css写法,例如padding: a b c d; 这种写法很容易获得兼容。webpack

其实这里,不管react-native仍是react-web都支持style={}这样的写法. 上面例子中的web截图实际上是没有引用css的,但inline样式对于web来讲并非优选。ios

实现思路

首先大概整理一下咱们须要解决的问题:git

  1. 如何区分web和native
  2. js如何对应不一样的平台来编译,由于react-native使用的是本身的依赖管理packager
  3. css如何编译为js
  4. 代码结构应该是怎样的

问题一 : 如何区分web和native

react-native 里边会有window变量吗?我试了一下,是有的,那window变量里边不可能有location,document之类的吧, 借着这种想法,可用以下方法来区分native和web

javascriptvar isNative = !window.location;

问题二:如何对应不一样平台打包

对于react-native,是经过packager来打包的,具体的实现和逻辑能够随时查看packager的readme文档。 那咱们怎么将适用于native的代码打包成web的代码,首先想到的是browserify, webpack。 都是遵循commonJs规范,我的更喜欢前者, 用它来应该足以知足需求。

问题三: css如何编译为js

前面提到了native-css , 能够用它来帮助咱们完成打包。

问题四:代码结构应该是怎样的

web和native的代码都写在同一个地方,如何作区分呢? 这个问题固然最好就是不作区分,或者就像女生的衣服,指望是越少越好,但永远不可能木有(猥琐了:-】)。

我设想中的一个最简模型的目录结构,web和ios有不一样的入口,web和ios有单独的目录, 组件共享, 以下:

├── compo.js            // 咱们会使用到得公共组件
├── styles.css          // compo的样式文件
├── index.web.js        // web 入口
├── index.ios.js        // ios 入口
├── shared.js           // 作兼容的共享变量文件
├── ios                 // ios 目录
└── web                 // web 目录
    ├── index.html      // web 页面
    ├── index.web.js    // 打包事后的js
    └── react.js        // react.js依赖

好像很复杂的样子, 其实相对于本来的SampleApp,只是多了index.web.js , web目录, shared三者。 而后style经过style.css来描述。

具体实现

咱们已经整理了具体的实现思路,下面是我就会直接吐出个人实现代码, 重点的地方都会在源码里边有注释

先看应用代码:

ios入口:index.ios.js

javascript/**
     * Sample React Native App
     * https://github.com/facebook/react-native
     */
    'use strict';
    var React = require('react-native');
    var Compo = require('./compo');
    React.AppRegistry.registerComponent('ShareCodeProject', () =>  Compo);

web入口:index.web.js

javascript/**
     * for web
     */
    var Compo = require('./compo');
    React.render(<Compo />, document.getElementById('App'));

样例组件:compo.js

javascript// 依赖的公共库,经过它获取兼容的组件
    var Share = require('./shared');
    // styles是style.css build事后生成的style.js
    var styles = require('./styles');
    var React = Share.React;
    var {
      View,
      P,
      Span
    } = Share;

    var Compo = React.createClass({
      render: function() {
        return (
          <View style={styles.container}>
            <P style={styles.welcome}>
              Welcome to React Native!
            </P>
            <P style={styles.instructions}>
              To get started, edit index.ios.js
            </P>
            <P style={styles.instructions}>
              Press Cmd+R to reload,{'\n'}
              Cmd+Control+Z for dev menu
            </P>
          </View>
        );
      }
    });

    module.exports = Compo;

组件样式: style.css

css/**
     * 你们可能发现了css的写法仍是小驼峰,是的不是横杠,暂时咱们仍是以这种方式处理
     * native-css 目测不支持横杠,(本身重写native-css相对来讲是比较容易的,彻底能够作到css兼容到react-native的css子集)
     */
    .container {
        flex: 1;
        justifyContent: center;
        alignItems: center;
        backgroundColor: #F5FCFF;
    }

    .welcome {
        fontSize: 20;
        textAlign: center;
        margin: 10;
    }

    .instructions {
        textAlign: center;
        color: #333333;
        marginBottom: 5;
    }

index.html

html<!DOCTYPE html>
    <html>
      <head>
        <title>Hello React!</title>
        <script src="./react.js"></script>
        <!-- No need for JSXTransformer! -->
      </head>
      <body>
        <div id="App"></div>
        <script src="./index.web.js"></script>
      </body>
    </html>

Share部分的处理

shared.js

javascriptvar Share = {};
    var React = require('react-native');
    var isNative = !window.location;
    /**
     * 判断是web的时候,从新赋值React
     */
    if (window.React) {
        React = window.React;
    } 
    Share.React = React;

    /**
     * 作底层的兼容, 固然这里只是作了一个最简demo,具体实现的时候可能会对props作各类兼容处理
     */
    if (!isNative) {

        Share.View = React.createClass({
            render: function() {
                return <div {...this.props}/>
            }
        });

        Share.P = React.createClass({
            render: function() {
                return <p {...this.props}/>
            }
        });

        Share.Span = React.createClass({
            render: function() {
                return <span {...this.props}/>
            }
        });
    } else {
        // alert('isNative')
        Share.View = React.View;
        Share.P = React.Text;
        Share.Span = React.Text;
        Share.Text = React.Text;
    }

    module.exports = Share;

build打包程序

javascriptvar fs = require('fs');
    var nativeCSS = require('native-css'),
    var cssObject = nativeCSS.convert('./styles.css');

    toStyleJs(cssObject, './styles.js');
    buildWebReact();

    /**
     * native-css获取到得是一个对象,须要将cssObject转化为js代码
     */
    function toStyleJs(cssObject, name) {
        console.log('build styles.js \n');
        var tab = '    ';
        var str = '';

        str += '/* build header */\n';
        str += 'var styles = {\n';

        for(var key in cssObject) {
            var rules = cssObject[key];
            str += tab + key + ': {\n';
            for(var attr in rules) {
                var rule = rules[attr];
                str += tab + tab + attr + ': ' + format(rule) + ',\n'
            }
            str += tab + '},\n' 
        }

        str += '};\n'
        str += 'module.exports = styles;\n'

        fs.writeFile(name, str)
        function format(rule) {
            if (!isNaN(rule - 0)) {
                return rule;
            }
            return '"' + rule + '"';
        }
    }

    /**
     * 构造web使用的react
     */
    function buildWebReact() {
        console.log('build web bundle');
        var browserify = require('browserify');
        var b = browserify();
        b.add('./index.web.js');

        // 添加es6支持
        b.transform('reactify', {'es6': true});

        // ignore掉react-native 
        b.ignore('react-native')
        var wstream = fs.createWriteStream('./web/index.web.js');
        b.bundle().pipe(wstream);
    }

也尝试一下由react-native 到react-web的兼容方案

问题

  1. flexbox的写法在react-native上面咱们会发现, 不用在父元素上声明display: flex; 在web上必需要作这样的声明, 因此咱们须要让设置了flex:*的元素的父元素display: flex;
  2. flexbox在android上是由不少bug的,因此必需要解决兼容性问题webkit-box

解决方案

1. nested 的style写法

javascriptstyles = StyleSheet.create({
        mod: {
            flexDirection: 'row',
            item: {
                flex: 1
            }
        }
    });

这样的写法有些像less,咱们能够知道元素的层级关系, 这样我能够遍历这个对象,查找子元素有设置flex的,父元素加上display:flexbox

2. 经过自定义元素

javascriptvar GridSystem = require('GridSystem');
 var {
    Row,
    Grid,
    Grid6,
    Grid4
 } = GridSystem;
 <Row ...>
    <Grid/>
    <Grid/>
 </Row>

经过标签的方式, 至关于给react-native或者react添加了一个网格系统,同时咱们能够直接在Row上设置display:flex.

3. 遍历查找

彻底同react-native原生的写法,直接在web中兼容,遍历全部有flex样式的节点,直接作兼容。

javascriptcomponentDidMount: function() {
        var $node = this.getDOMNode();
        var $parent = $node.parentNode;
        var $docfrag = document.createDocumentFragment();
        $docfrag.appendChild($node);

        var treeWalker = document.createTreeWalker($node, NodeFilter.SHOW_ELEMENT, { 
            acceptNode: function(node) { 
                return NodeFilter.FILTER_ACCEPT; 
            } 
        }, false);

        while(treeWalker.nextNode()) {
            var node = treeWalker.currentNode;
            if (node.style.flex) {
                flexChild(node);
                flexParent(node.parentNode);
            }
        };

        $parent.appendChild($docfrag);
    }

    function flexChild(node) {
        if (node.__flexchild__) {
            return;
        }
        node.__flexchild__ = true;
        var flexGrow = node.style.flexGrow;
        addStyle(node, `
            -webkit-box-flex: ${flexGrow};
            -webkit-flex: ${flexGrow};
            -ms-flex: ${flexGrow};
            flex: ${flexGrow};
        `);
        node.classList.add('mui-flex-cell');
    }

    function flexParent(node) {
        if (node.__flexparentd__) {
            return;
        }
        node.__flexparentd__ = true;
        node.classList.add('mui-flex');
    }
css.mui-flex {
        display: -webkit-box!important;
        display: -webkit-flex!important;
        display: -ms-flexbox!important;
        display: flex!important;
        -webkit-flex-wrap: wrap;
        -ms-flex-wrap: wrap;
        flex-wrap: wrap;
        -webkit-box-orient: vertical;
        -webkit-box-direction: normal;
        -webkit-flex-direction: column;
        -ms-flex-direction: column;
        flex-direction: column;
    }

    .mui-flex-cell {
        -webkit-flex-basis: 0;
        -ms-flex-preferred-size: 0;
        flex-basis: 0;
        max-width: 100%;
        display: block;
        position: relative;
    }

总结

这个demo很简单,实际应用中应该会有不少地方的坑, 好比:

  1. 模块中依赖只有native才有的组件
  2. Native模块的事件处理和web大不相同
  3. 现实环境中的模块更多,更复杂,如何作模块的管理

对于write once, run anywhere 这个观点. 相信不一样的人会有不一样的见解,但不管如何,若是兼容成本不大,这样的兼容技术方案对业务开发是有极大意义的。

ps0: 这里仅仅作可行性方案的分析,不表明我认同或不认同这种方案。 ps1: 你们若是有更好的方案,求教,求讨论。 ps2: 听说微博关注@sysu_学家不会怀孕

相关文章
相关标签/搜索