personnages

React可以用于任何Web应用程序。它可以嵌入到其他应用程序中,并且可以将其他应用程序嵌入到React中。本指南将检查一些更常见的用例,重点介绍与jQuery和Backbone的集成,但是可以将相同的思想应用于将组件与任何现有代码集成。

与DOM操作插件集成

React不知道在React之外对DOM做出的更改。它根据自己的内部表示来确定更新,并且如果相同的DOM节点被另一个库操纵,则React会感到困惑并且无法恢复。

这并不意味着将React与其他影响DOM的方式结合起来是不可能的,甚至是一定困难的,您只需要注意每个人正在做什么。

避免冲突的最简单方法是防止更新React组件。你可以通过渲染React没有理由更新的元素来做到这一点,比如空白

如何解决这个问题

为了演示这一点,让我们勾勒一个通用jQuery插件的包装。

我们将附加一个ref到根DOM元素。在里面componentDidMount,我们会得到一个引用,所以我们可以将它传递给jQuery插件。

为了防止触摸安装后的DOM反应,我们会返回一个空

render()方法。该
元素没有属性或子元素,因此React没有理由更新它,让jQuery插件可以自由地管理DOM的这一部分:

class SomePlugin extends React.Component { componentDidMount() { this.$el = $(this.el); this.$el.somePlugin(); } componentWillUnmount() { this.$el.somePlugin('destroy'); } render() { return 
this.el = el} />; }}

请注意,我们定义了两个componentDidMountcomponentWillUnmount生命周期钩子。许多jQuery插件将事件监听器附加到DOM,因此将它们分开是非常重要的componentWillUnmount。如果插件没有提供清理方法,则可能需要提供自己的插件,记住删除插件注册的任何事件侦听器,以防止内存泄漏。

与jQuery选择插件集成

对于这些概念的更具体的例子,我们来为插件Chosen写一个最小包装,它增加了DOM节点上调用它,它会从原始DOM节点读取属性,将其隐藏为内联样式,然后在其后面附加一个单独的DOM节点,并在其后面附带自己的可视化表示裹着

class Chosen extends React.Component { render() { return ( 
); } }

注意我们我们传递给它的节点之后追加另一个DOM元素。但是,就React而言,

总是只有一个孩子。这就是我们如何确保React更新不会与由Chosen附加的额外DOM节点发生冲突的原因。重要的是,如果您在React流的外部修改DOM,则必须确保React没有理由触摸这些DOM节点。

接下来,我们将实现生命周期挂钩。我们需要初始化选中的参考 this.el = el}>

这足以让我们的组件渲染,但我们也希望得到关于值更改的通知。为此,我们将订阅由Chosen管理的jQuerychange事件,但我们还将添加一个componentDidUpdate()生命周期挂钩,通知Chosen关于子列表中的更改:

componentDidUpdate(prevProps) { if (prevProps.children !== this.props.children) { this.$el.trigger("chosen:updated"); }}

这样,当 this.el = el}> {this.props.children}

); } }

在CodePen上试用它。

与其他视图库集成

由于灵活性,React可以嵌入到其他应用程序中ReactDOM.render()

虽然React在启动时通常用于将单个根React组件加载到DOM中,ReactDOM.render()但也可以为UI的独立部分多次调用,该部分可以像按钮一样小,也可以与应用程序一样大。

事实上,这正是Facebook如何使用React。这让我们可以在React中逐个编写应用程序,并将其与我们现有的服务器生成的模板和其他客户端代码结合使用。

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

旧Web应用程序中的一种常见模式是将DOM的块描述为字符串,并将其插入到DOM中,如下所示:$el.html(htmlString)。代码库中的这些点非常适合引入React。只需将基于字符串的渲染重写为React组件即可。

所以下面的jQuery实现...

$('#container').html('');$('#btn').click(function() { alert('Hello!');});

...可以使用React组件重写:

function Button() { return ;}ReactDOM.render( 

从这里开始,您可以开始将更多逻辑转移到组件中,并开始采用更常见的React实践。例如,在组件中,最好不要依赖ID,因为可以多次渲染相同的组件。相反,我们将使用React事件系统,并将点击处理程序直接注册到React

function Button(props) { return ; }function HelloButton() { function handleClick() { alert('Hello!'); } return 

在CodePen上试用它。

您可以拥有任意数量的此类隔离组件,并使用ReactDOM.render()它们将它们呈现给不同的DOM容器。逐渐地,当您将更多应用程序转换为React时,您将能够将它们组合成更大的组件,并将一些ReactDOM.render()调用移动到层次结构中。

在骨干视图中嵌入React

主干视图通常使用HTML字符串或字符串生成模板函数来为其DOM元素创建内容。这个过程也可以用渲染React组件来替换。

下面,我们将创建一个名为Backbone的视图ParagraphView。它将覆盖Backbone的render()函数,将React组件渲染到由Backbone(this.el)提供的DOM元素中。在这里,我们也在使用ReactDOM.render()

function Paragraph(props) { return 

{props.text}

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

在CodePen上试用它。

我们也呼吁是非常重要ReactDOM.unmountComponentAtNode()remove方法,以便作出反应注销事件处理程序,并与组件树相关的其他资源,当它被分离。

当一个组件一个React树中被移除时,清理会自动执行,但因为我们要手动移除整个树,所以我们必须把它称为这个方法。

与模型层集成

虽然通常建议使用单向数据流,例如React状态,Flux或Redux,但React组件可以使用其他框架和库中的模型层。

在React组件中使用Backbone模型

使用React组件的Backbone模型和集合的最简单方法是侦听各种更改事件并手动强制更新。

负责渲染模型的组件将监听'change'事件,而负责渲染集合的组件将监听'add''remove'事件。在这两种情况下,都需要this.forceUpdate()使用新数据调用组件。

在下面的示例中,List组件呈现Backbone集合,使用该Item组件呈现单个项目。

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 
  • {this.props.model.get('text')}
  • ; } } 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 (
      {this.props.collection.map(model => ( )) }
    ); } }

    在CodePen上试用它。

    从主干模型中提取数据

    上述方法要求您的React组件知道Backbone模型和集合。如果您后来计划迁移到另一个数据管理解决方案,则可能需要尽可能少地将有关Backbone的知识集中在代码中。

    解决这个问题的一个办法是在模型的属性发生变化时将模型的属性作为普通数据提取出来,并将这个逻辑放在一个地方。以下是一个高阶组件,它将Backbone模型的所有属性提取到状态中,并将数据传递给包装组件。

    这样,只有高阶组件需要了解Backbone模型内部,并且应用程序中的大多数组件都可以不依赖于Backbone。

    在下面的例子中,我们将复制模型的属性以形成初始状态。我们订阅change事件(并取消订阅卸载),当它发生时,我们用模型的当前属性更新状态。最后,我们确保如果model道具本身发生变化,我们不会忘记退订旧模型,并订阅新模型。

    请注意,这个例子并不意味着在使用Backbone方面是详尽的,但它应该给你一个关于如何以一种通用的方式来解决这个问题的想法:

    function connectToBackboneModel(WrappedComponent) { return class BackboneComponent extends React.Component { constructor(props) { super(props); this.state = Object.assign({}, props.model.attributes); this.handleChange = this.handleChange.bind(this); } componentDidMount() { this.props.model.on('change', this.handleChange); } componentWillReceiveProps(nextProps) { this.setState(Object.assign({}, nextProps.model.attributes)); if (nextProps.model !== this.props.model) { this.props.model.off('change', this.handleChange); nextProps.model.on('change', this.handleChange); } } componentWillUnmount() { this.props.model.off('change', this.handleChange); } handleChange(model) { this.setState(model.changedAttributes()); } render() { const propsExceptModel = Object.assign({}, this.props); delete propsExceptModel.model; return ; } } }

    为了演示如何使用它,我们将NameInputReact组件连接到Backbone模型,并在firstName每次输入更改时更新其属性:

    function NameInput(props) { return ( 


    My name is {props.firstName}.

    );} const BackboneNameInput = connectToBackboneModel(NameInput);function Example(props) { function handleChange(e) { model.set('firstName', e.target.value); } return ( );} const model = new Backbone.Model({ firstName: 'Frodo' }); ReactDOM.render( , document.getElementById('root'));

    在CodePen上试用它。

    这项技术不限于Backbone。您可以通过订阅其生命周期挂钩中的更改并将数据复制到本地React状态,从而将React用于任何模型库。

    Article précédent: Article suivant: