React可以被应用于任何web应用中,也可以被嵌入到其他应用中,同时其他应用也可以嵌入到React中。下文将介绍React是如何与其他库进行协同的。

集成带有DOM操作的插件

React不会理会React自身之外的DOM操作,它会根据内部虚拟DOM来决定是否需要更新,而且如果同一个DOM节点被另一个库操作了,React会觉得困惑而且没有办法恢复。避免冲突的最简单方式就是防止React组件更新。

如何解决这个问题

下面举个例子,一个用于通用JQuery插件的wrapper,首先添加一个ref到根DOM元素,然后再componentDidMount中获取到这个DOM元素的引用,然后将其传递给JQuery插件。为了防止Reacat在挂载后出触碰这个DOM,我们的render函数返回的是一个空的<div />.这个div元素既没有属性也没有子元素,所以React没有理由去更新它,使得Jquery插件可以自由的去管理这部分DOM。我们同时定义了componentDidMount和componentWillUnmount生命周期函数,因为我们绑定后还要进行注销监听,因为这样可以避免内存泄漏。

集成JQuery Chosen插件

下面的例子将给用于增强select输入的Chosen插件写一个wrapper.

Chosen对DOM做了哪些操作?

  • 读取原DOM节点的属性,然后使用行内样式隐藏它。
  • 紧挨着这个select之后增加一个独立的具有它自身显示表现的DOM节点。
  • 值发生变化的时候触发JQuery事件来通知我们这些变化。

最终想要实现的效果

function Example() {
  return (
    <Chosen onChange={value => console.log(value)}>
      <option>vanilla</option>
      <option>chocolate</option>
      <option>strawberry</option>
    </Chosen>
  );
}
  • 首先创建一个空组件,这个组件的render函数会返回一个包含select的div
class Chosen extends React.Component {
  render() {
    return (
      <div>
        <select className="Chosen-select" ref={el => this.el = el}>
          {this.props.children}
        </select>
      </div>
    );
  }
}
  • 我们为什么要把select使用一个额外的div包裹起来,是因为上文我们说过,Chosen会紧挨着我们传递给它的select节点追加一个DOM元素。但是React规定一个组件只能有一个根DOM元素,所以我们通过一个div进行包裹,就不会使得React更新和Chosen追加的额外DOM节点发生冲突。

  • 实现声明周期函数

componentDidMount() {
  this.$el = $(this.el);
  this.$el.chosen();
}

componentWillUnmount() {
  this.$el.chosen('destroy');
}
  • 实现在值变化的时候被通知到,需要在订阅由Chosen管理的select上的JQuery change事件。不直接把this.props.onChange传递给Chosen,是因为组件的props可能随时变化,并且也包括事件处理函数。我们通过定义一个handleChange方法来调用this.props.onChange,然后订阅JQuery的change事件:
componentDidMount() {
  this.$el = $(this.el);
  this.$el.chosen();

  this.handleChange = this.handleChange.bind(this);
  this.$el.on('change', this.handleChange);
}

componentWillUnmount() {
  this.$el.off('change', this.handleChange);
  this.$el.chosen('destroy');
}

handleChange(e) {
  this.props.onChange(e.target.value);
}
  • Chosen的文档建议我们使用JQuery的trigger()来通知原始DOM元素这些变化,我们需要增加一个componentDidUpdate生命周期函数来通知Chosen关于children列表的变化。
componentDidUpdate(prevProps) {
  if (prevProps.children !== this.props.children) {
    this.$el.trigger("chosen:updated");
  }
}
  • Chosen组件的完整实现
class Chosen extends React.Component {
  componentDidMount() {
    this.$el = $(this.el);
    this.$el.chosen();

    this.handleChange = this.handleChange.bind(this);
    this.$el.on('change', this.handleChange);
  }
  
  componentDidUpdate(prevProps) {
    if (prevProps.children !== this.props.children) {
      this.$el.trigger("chosen:updated");
    }
  }

  componentWillUnmount() {
    this.$el.off('change', this.handleChange);
    this.$el.chosen('destroy');
  }
  
  handleChange(e) {
    this.props.onChange(e.target.value);
  }

  render() {
    return (
      <div>
        <select className="Chosen-select" ref={el => this.el = el}>
          {this.props.children}
        </select>
      </div>
    );
  }
}

和其他视图库集成

得益于ReactDOM.render()的灵活性,使得React可以被嵌入到其他的应用中。

利用React替换基于字符串的渲染

  • 在旧的Web应用中使用字符串渲染DOM是非常常见的,比如下面这段例子,这种例子是非常适合引入React的,我们可以直接把基于字符串的渲染重写为React组件。
$('#container').html('<button id="btn">Say Hello</button>');
$('#btn').click(function() {
  alert('Hello!');
});
  • 使用React组件进行重写
function Button() {
  return <button id="btn">Say Hello</button>;
}

ReactDOM.render(
  <Button />,
  document.getElementById('container'),
  function() {
    $('#btn').click(function() {
      alert('Hello!');
    });
  }
);
  • 使用React事件系统直接注册click处理函数到React button元素
function Button(props) {
  return <button onClick={props.onClick}>Say Hello</button>;
}

function HelloButton() {
  function handleClick() {
    alert('Hello!');
  }
  return <Button onClick={handleClick} />;
}

ReactDOM.render(
  <HelloButton />,
  document.getElementById('container')
);

把React嵌入到Backbone视图

  • Backbone是通过使用HTML字符串或者产生字符串的模板函数来创建DOM元素的过程的,这个过程同样可以通过渲染一个React组件来替换掉。下面我们会创建一个名为ParagraphView的Backbone视图,这个视图会重载render函数来渲染一个React Paragraph组件到Backbone提供的DOM元素中,在这里我们仍然需要使用ReactDOM.render()
function Paragraph(props) {
  return <p>{props.text}</p>;
}

const ParagraphView = Backbone.View.extend({
  render() {
    const text = this.model.get('text');
    ReactDOM.render(<Paragraph text={text} />, this.el);
    return this;
  },
  remove() {
    ReactDOM.unmountComponentAtNode(this.el);
    Backbone.View.prototype.remove.call(this);
  }
});

当一个组件在React树中从内部删除的时候,清理的工作是自动完成的,但是因为我们现在手动移出整个树,所以必须调用ReactDOM.unmountComponentAtNode()。

和Model层集成

React组件也可以使用其他框架和库的Model层。

在React组件中使用Backbone的Modal

在React组件中使用Backbone的Model和Collection最简单的方法就是监听多种变化事件并且手动强制触发一个更新。

class Item extends React.Component {
  constructor(props) {
    super(props);
    this.handleChange = this.handleChange.bind(this);
  }

  handleChange() {
    this.forceUpdate();
  }

  componentDidMount() {
    this.props.model.on('change', this.handleChange);
  }

  componentWillUnmount() {
    this.props.model.off('change', this.handleChange);
  }

  render() {
    return <li>{this.props.model.get('text')}</li>;
  }
}

class List extends React.Component {
  constructor(props) {
    super(props);
    this.handleChange = this.handleChange.bind(this);
  }

  handleChange() {
    this.forceUpdate();
  }

  componentDidMount() {
    this.props.collection.on('add', 'remove', this.handleChange);
  }

  componentWillUnmount() {
    this.props.collection.off('add', 'remove', this.handleChange);
  }

  render() {
    return (
      <ul>
        {this.props.collection.map(model => (
          <Item key={model.cid} model={model} />
        ))}
      </ul>
    );
  }
}