JSRUN 用代码说话

组件定义

编辑教程

组件定义

本章代码统一访问项目chapter4_components下,项目代码可以在以下网址上找到: https://github.com/ubuntuvim/my_emberjs_code

与之前的文章一样,项目仍然是使用Ember CLI命令创建项目和各个组件文件。

创建项目并测试运行,首先执行如下四条命令,最后在浏览器执行:http://localhost:4200/。

ember new chapter4_components
cd chapter4_components
ember server

如果在页面上看到Welcome to Ember说明项目框架搭建成功!

自定义组件及使用

创建组件方法很简单:ember generate component my-component-name。一条命令即可,但是需要注意的是组件的名称必须要包含中划线-,比如blog-post、test-component、audio-player-controls这种格式的命名是合法,但是post、test这种方式的命名是不合法的!其一是为了防止用户自定义的组件名与W3C规定的元素标签名重复;其二是为了确保Ember能自动检测到用户自定义的组件。

下面定义一个组件,ember g component blog-post。Ember CLI会自动为你创建组件对应的的模板,执行这条命令之后你可以在app/components和app/templates/components下看到创建的文件。

    <h1>{{title}}</h1>
    <p>{{yield}}</p>
    <p>Edit title: {{input type="text" value=title}}</p>

为了演示组件的使用需要做些准备工作: ember g route index

//  app/routes/index.js

import Ember from 'ember';

export default Ember.Route.extend({

    model: function() {
         return [
            { id: 1, title: 'Bower: dependencies and resolutions new', body: "In the bower.json file, I see 2 keys dependencies and resolutionsWhy is that so? I understand Bower has a flat dependency structure. So has it got anything to do with that ?", category: 'java' },
            { id: 2, title: 'Highly Nested JSON Payload - hasMany error', body: "Welcome to the Ember.js discussion forum. We're running on the open source, Ember.js-powered Discourse forum software. They are also providing the hosting for us. Thanks guys! Please use this space for discussion abo… read more", category: 'php' },
            { id: 3, title: 'Passing a jwt to my REST adapter new ', body: "This sets up a binding between the category query param in the URL, and the category property on controller:articles. In other words, once the articles route has been entered, any changes to the category query param in the URL will update the category property on controller:articles, and vice versa.", category: 'java'}
        ];

    }
});

{{#each model as |item|}}

    {{#blog-post title=item.title}}
        {{item.body}}
    {{/blog-post}}
{{/each}}

在这段代码中,使用了自定义的组件来显示数据。

自定义的组件被渲染到了模板index.hbs使用blog-post的地方。并且自定义组件的HTML标签没有变化。 到这里大概应该知道怎么去使用组件了,至于它是怎么就渲染到了使用组件的地方,以及它是怎么渲染上去的。别急~~后面的文章会为你一一解答。

说明:默认情况下,自定义的组件会被渲染到div标签内,当然这种默认情况也是可以修改的,比较简单在此不过多介绍,请自行学习,网址:customizing-a-components-element。

自定义组件类

用户自定义的组件类都需要继承Ember.Component类。

通常情况下我们会把经常使用的模板片段封装成组件,只需要定义一次就可以在项目任何一个模板中使用,而且不需要编写任何的javascript代码。比如上述第一点“自定义组件及使用”中描述的一样。

但是如果组件有特殊的行为,并且这些行为是默认组件类无法提供的(比如:改变包裹组件的标签、响应组件模板初始化某个状态等),那么此时你可以自定义组件类,但是要继承Ember.Component,如果你自定义的组件类没有继承这个类,你自定义的组件就很有可能会出现一些不可预知的问题。

Ember所能识别的自定义组件类的名称是有规范的。比如,你定义了一个名为blog-post的组件,那么你的组件类的名称应该是app/components/blog-post.js。如果组件名为audio-player-controls那么对应的组件类名为app/components/audio-player-controls.js。即:组件类名与组件同名,这个是v2.0的命名方法,请区别就版本的Ember,旧版本的组件命名规则是驼峰式的命名规则。

举个简单的例子,在第一点“自定义组件及使用”中讲过,组件默认会被渲染到div标签内,可以在组件类中修改这个默认标签。

//  app/components/blog-post.js

import Ember from 'ember';

export default Ember.Component.extend({
    tagName: 'nav'
});

动态渲染组件

组件的动态渲染与Java的多态有点相似。{{component}}助手会延迟到运行时才决定使用那个组件渲染页面。当程序需要根据数据不同渲染不同组件的时,这种动态渲染就显得特别有用。可以使你的逻辑和试图分离开。

那么要怎么使用呢?非常简单,只需要把组件名作为参数传递过去即可,比如:使用{{component 'blog-post'}}与{{blog-post}}结果是一致的。我们可以修改第一点“自定义组件及使用”实例中模板index.hbs的代码。

{{#each model as |item|}}

    {{component 'blog-post' title=item.title}}
    {{item.body}}
{{/each}}

页面刷新之后,可以看到结果是一样的。

下面演示如何根据数据不同渲染不同的组件。

按照惯例,先做好准备工作,使用Ember CLI命令创建2个不同的组件。

ember g component foo-component
ember g component bar-component

<h1>Hello from bar</h1>
<p>{{post.body}}</p>

为何能用post获取数据,因为在使用组件的地方传递了参数。在模板index.hbs中可以看到。

<h1>Hello from foo</h1>
<p>{{post.body}}</p>

修改显示的数据,注意数据的最后增加一个属性pn,pn的值就是组件的名称。

//  app/routes/index.js

import Ember from 'ember';

export default Ember.Route.extend({

    model: function() {
         return [
            { id: 1, title: 'Bower: dependencies and resolutions new', body: "In the bower.json file, I see 2 keys dependencies and resolutionsWhy is that so? I understand Bower has a flat dependency structure. So has it got anything to do with that ?", pn: 'bar-component' },
            { id: 2, title: 'Highly Nested JSON Payload - hasMany error', body: "Welcome to the Ember.js discussion forum. We're running on the open source, Ember.js-powered Discourse forum software. They are also providing the hosting for us. Thanks guys! Please use this space for discussion abo… read more", pn: 'foo-component' },
            { id: 3, title: 'Passing a jwt to my REST adapter new ', body: "This sets up a binding between the category query param in the URL, and the category property on controller:articles. In other words, once the articles route has been entered, any changes to the category query param in the URL will update the category property on controller:articles, and vice versa.", pn: 'bar-component'}
        ];

    }
});

修改调用组件的模板index.hbs。


{{#each model as |item|}}

    {{component item.pn post=item}}
{{/each}}

模板编译之后会得到形如{{component foo-component post}}的组件调用代码。

JSRUN闪电教程系统是国内最先开创的教程维护系统, 所有工程师都可以参与共同维护的闪电教程,让知识的积累变得统一完整、自成体系。 大家可以一起参与进共编,让零散的知识点帮助更多的人。
X
支付宝
9.99
无法付款,请点击这里
金额: 0
备注:
转账时请填写正确的金额和备注信息,到账由人工处理,可能需要较长时间
如有疑问请联系QQ:565830900
正在生成二维码, 此过程可能需要15秒钟