探讨Web组件化的实现,组件化的Web王国

致我们必然组件化的Web

2015/11/25 · HTML5 · 1
评论 ·
组件化

初稿出处:
AlloyTeam   

这篇小说将从两年前的三回技术争议起来。争辨的聚焦便是下图的多个目录分层结构。笔者说按模块划分好,他说您傻逼啊,当然是按财富划分。

ca亚洲城88 1 《=》ca亚洲城88 2

”按模块划分“目录结构,把近来模块下的持有逻辑和能源都放一块了,那对于五人独立开发和保证个人模块不是很可以吗?当然了,那顶牛的结果是自家婴孩地改回主流的”按财富划分“的目录结构。因为,没有做到JS模块化和财富模块化,仅仅物理地点上的模块划分是没有意思的,只会增添创设的基金而已。

尽管她说得好有道理我无言以对,不过本身心不甘,等待她近日端组件化成熟了,再来第一回大战!

如今日就是自家一再正义的光阴!只是那时候越发跟你撕逼的人不在。

模块化的欠缺

模块一般指能够独立拆分且通用的代码单元。由于JavaScript语言本身没有放置的模块机制(ES6有了!!),大家一般会使用CMD或ADM建立起模块机制。现在半数以上不怎么大型一点的类型,都会采取requirejs也许seajs来贯彻JS的模块化。多少人分工同盟开发,其各自定义依赖和暴光接口,维护功用模块间独立性,对于项目标支出成效和项目早先时期扩大和护卫,都是是有非常大的鼎力相助意义。

但,麻烦大家不怎么略读一下底下的代码

JavaScript

require([
‘Tmpl!../tmpl/list.html’,’lib/qqapi’,’module/position’,’module/refresh’,’module/page’,’module/net’
], function(listTmpl, QQapi, Position, Refresh, Page, NET){ var foo =
”, bar = []; QQapi.report(); Position.getLocaiton(function(data){
//… }); var init = function(){ bind();
NET.get(‘/cgi-bin/xxx/xxx’,function(data){ renderA(data.banner);
renderB(data.list); }); }; var processData = function(){ }; var bind =
function(){ }; var renderA = function(){ }; var renderB =
function(data){ listTmpl.render(‘#listContent’,processData(data)); };
var refresh = function(){ Page.refresh(); }; // app start init(); });

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
require([
    ‘Tmpl!../tmpl/list.html’,’lib/qqapi’,’module/position’,’module/refresh’,’module/page’,’module/net’
], function(listTmpl, QQapi, Position, Refresh, Page, NET){
    var foo = ”,
        bar = [];
    QQapi.report();
    Position.getLocaiton(function(data){
        //…
    });
    var init = function(){
        bind();
        NET.get(‘/cgi-bin/xxx/xxx’,function(data){
            renderA(data.banner);
            renderB(data.list);
        });
    };
    var processData = function(){
    };
    var bind = function(){
    };
    var renderA = function(){
    };
    var renderB = function(data){
        listTmpl.render(‘#listContent’,processData(data));
    };
    var refresh = function(){
        Page.refresh();
    };
    // app start
    init();
});

上边是有血有肉有个别页面包车型地铁主js,已经封装了像Position,NET,Refresh等成效模块,但页面包车型地铁主逻辑依然是”面向进程“的代码结构。所谓面向进度,是指根据页面包车型大巴渲染进程来编排代码结构。像:init
-> getData -> processData -> bindevent -> report -> xxx

方法之间线性跳转,你大约也能感受那样代码弊端。随着页面逻辑更是复杂,那条”进度线“也会愈来愈长,并且越来越绕。加之缺少专业约束,别的种类成员依据各自需求,在”进度线“加插各自逻辑,最后这一个页面包车型大巴逻辑变得难以保险。

ca亚洲城88 3

支出须要谨慎,生怕影响“进度线”前边符合规律逻辑。并且每一次加插或改动都以bug泛滥,无不令产品有关人员一律惶惶不安。

 页面结构模块化

基于下面的面向进程的题材,行行业内部也有许多化解方案,而我们团队也总计出一套成熟的缓解方案:Abstractjs,页面结构模块化。我们能够把我们的页面想象为一个乐高机器人,须要分歧零件组装,如下图,假诺页面划分为tabContainer,listContainer和imgsContainer多少个模块。最后把这一个模块add到结尾的pageModel里面,最后接纳rock方法让页面运转起来。

ca亚洲城88 4
(原经过线示例图)

ca亚洲城88 5
(页面结构化示例图)

下边是伪代码的兑现

JavaScript

require([
‘Tmpl!../tmpl/list.html’,’Tmpl!../tmpl/imgs.html’,’lib/qqapi’,’module/refresh’,’module/page’
], function(listTmpl, imgsTmpl, QQapi, Refresh, Page ){ var
tabContainer = new RenderModel({ renderContainer: ‘#tabWrap’, data: {},
renderTmpl: “<li soda-repeat=’item in
data.tabs’>{{item}}</li>”, event: function(){ // tab’s event }
}); var listContainer = new ScrollModel({ scrollEl: $.os.ios ?
$(‘#Page’) : window, renderContainer: ‘#listWrap’, renderTmpl:
listTmpl, cgiName: ‘/cgi-bin/index-list?num=1’, processData:
function(data) { //… }, event: function(){ // listElement’s event },
error: function(data) { Page.show(‘数据重回卓殊[‘ + data.retcode +
‘]’); } }); var imgsContainer = new renderModel({ renderContainer:
‘#imgsWrap’, renderTmpl: listTmpl, cgiName: ‘/cgi-bin/getPics’,
processData: function(data) { //… }, event: function(){ //
imgsElement’s event }, complete: function(data) { QQapi.report(); } });
var page = new PageModel();
page.add([tabContainer,listContainer,imgsContainer]); page.rock(); });

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
require([
    ‘Tmpl!../tmpl/list.html’,’Tmpl!../tmpl/imgs.html’,’lib/qqapi’,’module/refresh’,’module/page’
], function(listTmpl, imgsTmpl, QQapi, Refresh, Page ){
 
    var tabContainer = new RenderModel({
        renderContainer: ‘#tabWrap’,
        data: {},
        renderTmpl: "<li soda-repeat=’item in data.tabs’>{{item}}</li>",
        event: function(){
            // tab’s event
        }
    });
 
    var listContainer = new ScrollModel({
        scrollEl: $.os.ios ? $(‘#Page’) : window,
        renderContainer: ‘#listWrap’,
        renderTmpl: listTmpl,
        cgiName: ‘/cgi-bin/index-list?num=1’,
        processData: function(data) {
            //…
        },
        event: function(){
            // listElement’s event
        },
        error: function(data) {
            Page.show(‘数据返回异常[‘ + data.retcode + ‘]’);
        }
    });
 
    var imgsContainer = new renderModel({
        renderContainer: ‘#imgsWrap’,
        renderTmpl: listTmpl,
        cgiName: ‘/cgi-bin/getPics’,
        processData: function(data) {
            //…
        },
        event: function(){
            // imgsElement’s event
        },
        complete: function(data) {
           QQapi.report();
        }
    });
 
    var page = new PageModel();
    page.add([tabContainer,listContainer,imgsContainer]);
    page.rock();
 
});

我们把那些常用的呼吁CGI,处理多少,事件绑定,上报,容错处理等一文山会海逻辑格局,以页面块为单位封装成一个Model模块。

诸如此类的三个华而不实层Model,大家得以清晰地见到该页面块,请求的CGI是哪些,绑定了怎么风波,做了何等上报,出错怎么处理。新增的代码就相应放置在对应的模块上相应的事态方法(preload,process,event,complete…),杜绝了往年的无规则乱增代码的创作。并且,遵照差异工作逻辑封装差异体系的Model,如列表滚动的ScrollModel,滑块成效的SliderModel等等,能够开始展览中度封装,集中优化。

近期基于Model的页面结构开发,已经包罗一点”组件化“的味道。种种Model都饱含各自的多寡,模板,逻辑。已经算是3个完好无缺的效果单元。但距离真正的WebComponent依旧有一段距离,至少满足不断笔者的”理想目录结构“。

 WebComponents 标准

咱俩想起一下选用2个datapicker的jquery的插件,所急需的步奏:

  1. 引入插件js

  2. 引入插件所需的css(假使有)

  3. copy 组件的所需的html片段

  4. 丰盛代码触发组件运维

眼前的“组件”基本上只可以落得是某些意义单元上的聚集。他的财富都以松散地分散在二种财富文件中,而且组件成效域揭露在全局意义域下,缺少内聚性很不难就会跟其余零件发生抵触,如最简易的css命名争论。对于那种“组件”,还不如下面的页面结构模块化。

于是乎W3C按耐不住了,制定1个WebComponents标准,为组件化的前途引导了明路。

上面以较为简单的不二法门介绍这份正经,力求大家能够赶快精通达成组件化的情节。(对那有的询问的同室,能够跳过这一小节)

1. <template>模板能力

模板那东西浙大学家最熟知可是了,后年见的较多的沙盘品质大战artTemplate,juicer,tmpl,underscoretemplate等等。而以后又有mustachejs无逻辑模板引擎等新入选手。但是我们有没有想过,这么基础的能力,原生HTML5是不援助的(T_探讨Web组件化的实现,组件化的Web王国。T)。

而明日WebComponent将要提供原生的模版能力

XHTML

<template id=”datapcikerTmpl”>
<div>我是原生的模板</div> </template>

1
2
3
<template id="datapcikerTmpl">
<div>我是原生的模板</div>
</template>

template标签钦定义了myTmpl的沙盘,需求动用的时候就要innerHTML= document.querySelector('#myTmpl').content;能够见到那几个原生的模版够原始,模板占位符等职能都不曾,对于动态数据渲染模板能力只可以自力更新。

2. ShadowDom 封装组件独立的内部结构

ShadowDom能够知道为一份有单独成效域的html片段。这几个html片段的CSS环境和主文书档案隔断的,各自笔者保护持内部的独立性。也多亏ShadowDom的独立本性,使得组件化成为了说不定。

JavaScript

var wrap = document.querySelector(‘#wrap’); var shadow =
wrap.createShadowRoot(); shadow.innerHTML = ‘<p>you can not see me
</p>’

1
2
3
var wrap = document.querySelector(‘#wrap’);
var shadow = wrap.createShadowRoot();
shadow.innerHTML = ‘<p>you can not see me </p>’

在实际dom节点上采取createShadowRoot方法即可生成其ShadowDom。就像是在整份Html的屋子里面,新建了二个shadow的房间。房间外的人都不知晓房间内有如何,保持shadowDom的独立性。

3. 自定义原生标签

第①接触Angularjs的directive指令功用,设定好组件的逻辑后,三个<Datepicker
/>就能引入整个组件。如此狂炫酷炸碉堡天的法力,实在令人弹冠相庆,跃地三尺。

JavaScript

var tmpl = document.querySelector(‘#datapickerTmpl’); var
datapickerProto = Object.create(HTMLElement.prototype); //
设置把我们模板内容我们的shadowDom datapickerProto.createdCallback =
function() { var root = this.createShadowRoot();
root.appendChild(document.importNode(tmpl.content, true)); }; var
datapicker = docuemnt.registerElement(‘datapicker’,{ prototype:
datapickerProto });

1
2
3
4
5
6
7
8
9
10
11
12
var tmpl = document.querySelector(‘#datapickerTmpl’);
var datapickerProto = Object.create(HTMLElement.prototype);
 
// 设置把我们模板内容我们的shadowDom
datapickerProto.createdCallback = function() {
    var root = this.createShadowRoot();
    root.appendChild(document.importNode(tmpl.content, true));
};
 
var datapicker = docuemnt.registerElement(‘datapicker’,{
    prototype: datapickerProto
});

Object.create方式持续HTMLElement.prototype,得到二个新的prototype。当解析器发现大家在文书档案中标记它将检查是还是不是1个名为createdCallback的办法。假若找到那个方法它将登时运行它,所以大家把克隆模板的剧情来创制的ShadowDom。

末段,registerElement的艺术传递大家的prototype来注册自定义标签。

地点的代码发轫略显复杂了,把前面多个力量“模板”“shadowDom”结合,形成组件的里边逻辑。最后通过registerElement的章程注册组件。之后方可愉悦地<datapicker></datapicker>的使用。

4. imports化解组件间的信赖性

XHTML

<link rel=”import” href=”datapciker.html”>

1
<link rel="import" href="datapciker.html">

以此类php最常用的html导入功能,HTML原生也能支撑了。

WebComponents标准内容大概到那边,是的,笔者那里没有啥样德姆o,也并未实践经验分享。由于webComponents新特色,基本上除了高版本的Chrome帮忙外,其余浏览器的支撑度甚少。就算有polymer帮助拉动webcompoents的仓库储存在,不过polymer自己的须要版本也是可怜高(IE10+)。所从前天的支柱并不是她。

大家简要来回看一下WebCompoents的四部分功用:

1 .<template>定义组件的HTML模板能力

  1. Shadow Dom封装组件的内部结构,并且维持其独立性

  2. Custom Element 对外提供组件的价签,完成自定义标签

  3. import化解组件结合和依赖加载

 组件化实践方案

官方的规范看完了,大家思想一下。一份真正成熟可相信的组件化方案,要求全部的力量。

“财富高内聚”—— 组件能源内部高内聚,组件资源由作者加载控制

“成效域独立”—— 内部结构密封,不与全局或任何零件发生影响

“自定义标签”—— 定义组件的行使方法

“可相互结合”—— 组件正在有力的地点,组件间组装整合

“接口规范化”—— 组件接口有联合标准,也许是生命周期的保管

私家认为,模板能力是基础力量,跟是不是组件化没有强联系,所以没有提出三个大点。

既然如此是实践,现阶段WebComponent的支持度还不成熟,不能作为方案的手腕。而其它一套以高品质虚拟Dom为切入点的零部件框架React,在facebook的造势下,社区得到了大力发展。别的一名骨干Webpack,负责化解组件能源内聚,同时跟React非常切合形成补充。

所以【Webpack】+【React】将会是这套方案的大旨技术。

不驾驭你未来是“又是react+webpack”感到失望ca亚洲城88 6,依旧“太好了是react+webpack”不用再学叁次新框架的开心ca亚洲城88 7。无论怎样上面包车型地铁始末不会让你失望的。

一,组件生命周期

ca亚洲城88 8

React天生正是强制性组件化的,所以能够从根个性上消除面向进度代码所带来的麻烦。React组件自己有生命周期方法,能够知足“接口规范化”能力点。并且跟“页面结构模块化”的所封装抽离的多少个章程能挨个对应。别的react的jsx自带模板成效,把html页面片直接写在render方法内,组件内聚性尤其严酷。

是因为React编写的JSX是会先生成虚拟Dom的,供给时机才真正插入到Dom树。使用React必要求理解组件的生命周期,其生命周期四个景况:

Mount: 插入Dom

Update: 更新Dom

Unmount: 拔出Dom

mount那单词翻译扩充,嵌入等。笔者倒是提议“插入”更好精通。插入!拔出!插入!拔出!默念二遍,懂了没?别少看黄段子的能力,

ca亚洲城88 9

零件状态便是: 插入-> 更新 ->拔出。

下一场种种组件状态会有两种处理函数,一前一后,will函数和did函数。

componentWillMount()  准备插入前

componentDidlMount()  插入后

componentWillUpdate() 准备更新前

componentDidUpdate()  更新后

componentWillUnmount() 准备拔出前

因为拔出后为主都是贤者形态(作者说的是组件),所以并未DidUnmount这一个格局。

别的React此外贰个主干:数据模型props和state,对应着也有自个状态方法

getInitialState()     获取初阶化state。

getDefaultProps() 获取暗中同意props。对于那个没有父组件传递的props,通过该办法设置暗中同意的props

componentWillReceiveProps()  已插入的零部件收到新的props时调用

还有叁个非正规景况的处理函数,用于优化处理

shouldComponentUpdate():判断组件是不是须求update调用

添加最重点的render方法,React自个儿带的章程刚刚好拾个。对于初学者的话是相比较为难消化。但骨子里getInitialStatecomponentDidMountrender八个状态方法都能不负众望半数以上零部件,不必惧怕。

回去组件化的宗旨。

1个页面结构模块化的零部件,能独立包装整个组件的进度线

ca亚洲城88 10

小编们换算成React生命周期方法:

ca亚洲城88 11

 

零件的情形方法流中,有两点必要尤其表达:

1,贰次渲染:

出于React的虚构Dom性格,组件的render函数不需自个儿触发,依照props和state的改动自个通过差别算法,得出最优的渲染。

请求CGI一般都以异步,所以一定带来三遍渲染。只是空数据渲染的时候,有可能会被React优化掉。当数码回来,通过setState,触发3遍render

 

2,componentWiillMount与componentDidMount的差别

和抢先八分之四React的教程文章分裂,ajax请求笔者提出在威尔Mount的方式内推行,而不是组件初步化成功现在的DidMount。那样能在“空数据渲染”阶段在此以前请求数据,尽早地回落一遍渲染的时日。

willMount只会实施一遍,万分适合做init的工作。

didMount也只会执行三遍,并且那时候真实的Dom已经形成,卓殊适合事件绑定和complete类的逻辑。

 

 二,JSX极丑,可是组件内聚的首要性!

WebComponents的科班之一,须求模板能力。本是觉得是我们熟练的模板能力,但React中的JSX那样的怪物照旧令人切磋纷繁。React还没有火起来的时候,我们就已经在今日头条上狠狠地吐槽了“JSX写的代码这TM的丑”。这实际上只是Demo阶段JSX,等到实战的大型项目中的JSX,包蕴多情况多数据多事件的时候,你会发觉………….JSX写的代码依旧极难看。

ca亚洲城88 12
(即便用sublime-babel等插件高亮,逻辑和渲染耦合一起,阅读性依旧略差)

何以我们会认为丑?因为大家早已经对“视图-样式-逻辑”分离的做法潜移默化。

据他们说维护性和可读性,甚至质量,大家都不建议直接在Dom下面绑定事件依然直接写style属性。我们会在JS写事件代理,在CSS上写上classname,html上的就是清楚的Dom结构。大家很好地珍视着MVC的设计格局,一切有惊无险。直到JSX把她们都夹杂在联合署名,所守护的技巧栈受到侵袭,难免存有抗拒。

 

唯独从组件化的指标来看,那种高内聚的做法未尝不可。

ca亚洲城88,上边包车型地铁代码,在此以前的“逻辑视图分离”情势,大家供给去找相应的js文件,相应的event函数体内,找到td-info的class所绑定的风云。

对照起JSX的莫斯中国科学技术大学学内聚,全数事件逻辑就是在自个儿jsx文件内,绑定的正是本身的showInfo方法。组件化的特色能即时展现出来。

(注意:纵然写法上大家好像是HTML的内联事件处理器,不过在React底层并没有实际赋值类似onClick属性,内层依旧采纳类似事件代理的措施,高效地维护着事件处理器)

再来看一段style的jsx。其实jsx没有对体制有硬性规定,大家完全可比照此前的定义class的逻辑。任何一段样式都应当用class来定义。在jsx你也完全能够如此做。然则出于组件的独立性,作者提议部分只有“一次性”的样式直接行使style赋值更好。减弱冗余的class。

XHTML

<div className=”list” style={{background: “#ddd”}}> {list_html}
</div>

1
2
3
<div className="list" style={{background: "#ddd"}}>
   {list_html}
</div>

可能JSX内部有担当繁琐的逻辑样式,可JSX的自定义标签能力,组件的黑盒性立马能感受出来,是还是不是一念之差美好了许多。

JavaScript

render: function(){ return ( <div> <Menus
bannerNums={this.state.list.length}></Menus> <TableList
data={this.state.list}></TableList> </div> ); }

1
2
3
4
5
6
7
8
render: function(){
    return (
      <div>
         <Menus bannerNums={this.state.list.length}></Menus>
         <TableList data={this.state.list}></TableList>
      </div>
   );
}

虽说JSX本质上是为了虚拟Dom而准备的,但那种逻辑和视图中度合一对于组件化未尝不是一件善事。

 

上学完React这些组件化框架后,看看组件化能力点的形成景况

“资源高内聚”—— (33%)  html与js内聚

“作用域独立”—— (百分之五十)  js的效能域独立

“自定义标签”—— (百分之百)jsx

“可相互结合”—— (二分一)  可整合,但缺乏可行的加载情势

“接口规范化”—— (百分之百)组件生命周期方法

 

Webpack 财富组件化

对此组件化的财富独立性,一般的模块加载工具和营造流程视乎变得辛苦。组件化的创设筑工程程化,不再是在此之前我们广阔的,css合二,js合三,而是体验在组件间的依赖于加载关系。webpack正好适合供给点,一方面填补组件化能力点,另一方帮衬大家健全组件化的完全营造环境。

率先要表美赞臣(Meadjohnson)点是,webpack是三个模块加载打包工具,用于管理你的模块财富正视打包难题。那跟大家熟习的requirejs模块加载工具,和grunt/gulp塑造筑工程具的概念,多多少少有些出入又有个别雷同。

ca亚洲城88 13

率先webpak对于CommonJS与速龙同时辅助,满意我们模块/组件的加载形式。

JavaScript

require(“module”); require(“../file.js”); exports.doStuff = function()
{}; module.exports = someValue;

1
2
3
4
require("module");
require("../file.js");
exports.doStuff = function() {};
module.exports = someValue;

JavaScript

define(“mymodule”, [“dep1”, “dep2”], function(d1, d2) { return
someExportedValue; });

1
2
3
define("mymodule", ["dep1", "dep2"], function(d1, d2) {
    return someExportedValue;
});

自然最精锐的,最非凡的,当然是模块打包功用。那便是这一职能,补充了组件化财富信赖,以及完整工程化的能力

依照webpack的宏图理念,全部财富都是“模块”,webpack内部贯彻了一套财富加运载飞机制,能够把想css,图片等能源等有依靠关系的“模块”加载。那跟我们采纳requirejs那种单纯处理js大大不一致。而那套加运载飞机制,通过一个个loader来实现。

 

JavaScript

// webpack.config.js module.exports = { entry: { entry: ‘./index.jsx’,
}, output: { path: __dirname, filename: ‘[name].min.js’ }, module:
{ loaders: [ {test: /\.css$/, loader: ‘style!css’ }, {test:
/\.(jsx|js)?$/, loader: ‘jsx?harmony’, exclude: /node_modules/},
{test: /\.(png|jpg|jpeg)$/, loader: ‘url-loader?limit=10240’} ] } };

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// webpack.config.js
module.exports = {
    entry: {
     entry: ‘./index.jsx’,
    },
    output: {
        path: __dirname,
        filename: ‘[name].min.js’
    },
    module: {
        loaders: [
            {test: /\.css$/, loader: ‘style!css’ },
            {test: /\.(jsx|js)?$/, loader: ‘jsx?harmony’, exclude: /node_modules/},
            {test: /\.(png|jpg|jpeg)$/, loader: ‘url-loader?limit=10240’}
        ]
    }
};

地点一份简单的webpack配置文件,留意loaders的配备,数组内三个object配置为一种模块财富的加运载飞机制。test的正则为合营文件规则,loader的为匹配到文件将由什么加载器处理,两个电脑之间用相隔,处理顺序从右到左。

 

style!css,css文件通过css-loader(处理css),再到style-loader(inline到html)的加工处理流。

jsx文件通过jsx-loader编写翻译,‘?’开启加载参数,harmony协助ES6的语法。

图片能源通过url-loader加载器,配置参数limit,控制少于10KB的图样将会base64化。

 财富文件怎么样被require?

JavaScript

// 加载组件本人css require(‘./slider.css’); // 加载组件正视的模块 var
Clip = require(‘./clipitem.js’); // 加载图片财富 var spinnerImg =
require(‘./loading.png’);

1
2
3
4
5
6
// 加载组件自身css
require(‘./slider.css’);
// 加载组件依赖的模块
var Clip = require(‘./clipitem.js’);
// 加载图片资源
var spinnerImg = require(‘./loading.png’);

在webpack的js文件中大家除了require大家正常的js文件,css和png等静态文件也能够被require进来。大家经过webpack命令,编写翻译之后,看看输出结果什么:

JavaScript

webpackJsonp([0], { /* 0 */ /***/ function(module, exports,
__webpack_require__) { // 加载组件本人css
__webpack_require__(1); // 加载组件正视的模块 var Clip =
__webpack_require__(5); // 加载图片财富 var spinnerImg =
__webpack_require__(6); /***/ }, /* 1 */ /***/
function(module, exports, __webpack_require__) { /***/ }, /* 2
*/ /***/ function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(3)();
exports.push([module.id, “.slider-wrap{\r\n position: relative;\r\n
width: 100%;\r\n margin: 50px;\r\n background:
#fff;\r\n}\r\n\r\n.slider-wrap li{\r\n text-align:
center;\r\n line-height: 20px;\r\n}”, “”]); /***/ }, /* 3 */
/***/ function(module, exports) { /***/ }, /* 4 */ /***/
function(module, exports, __webpack_require__) { /***/ }, /* 5
*/ /***/ function(module, exports) { console.log(‘hello, here is
clipitem.js’) ; /***/ }, /* 6 */ /***/ function(module, exports)
{ module.exports = “……” /***/ }
]);

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
webpackJsonp([0], {
/* 0 */
/***/ function(module, exports, __webpack_require__) {
          // 加载组件自身css
          __webpack_require__(1);
          // 加载组件依赖的模块
          var Clip = __webpack_require__(5);
          // 加载图片资源
          var spinnerImg = __webpack_require__(6);
/***/ },
/* 1 */
/***/ function(module, exports, __webpack_require__) {
 
/***/ },
/* 2 */
/***/ function(module, exports, __webpack_require__) {
          exports = module.exports = __webpack_require__(3)();
          exports.push([module.id, ".slider-wrap{\r\n position: relative;\r\n width: 100%;\r\n margin: 50px;\r\n background: #fff;\r\n}\r\n\r\n.slider-wrap li{\r\n text-align: center;\r\n line-height: 20px;\r\n}", ""]);
 
/***/ },
/* 3 */
/***/ function(module, exports) {
 
/***/ },
 
/* 4 */
/***/ function(module, exports, __webpack_require__) {
/***/ },
 
/* 5 */
/***/ function(module, exports) {
          console.log(‘hello, here is clipitem.js’) ;
/***/ },
/* 6 */
/***/ function(module, exports) {
          module.exports = "……"
/***/ }
]);

webpack编写翻译之后,输出文件视乎乱糟糟的,但骨子里每一个能源都被封装在2个函数体内,并且以编号的花样标记(注释)。这几个模块,由webpack的__webpack_require__当中方法加载。入口文件为编号0的函数index.js,能够阅览__webpack_require__加载别的编号的模块。

css文件在编号1,由于应用css-loader和style-loader,编号1-4都以处理css。个中编号2大家能够看大家的css的string体。最后会以内联的艺术插入到html中。

图表文件在编号6,能够看出exports出base64化的图纸。

 组件一体输出

JavaScript

// 加载组件本人css require(‘./slider.css’); // 加载组件信赖的模块 var
React = require(‘react’); var Clip = require(‘../ui/clipitem.jsx’); //
加载图片财富 var spinnerImg = require(‘./loading.png’); var Slider =
React.createClass({ getInitialState: function() { // … },
componentDidMount: function(){ // … }, render: function() { return (
<div> <Clip data={this.props.imgs} /> <img
className=”loading” src={spinnerImg} /> </div> ); } });
module.exports = Slider;

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
// 加载组件自身css
require(‘./slider.css’);
// 加载组件依赖的模块
var React = require(‘react’);
var Clip = require(‘../ui/clipitem.jsx’);
// 加载图片资源
var spinnerImg = require(‘./loading.png’);
var Slider = React.createClass({
    getInitialState: function() {
        // …
    },
    componentDidMount: function(){
        // …
    },
    render: function() {
        return (
            <div>
               <Clip data={this.props.imgs} />
               <img className="loading" src={spinnerImg} />
            </div>
        );
    }
});
module.exports = Slider;

假设说,react使到html和js合为紧密。

那正是说丰盛webpack,两者结合一起的话。js,css,png(base64),html
全部web能源都能合成3个JS文件。那多亏这套方案的着力所在:零件独立一体化。假使要引用一个零件,仅仅require('./slider.js') 即可完成。

 

参与webpack的模块加载器之后,我们组件的加载问题,内聚难点也都成功地缓解掉

“能源高内聚”—— (百分百) 全部能源得以一js输出

“可相互结合”—— (百分百)  可结合可凭借加载

 

 CSS模块化实践

很欢欣,你能翻阅到此地。最近大家的组件达成度分外的高,能源内聚,易于组合,作用域独立互不污染。。。。等等ca亚洲城88 14,视乎CSS模块的实现度有不足。

那正是说方今组件达成度来看,CSS效用域其实是全局性的,并非组件内部独立。下一步,大家要做得就是何等让大家组件内部的CSS功用域独立。

那儿大概有人立即跳出,大喊一句“德玛西亚!”,哦不,应该是“用sass啊傻逼!”。可是项目组件化之后,组件的里边封装已经很好了,其内部dom结构和css趋向不难,独立,甚至是破碎的。LESS和SASS的一体式样式框架的统一筹划,他的嵌套,变量,include,函数等充分的效用对于全体大型项目标体制管理10分管用。但对于二个职能单一组件内部样式,视乎就变的有些冲突。“不可能为了框架而框架,合适才是最棒的”。视乎原生的css能力已经满意组件的体裁需要,唯独便是地点的css成效域难题。

 

那边笔者付出思考的方案:
classname随便写,保持原生的办法。编译阶段,依照组件在项目路线的唯一性,由【组件classname+组件唯一途径】打成md5,生成全局唯一性classname。正当作者要写一个loader达成本身的想法的时候,发现歪果仁已经早在先走一步了。。。。

那边具体方案参考作者从前博客的译文:

此前我们探讨过JS的模块。未来透过Webpack被加载的CSS能源叫做“CSS模块”?小编觉着依然至极的。未来style-loader插件的达成精神上只是创办link[rel=stylesheet]要素插入到document中。那种作为和常见引入JS模块卓殊不一致。引入另三个JS模块是调用它所提供的接口,但引入二个CSS却并不“调用”CSS。所以引入CSS自身对于JS程序来说并不设有“模块化”意义,纯粹只是表明了一种财富注重——即该零件所要完结的意义还索要或多或少asset。

故此,那位歪果仁还扩张了“CSS模块化”的概念,除了上边的大家要求部分功能域外,还有很多效果,那里不详述。具体参考原版的书文 

相当的赞的一点,正是cssmodules已经被css-loader收纳。所以大家不必要依靠额外的loader,基本的css-loader开启参数modules即可

JavaScript

//webpack.config.js … module: { loaders: [ {test: /\.css$/, loader:
‘style!css?modules&localIdentName=[local]__[name]_[hash:base64:5]’
}, ] } ….

1
2
3
4
5
6
7
8
//webpack.config.js
…  
    module: {
        loaders: [
            {test: /\.css$/, loader: ‘style!css?modules&localIdentName=[local]__[name]_[hash:base64:5]’ },
        ]  
    }
….

modules参数代表开启css-modules效率,loaclIdentName为设置大家编写翻译后的css名字,为了方便debug,我们把classname(local)和组件名字(name)输出。当然能够在终极输出的版本为了节省提交,仅仅使用hash值即可。其它在react中的用法大致如下。

JavaScript

var styles = require(‘./banner.css’); var Banner = new
React.createClass({ … render: function(){ return ( <div> <div
className={styles.classA}></div> </div> ) } });

1
2
3
4
5
6
7
8
9
10
11
var styles = require(‘./banner.css’);
var Banner = new React.createClass({
    …
    render: function(){
        return (
            <div>
                <div className={styles.classA}></div>
            </div>
        )
    }
});

最终那里关于出于对CSS一些考虑,

至于css-modules的别的成效,作者并不打算选取。在中间分享【大家竭尽所能地让CSS变得复杂】中提及:

咱俩项目中多数的CSS都不会像boostrap那样必要变量来安装,身为一线开发者的大家大概能够感受到:设计师们改版UI,相对不是简单的换个色或改个间距,而是万物更新的全新UI,那纯属不是贰个变量所能消除的”维护性“。

反倒项目实战进程中,真正要化解的是:在本子迭代进度中那二个淘汰掉的过期CSS,多量地堆放在类型个中。大家像极了家中的欧巴酱不舍得丢掉没用的事物,因为那只是大家应用sass或less编写出具有莫斯中国科学技术大学学的可维护性的,肯定有复用的一天。

那个堆积的逾期CSS(or
sass)之间又有一部分信赖,一部分超时失效了,一部分又被新的体裁复用了,导致没人敢动那多个历史样式。结果现网项目迭代还带着多量两年前没用的体制文件。

组件化之后,css的布局同样被改善了。或许postcss才是你今后手上最适合的工具,而不在是sass。

 

到此处,大家算是把组件化最终一个标题也消除了。

“功用域独立”—— (百分百) 就好像shadowDom功效域独立

 

到此地,我们能够开一瓶82年的七喜,好好庆祝一下。不是吗?

ca亚洲城88 15

 

 组件化之路还在持续

webpack和react还有不少新分外首要的天性和效能,介于本文仅仅围绕着组件化的为焦点,没有各样演讲。此外,配搭gulp/grunt补充webpack创设能力,webpack的codeSplitting,react的机件通讯难题,开发与生产条件安插等等,都是百分之百大型项目方案的所必须的,限于篇幅难题。能够等等笔者更新下篇,或大家可以自动查阅。

可是,不得不再安利一下react-hotloader神器。热加载的花费格局相对是下一代前端开发必备。严酷说,如若没有了热加载,作者会很坚决地扬弃那套方案,即便那套方案再怎么好好,小编都讨厌react供给5~6s的编写翻译时间。可是hotloader能够在自小编不刷新页面包车型大巴情况下,动态修改代码,而且不单单是样式,连逻辑也是即时生效。

ca亚洲城88 16

如上在form表单内。使用热加载,表单不必要再度填写,修改submit的逻辑霎时见效。那样的费用效用真不是拉长仅仅一个水平。必须安利一下。

 

也许你意识,使用组件化方案未来,整个技术栈都被更新了一番。学习成本也不少,并且能够预见到,基于组件化的前端还会过多供不应求的题材,例如品质优化方案必要再度思考,甚至最核心的机件可复用性不肯定高。后边十分短一段时间,供给大家不断磨练与优化,探求最优的前端组件化之道。

最少大家得以设想,不再担心自个儿写的代码跟有些何人什么人冲突,不再为找某段逻辑在多少个文件和措施间穿梭,不再copy一片片逻辑然后改改。大家每一趟编写都是可选拔,可整合,独立且内聚的组件。而各类页面将会由3个个嵌套组合的零部件,相互独立却相互效率。

 

对此如此的前端今后,有所指望,不是很行吗

于今,多谢你的开卷。

1 赞 6 收藏 1
评论

ca亚洲城88 17

内容提要

接纳过多独门组件创设应用程序的想法并不卓绝。Web
Component的出现,是再次回想基于组件的应用程序开发方式的好机会。我们得以从那一个进程中收益,掌握怎么样使用现有技术做到目标,并且在今后做出自个儿的前端Web应用。
 

CMS组件化,不难架构示意图:

本文由 埃姆杰 翻译。未经许可,禁止转发! 英文出处:Future Insights。

怎么是组件?

软件开发是三个语义充裕(术语平时持续一个情趣)的园地。很醒目,那里的“组件”是三个很泛的称之为,所以有供给指明我们想要表明的,在前者Web应用的语言环境中的意思。

前端Web应用中的组件,是指部分统一筹划为通用性的,用来营造较大型应用程序的软件,那个组件有四种表现方式。它能够是有UI(用户界面)的,也足以是当做
“服务”的纯逻辑代码。

因为有视觉上的表现情势,UI组件更便于领会。UI组件不难的例证包涵按钮、输入框和文本域。不论是波士顿包状的菜系按钮(无论你是还是不是喜欢)、标签页、日历、选项菜单恐怕所见即所得的富文本编辑器则是有的越发高档的事例。

提供服务类型的机件恐怕会令人为难精晓,那系列型的例子包罗跨浏览器的AJAX援助,日志记录或然提供某种数据持久化的功力。

依照组件开发,最重点的便是组件能够用来构成任何零件,而富文本编辑器正是个很好的例子。它是由按钮、下拉菜单和局地可视化组件等组成。另一个事例是HTML5上的video成分。它一样带有按钮,也还要富含3个能从录制数据流渲染内容的成分。

ca亚洲城88 18

内容提要

应用过多独立组件创设应用程序的想法并不新鲜。Web
Component的产出,是双重返忆基于组件的应用程序开发方式的好机会。大家能够从那一个进度中收益,通晓什么利用现有技术形成目的,并且在今后做出自身的前端Web应用。

为何要创设组件?

既然未来一度清楚组件的意味,就看看使用组件的章程塑造前端采取的利益。

Web组件使用WebPage+WebAPI的补益:

什么是组件?

软件开发是多个语义丰硕(术语日常持续贰个意味)的世界。很明显,那里的“组件”是二个很泛的称为,所以有供给指明大家想要表达的,在前者Web应用的言语环境中的意思。

前端Web应用中的组件,是指部分规划为通用性的,用来营造较大型应用程序的软件,这个零部件有种种表现格局。它能够是有UI(用户界面)的,也得以是作为
“服务”的纯逻辑代码。

因为有视觉上的表现方式,UI组件更便于领悟。UI组件不难的例子包罗按钮、输入框和文本域。不论是开普敦包状的菜谱按钮(无论你是否喜欢)、标签页、日历、选项菜单只怕所见即所得的富文本编辑器则是一对特别高档的例证。

提供服务类型的组件可能会令人难以知晓,那体系型的例子包含跨浏览器的AJAX援救,日志记录大概提供某种数据持久化的成效。

传说组件开发,最重视的正是组件能够用来组合任何零件,而富文本编辑器就是个很好的例子。它是由按钮、下拉菜单和局地可视化组件等构成。另3个事例是HTML5上的video成分。它一样包括按钮,也同时富含一个能从录制数据流渲染内容的成分。

模块

您可能听闻过 “组件是后天模块”的传教。好吧,谢谢它,我们又要诠释那里的术语!

您可能会以为“组件”的传教更是符合用来叙述UI,而“模块”更适合描述提供服务的法力逻辑。而对于作者来说,模块和零部件意思相近,都提供团体、聚焦和包裹,是与有个别意义单位有关的。

Ø  组件复用(组件条件管理页面复用+获取组件数据API复用)。

干什么要创设组件?

既是今后一度知晓组件的意思,就看看使用组件的点子营造前端接纳的裨益。

高内聚

又是3个软件工程的高频词! 我们将相关的有的效益公司在一起,把全体封装起来,而在组件的例证中,就大概是相关的作用逻辑和静态能源:JavaScript、HTML、CSS以及图像等。那便是大家所说的内聚。

那种做法将让组件更便于保障,并且这么做之后,组件的可相信性也将增加。同时,它也能让组件的法力鲜明,增大组件重用的大概。

Ø  组件是分布式的第③方应用,本人高内聚、组件之间松耦合,能够理解为单独的子系统。

模块

你大概传说过 “组件是原始模块”的布道。好啊,多谢它,大家又要解释那里的术语!

你只怕会认为“组件”的说教更为切合用来讲述UI,而“模块”更符合描述提供劳动的功能逻辑。而对于本人的话,模块和零部件意思相近,都提供组织、聚焦和包裹,是与有个别成效单位有关的。

可重用

您看看的演示组件,越发是Web
Component,更关心可选用的标题。成效鲜明,达成清晰,API易于通晓。自然就能促进组件复用。通过营造可选用组件,大家不但保持了 D奥德赛Y(不要再度造轮子)原则,还赢得了对应的补益。

此处要提示: 不要过度尝试营造可选取组件。你更应有关爱应用程序上所急需的那个特定部分。如若现在相应供给出现,或许零部件的确到了可采纳的境界,就花一点外加时间让组件重用。事实上,开发者都爱好去创建可选择功用块(库、组件、模块、插件等),做得太早将会让您后来优伤不堪。所以,吸取基于组件开发的其余利益,并且接受不是负有组件都能重用的真情。

Ø  组件使用WebAPI,提供REST风格的WebService,轻量级,对客户端的渴求也更少。

高内聚

又是2个软件工程的高频词! 我们将相关的有的职能公司在一起,把全体封装起来,而在组件的例子中,就或许是相关的效用逻辑和静态财富:JavaScript、HTML、CSS以及图像等。那便是我们所说的内聚。

那种做法将让组件更易于保证,并且这么做之后,组件的可相信性也将增长。同时,它也能让组件的法力分明,增大组件重用的大概。

可互换

一个功能肯定好组件的API能让人私自地改成其内部的效率实现。假使程序内部的零件是松耦合的,那其实能够用3个组件轻易地更迭另多个零部件,只要依据相同的 API/接口/约定。

若是你使用GoInstant提供的实时效益劳动组件,那她们前一周闭馆服务那般的音讯会影响到你。不过,只要提供了平等的多少同步API,你也得以活动营造利用三个 FirebaseComponent 组件可能 PubNubComponent 组件。

Ø  组件能够独自支出,单独测试,高度封装,可以分别不相同环境,别的组件完全能够不做其余测试。

可重用

你看到的以身作则组件,特别是Web
Component,更爱戴可选拔的难题。效能肯定,达成清晰,API易于明白。自然就能带动组件复用。通过构建可采取组件,大家不光保险了 D福特ExplorerY(不要再度造轮子)规范,还拿走了对应的功利。

那边要提示: 不要过度尝试营造可选择组件。你更应有关爱应用程序上所必要的那二个特定部分。假设之后相应要求应运而生,或然零部件的确到了可采用的程度,就花一点杰出时间让组件重用。事实上,开发者都喜欢去成立可选取功效块(库、组件、模块、插件等),做得太早将会让你后来忧伤不堪。所以,吸取基于组件开发的别样利益,并且接受不是有着组件都能重用的实际。

可组合

事先也钻探过,基于组件的架构让组件组合成新组件尤其便于。这样的统一筹划让组件特别注意,也让别的零件中营造和揭破的效用更好利用。

甭管是给程序添加效果,照旧用来制作完整的次第,越发扑朔迷离的效益也能照猫画虎。那就是那种艺术的重要利益。

是不是有必不可少把具有的东西转换成组件,事实上取决于你协调。没有任何理由让您的顺序由 你自己 的组件组合成你最惊叹的功能 ,乃至 最花哨的功能。而这个零部件又扭曲构成任何零件。假如你从那个主意中取得了利益,就想方设法地去坚贞不屈它。但是要注意的是,不要用平等的方法把工作变得复杂,你并不须求过分关怀怎样让组件重用。而是要爱抚展现程序的成效。

Ø  组件热插拔,组件易替换,可增加性强,不会有dll相互正视、影响,dll管理等一一日千里难题去处理。

可互换

3个作用显然好组件的API能令人自由地转移其内部的效率完成。假使程序内部的零件是松耦合的,那实在能够用一个零件轻易地更迭另叁个组件,只要遵守千篇一律的 API/接口/约定。

假定你使用GoInstant提供的实时效益劳动组件,这她们前一周闭馆服务那般的音信会影响到您。但是,只要提供了一致的数量同步API,你也可以活动创设利用二个 FirebaseComponent 组件也许 PubNubComponent 组件。

当今就从头创设组件

在 Caplin
Systems 营造基于组件的自有应用程序时,作者动用了几条原则和履行。那些条件由 BladeRunnerJS(BRJS) 开源工具集支撑。它被称作”BladeRunnerJS”
是因为我们将次第成效都封装在称作 Blades 的事物中。Blade是能够在某些应用中收录的功力特色,然而不能在程序间重用。当功能真的
变得更其通用的时候,大家将相应的定义移到库文件中,供各类程序间接选举拔。特定应用中的组件(blade)和大家先后间的通用组件能够行使,大家只要找到最佳知足须要的任何库和框架。

那么,未来怎么着库和框架能够协助大家营造组件呢?

在控制营造利用时应运用何种技术时,只必要探视流行的 TodoMVC 网站就能够看出多量可供采取的前端库和框架。你可能会认为任何一种方案都能用来创设基于组件的应用程序。可是,他们内部的一些方案内置了对组件的支撑。在那之中相比较盛名的是AngularJS、Ember
和 React。

Ø  .net/java/php/都得以付出组件,可多元化发展。

可组合

在此以前也探究过,基于组件的框架结构让组件组合成新组件越发便于。那样的统一筹划让组件特别专注,也让其他零件中营造和揭发的意义更好应用。

无论是是给程序添加效果,照旧用来创制完整的次序,越发错综复杂的功效也能生搬硬套。那就是那种办法的首要利益。

是还是不是有必不可少把具有的事物转换到组件,事实上取决于你本人。没有任何理由让您的次第由 你自己 的机件组合成你最惊叹的功能 ,乃至 最花哨的功能。而这一个零件又扭曲构成任何零件。假如你从这么些办法中收获了便宜,就想尽地去持之以恒它。不过要留心的是,不要用相同的措施把作业变得复杂,你并不须要过分关怀如何让组件重用。而是要爱抚显示程序的功力。

零件间是哪些通讯的?

在深入示例在此之前有必不可少简单地提到组件间通信的题材。如若组件之间是“独立”、“模块化”的,他们又是怎样相互通讯的啊?

最醒指标答案就是让组件间互为引用并通过他们中间的API交互。那样做的标题就在于,这种做法会让组件相互注重。短期内或许幸亏,一段时间今后,你在改动程序的时候程序会失控,修改3个组件就会对另叁个组件产生巨大的震慑。决定移除1个不可能带动预期价值组件或然会让你的应用程序停止工作,因为它背后会有数个零部件依赖于它。

那会儿,化解方案是提供松耦合的,让组件之间很少依旧差不离不知晓互相的方案。组件并不直接创建别的零件,在她们须求通讯的时候,他们通过“接口/约定”也许经过 “服务”。大家在创设B福特ExplorerJS程序时考虑了众多这几个方面包车型大巴事物,并且采用 ServiceRegistry 访问用于组件间通信的服务大概是Web
API那样的能源。Angular和Ember选拔了服务和借助注入消除那类难题。

Ø  组件都有谈得来独立的版本,组件可以独自编写翻译,独立包装和配置,当有个别组件有转移的时候,能够单独对转移的零件进行版本升级,单独对转移组件进行陈设。

今天就起初营造组件

在 Caplin
Systems 营造基于组件的自有应用程序时,小编利用了几条规则和执行。这几个规范由 BladeRunnerJS(BRJS) 开源工具集支撑。它被称作”BladeRunnerJS”
是因为大家将次第功效都封装在称作 Blades 的东西中。Blade是能够在某些应用中收音和录音的遵守特色,可是不得以在先后间重用。当作用真的
变得尤其通用的时候,我们将相应的定义移到库文件中,供各种程序间使用。特定应用中的组件(blade)和大家先后间的通用组件能够动用,我们若是找到最棒满意需要的任何库和框架。

那正是说,今后哪些库和框架能够扶助大家营造组件呢?

在支配营造利用时应采纳何种技术时,只须要探视流行的 TodoMVC 网站就能够见见多量可供选用的前端库和框架。你恐怕会以为任何一种方案都能用来营造基于组件的应用程序。但是,他们之中的一些方案内置了对组件的支持。当中相比较有名的是AngularJS、Ember
和 React。

以身作则组件my-avatar

为了显得大家怎样用这几个库和框架创设最基本的零件,大家建立了三个包罗UI,用于取回和展示用户头像的简要示例。在或许的景况下,该零件会有 my-avatar 标签,会从以下八个属性中获取头像:

  • service 允许设置3个劳动。例如 twitter 或者 facebook
  • username 用于取回该用户名相对应的头像

Ø  组件数据的生成会有一定的条件,组件自治,组件完结格局自由发挥。

组件间是如何通讯的?

在深切示例之前有必不可少简单地关乎组件间通讯的题目。假诺组件之间是“独立”、“模块化”的,他们又是怎么样互相通讯的吧?

最醒目的答案就是让组件间相互引用并经过他们中间的API交互。那样做的标题就在于,那种做法会让组件相互依赖。短时间内或然万幸,一段时间现在,你在改动程序的时候程序会失控,修改二个零件就会对另三个组件发生巨大的震慑。决定移除一个不能够推动预期价值组件大概会让你的应用程序结束工作,因为它背后会有数个零件正视于它。

那时,消除方案是提供松耦合的,让组件之间很少恐怕大致不知情相互的方案。组件并不直接创制其余零件,在他们要求通讯的时候,他们经过“接口/约定”恐怕通过 “服务”。我们在营造BXC90JS程序时考虑了广大那些方面包车型地铁事物,并且应用 ServiceRegistry 访问用于组件间通信的劳务如故是Web
API那样的能源。Angular和Ember选择了服务和借助注入化解那类难题。

AngularJS

AngularJS 可能是当今用于营造程序最流行的前端化解方案了。作为创笔者的谷歌,重新思考HTML,考虑怎么器重新发明,满意方今Web开发的必要。

Angular中得以选择自定义指令概念组件。之后,你能够接纳 HTML
标记表明自定义组件。

翻开代码演示: 

那几个事例突显了应用Angular指令的简易程度。值scope 定义了从
 my-avatar 成分中拿走,并且之后用来创设相应的img标签和渲染成用户头像的品质。

Ø  能够独自做测试,能够单独安插、运转、调试等。

示范组件my-avatar

为了显得大家如何用那么些库和框架创设最基本的组件,我们建立了三个富含UI,用于取回和体现用户头像的简约示例。在大概的处境下,该零件会有 my-avatar 标签,会从以下两性格情中收获头像:

  • service 允许设置多少个劳动。例如 twitter 或者 facebook
  • username 用于取回该用户名相对应的头像

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图