您的位置:新葡亰496net > 新葡亰官网 > 新葡亰496net:致我们终将组件化的Web,前端工程

新葡亰496net:致我们终将组件化的Web,前端工程

发布时间:2019-09-17 06:17编辑:新葡亰官网浏览(56)

    前端组件化开采实行

    2015/07/12 · CSS, HTML5, JavaScript · 组件化

    原作出处: 美团技艺博客 - spring   

    致大家确定组件化的Web

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

    最早的文章出处: AlloyTeam   

    那篇小说将从三年前的三回技能争论起来。争论的聚焦正是下图的五个目录分层结构。笔者说按模块划分好,他说您傻逼啊,当然是按资源划分。

    新葡亰496net 1 《=》新葡亰496net 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 。 方法之间线性跳转,你差不离也能感受那样代码缺欠。随着页面逻辑更是复杂,这条”进度线“也会更为长,而且更加的绕。加之缺乏职业约束,其余种类成员根据各自供给,在”进程线“加插各自逻辑,最后这些页面包车型地铁逻辑变得难以维护。

    新葡亰496net 3

    支付需求一笔不苟,生怕影响“进程线”后边符合规律逻辑。而且每二回加插或修改都以bug泛滥,无不令产品有关人口无不忧心忡忡。

     页面结构模块化

    依附上面的面向进度的标题,行当内也会有过多缓慢解决方案,而我们组织也总计出一套成熟的应用方案:Abstractjs,页面结构模块化。大家得以把大家的页面想象为二个乐高机器人,须求不一致零件组装,如下图,假若页面划分为tabContainer,listContainer和imgsContainer八个模块。最后把那一个模块add到结尾的pageModel里面,最终利用rock方法让页面运营起来。

    新葡亰496net 4
    (原经过线示例图)

    新葡亰496net 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都含有各自的数码,模板,逻辑。已经算是二个完全的作用单元。但离开真正的WebComponent依旧有一段距离,至少满意不断笔者的”理想目录结构“。

     WebComponents 标准

    咱俩想起一下选取一个datapicker的jquery的插件,所需求的步奏:

    1. 引进插件js

    2. 引进插件所需的css(借使有)

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

    4. 加上代码触发组件运转

    当前的“组件”基本上只可以达到是某些意义单元上的成团。他的能源都是松散地分散在二种财富文件中,而且组件作用域揭破在大局意义域下,紧缺内聚性很轻巧就能够跟别的零件产生争论,如最轻易易行的css命名争辩。对于这种“组件”,还不及上边的页面结构模块化。

    于是W3C按耐不住了,拟定二个WebComponents标准,为组件化的未来教导了明路。

    上面以较为轻便的点子介绍那份正经,力求大家可以急速理解完结组件化的内容。(对那某个打探的同窗,能够跳过这一小节)

    1. <template>模板本领

    模板那东西厦学院家最纯熟不过了,2018年见的非常多的模板品质战役artTemplate,juicer,tmpl,underscoretemplate等等。这段日子后又有mustachejs无逻辑模板引擎等新入选手。但是大家有未有想过,这么基础的力量,原生HTML5是不帮忙的(T_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 />就会引进整个组件。如此狂绚烂炸碉堡天的效用,实在令人大快人心,跃地三尺。

    新葡亰496net:致我们终将组件化的Web,前端工程化学习。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。当分析器发现大家在文档中标志它将检查是还是不是三个名字为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”感觉失望新葡亰496net 6,依然“太好了是react webpack”不用再学一回新框架的欣然自得新葡亰496net 7。无论怎么着下边包车型客车开始和结果不会令你失望的。

    一,组件生命周期

    新葡亰496net 8

    React天生就是强制性组件化的,所以能够从根个性上缓解面向进度代码所带来的分神。React组件自己有生命周期方法,能够满意“接口标准化”技巧点。何况跟“页面结构模块化”的所封装抽离的多少个法子能挨个对应。别的react的jsx自带模板成效,把html页面片间接写在render方法内,组件内聚性尤其紧凑。

    是因为React编写的JSX是会先生成虚构Dom的,必要机会才真正插入到Dom树。使用React一定要知道组件的生命周期,其生命周期七个状态:

    Mount: 插入Dom

    Update: 更新Dom

    Unmount: 拔出Dom

    mount那单词翻译扩张,嵌入等。小编倒是建议“插入”越来越好通晓。插入!拔出!插入!拔出!默念贰回,懂了没?别少看黄段子的本领,

    新葡亰496net 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四个情景方法都能做到大部分组件,不必惧怕。

    回去组件化的大旨。

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

    新葡亰496net 10

    大家换算成React生命周期方法:

    新葡亰496net 11

     

    组件的情况方法流中,有两点须求特别声明:

    1,叁次渲染:

    鉴于React的杜撰Dom天性,组件的render函数不需本人触发,依照props和state的改变自个通过差别算法,得出最优的渲染。

    恳请CGI一般皆以异步,所以确定带来三回渲染。只是空数据渲染的时候,有异常的大可能率会被React优化掉。当数码回来,通过setState,触发二回render

     

    2,componentWiillMount与componentDidMount的差别

    和大多数React的教程文章不均等,ajax央浼小编提议在WillMount的方法内施行,实际不是组件最早化成功未来的DidMount。这样能在“空数据渲染”阶段此前伏乞数据,尽早地减小叁回渲染的年月。

    willMount只会施行二回,特别适合做init的事体。

    didMount也只会施行一回,並且那时候真实的Dom已经形成,特别适合事件绑定和complete类的逻辑。

     

     二,JSX极丑,不过组件内聚的主要!

    WebComponents的专门的职业之一,供给模板本领。本是感到是大家纯熟的模版才具,但React中的JSX那样的怪人仍然令人切磋纷繁。React还尚无火起来的时候,我们就曾经在天涯论坛上狠狠地作弄了“JSX写的代码那TM的丑”。那事实上只是德姆o阶段JSX,等到实战的大型项目中的JSX,富含多情形非常多据多事件的时候,你会发觉………….JSX写的代码依旧非常不好看。

    新葡亰496net 12
    (尽管用sublime-babel等插件高亮,逻辑和渲染耦合一同,阅读性还是略差)

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

    遗闻维护性和可读性,以至质量,大家都不建议直接在Dom上边绑定事件依然直接写style属性。大家会在JS写事件代理,在CSS上写上classname,html上的便是显然的Dom结构。大家很好地掩护着MVC的设计形式,一切有惊无险。直到JSX把她们都夹杂在一道,所守护的本事栈受到凌犯,难免存有抗拒。

     

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

    下边包车型地铁代码,以前的“逻辑视图分离”格局,大家须求去找相应的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正好适合供给点,一方面填补组件化本领点,另一方援助大家健全组件化的完整营造意况。

    率先要表美素佳儿(Friso)点是,webpack是一个模块加载打包工具,用于管理你的模块能源依赖打包难点。这跟我们听得多了自然能详细说出来的requirejs模块加载工具,和grunt/gulp塑造筑工程具的定义,多多少少有个别出入又有一点雷同。

    新葡亰496net 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的为相称到文件将由哪些加载器管理,八个Computer之间用相隔,管理顺序从右到左。

     

    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{rn position: relative;rn width: 100%;rn margin: 50px;rn background: #fff;rn}rnrn.slider-wrap li{rn text-align: center;rn line-height: 20px;rn}", ""]); /***/ }, /* 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{rn position: relative;rn width: 100%;rn margin: 50px;rn background: #fff;rn}rnrn.slider-wrap li{rn text-align: center;rn line-height: 20px;rn}", ""]);
     
    /***/ },
    /* 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编写翻译之后,输出文件视乎乱糟糟的,但其实每一个财富都被封装在四个函数体内,并且以编号的款型标识(注释)。这一个模块,由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能源都能合成三个JS文件。那正是那套方案的基本所在:零件独立一体化。即使要援引多个零件,仅仅require('./slider.js') 就可以完结。

     

    加盟webpack的模块加载器之后,大家组件的加载难题,内聚难点也都工作有成地化解掉

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

    “可相互结合”—— (百分之百)  可组成可依赖加载

     

     CSS模块化执行

    很欢喜,你能读书到此地。前段时间我们的零部件达成度特别的高,财富内聚,易于组合,作用域独立互不污染。。。。等等新葡亰496net 14,视乎CSS模块的实现度有欠缺。

    那么方今组件完毕度来看,CSS功用域其实是全局性的,并非组件内部独立。下一步,大家要做得正是怎么让我们组件内部的CSS效能域独立。

    那会儿可能有人登时跳出,大喊一句“德玛西亚!”,哦不,应该是“用sass啊傻逼!”。可是类型组件化之后,组件的中间封装已经很好了,其内部dom结商谈css趋向轻巧,独立,以致是创痍满指标。LESS和SASS的一体式样式框架的安排性,他的嵌套,变量,include,函数等充足的职能对于全部大型项指标样式管理相当的实惠。但对此三个职能单一组件内部样式,视乎就变的有点格不相入。“无法为了框架而框架,合适才是最棒的”。视乎原生的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年的Sprite,好好庆祝一下。不是吧?

    新葡亰496net 15

     

     组件化之路还在承接

    webpack和react还也许有那些新相当的重大的个性和意义,介于本文仅仅围绕着组件化的为基本,未有各样解说。其余,配搭gulp/grunt补充webpack营造本领,webpack的codeSplitting,react的组件通讯难点,开荒与生产条件布置等等,都以一切大型项目方案的所不可不的,限于篇幅难题。能够等等小编更新下篇,或大家能够活动查阅。

    只是,不得不再安利一下react-hotloader神器。热加载的支出格局相对是下一代前端开荒必备。严谨说,一旦未有了热加载,笔者会很坚决地扬弃那套方案,就算那套方案再怎么好好,笔者都讨厌react供给5~6s的编写翻译时间。但是hotloader能够在自己不刷新页面包车型大巴动静下,动态修改代码,何况不单单是样式,连逻辑也是即时生效。

    新葡亰496net 16

    如上在form表单内。使用热加载,表单不须要再行填写,修改submit的逻辑立即见效。这样的花费功效真不是增加仅仅一个水平。必得安利一下。

     

    大概你意识,使用组件化方案今后,整个本事栈都被更新了一番。学习花费也相当多,何况能够预言到,基于组件化的前端还有恐怕会点不清相差的难点,比如性能优化方案须要再一次思量,以致最主旨的组件可复用性不分明高。前边相当短一段时间,必要大家不停锻练与优化,探究最优的前端组件化之道。

    最少大家得以想像,不再忧虑本身写的代码跟某些何人什么人冲突,不再为找某段逻辑在三个文件和艺术间穿梭,不再copy一片片逻辑然后改改。大家每一遍编写都以可选取,可组成,独立且内聚的组件。而各类页面将会由贰个个嵌套组合的机件,互相独立却相互成效。

     

    对此如此的前端未来,有所指望,不是很好呢

    迄今停止,谢谢您的读书。

    1 赞 6 收藏 1 评论

    新葡亰496net 17

    摘要


    • 后端懒人一枚,一时供给搞个管理端,曾在这个学校有用jquery bootstrap做过类似的前端项目,不过现在想起,旧代码如屎一坨,更别提维护了。有了事先的教训,境遇那几个必要就决定认真做做。
    • 对前面一个照旧略小白,写得倒霉,还望多多指教。
    • 已做到的项目应用的咬合是react webpack redux-form(帮忙ie8)

    正文介绍seajs和hanlebars,并落实前端控件组件化.

    前面叁个发展与现状

    我们都清楚前端是由HTML、CSS、Js组成的,一早先那样写出来的页面,无法某个加载,复用性相当差,重复工作非常多。微软就生产了ifram标签,便是一对一于在网页中嵌套三个网页,切换目录只是切换ifram中的网页,照旧一贯加载有些完整的html分界面。接着ajax的出现,完结了有的刷新,优化了客商体验。后来跻身了jQuery时期,jQuery封装了比很多原生方法,收缩了代码量。以后我们前端进入了内外端分离时代,流行 MV* 框架(MVC、MVP、MVVM),MVVM框架有Angular、Vue、React。

    新葡亰496net 18

    MVVM框架

    近来大家后台管理体系是依照Vue开拓的单页面应用(SPA)。

    前言

    一人管理器前辈曾说过:

    Controlling complexity is the essence of computer programming

    1
    Controlling complexity is the essence of computer programming

    乘机前端开辟复杂度的逐年升级,组件化开辟应际而生,并乘胜 FIS、React 等天时地利框架的出现各处开花。这一进程同样发生在美团,面对工作范围的敏捷腾飞和程序员团队的再三扩充,我们历经引进组件化解决财富整合难题、稳步进步组件功用推动开辟效用、重新创设新一代组件化方案适应全栈开垦和分享一起创建等阶段,努力“controlling complexity”。本文将介绍大家组件化开拓的实施进度。

    背景与主题素材


    • Header,Banner,Footer等前端布局的集体部分,重复编写代码或借助php等后端语言实行渲染,会导致前后端代码高耦合。
    • 任凭全局变量的存在,给项目扩充带来未知难题。
    • 乘机项目开拓持续开展,项目也会变得臃肿,不好的项目结构会使有限支撑工作更加的困难。复杂的页面须求调节越来越多情形,并根据情状的转移,试行更加的多相应的逻辑。
    • 本着管理端,举行ie8 的卓绝。
    • 使用的第三方库相当多,前端发送的静态文件(js,css)诉求非常多,导致网页加载速度一点也不快。
    • 管理平台的体裁多有同样,若样式定制开垦,会发生不供给的专门的学问量。

    自个儿想完成的是把组件单唯三个文本夹,js和css,html都坐落文件夹中,要动用组件的页面使用seajs引进组件,并平昔调用组件的不二诀窍只怕事件。这些方案是团结的合计,招待大家拍砖。

    Vue简介

    1、Vue.js是二个创设数据驱动的web框架
    2、Vue.js达成了数据的双向绑定和组件化
    3、Vue.js只须求关周到据的变化,不需求繁琐的收获和操作dom
    例如给二个成分绑定事件并赋值,jQuery的做法是:

    <input class="ipt" type="text">
    <button class="btn"></button>
    <script type="text/javascript">
    $.ready(function () {
            $('.ipt').value();
            $('.btn').click(function() {    
            })
     })
    </script>
    

    vue的写法是:

    <input class="ipt" v-model="value" type="text">
    <button class="btn" @click="click"></button>
    

    .vue文件的写法

    <template>
        这里写HTML
    </template>
    <script type="text/ecmascript-6">
        这里写数据和方法
    </script>
    <style lang="stylus" rel="stylesheet/stylus">
        这里写css
    </style>
    

    组件化 1.0:能源整合

    在美团开始时期,前端财富是遵照页面可能类似事情页面集合的情势打开公司的。比如order.js 对应订单相关页面包车型地铁并行,account.css 对应账户相关页面包车型地铁体制。这种方法在过去的较长一段时间内,持续补助了一切项指标例行推动,居功至伟。

    新葡亰496net 19

    趁着业务规模的加码和支出公司的恢宏,那套机制稳步展现出它的部分欠缺:

    • 能源冗余页面包车型客车逐月加多,交互的逐年复杂化,导致对应的 css 和 js 都有高大增进,进而现身为了借助某个 js 中的贰个函数,供给加载整个模块,或然为了利用有个别 css 中的部分样式正视整个 css,冗剩余资金源非常多
    • 对应提到不直观未有刚强的相应法规,导致的多个难点是修改某些业务模块的 css 也许 js 时,差不离只可以借助 grep。靠人来维护页面模块 html、css 和 js 之间的正视关系,轻巧犯错,平时出现内容早就去除不过 css 或 js 还留存的标题
    • 难于单元测验以页面为最小粒度进行能源整合,区别功用的业务模块相互影响,复杂度太高,自动化测量检验难以推进

    二零一一 年开始,在调查研商了 FIS、BEM 等方案未来,结合美团开垦框架的莫过于,大家开端实现了一套轻量级的组件化开拓方案。首要的订正是:

    • 以页面效果组件为单位聚合前端财富
    • 自行加载符合约定的 css、js 财富
    • 将业务数据到渲染数据的转换进程独立出来

    新葡亰496net 20

    比喻来讲,美团顶端的寻觅框就被完成为三个零部件。

    新葡亰496net 21

    代码构成:

    www/component/smart-box/ ├── smart-box.js # 交互 ├── smart-box.php # 渲染数据生产、组件配置 ├── smart-box.scss # 样式 ├── smart-box.tpl # 内容 └── test ├── default.js # 自动化测验 └── default.php # 单测页面

    1
    2
    3
    4
    5
    6
    7
    8
    www/component/smart-box/
    ├── smart-box.js    # 交互
    ├── smart-box.php   # 渲染数据生产、组件配置
    ├── smart-box.scss  # 样式
    ├── smart-box.tpl   # 内容
    └── test
        ├── default.js  # 自动化测试
        └── default.php # 单测页面

    调用组件变得丰裕轻巧:

    JavaScript

    echo View::useComponent('smart-box', [ 'keyword' => $keyword ]);

    1
    2
    3
    echo View::useComponent('smart-box', [
        'keyword' => $keyword
    ]);

    对待在此之前,能够看到组件化的一对风味:

    • 按需加载只加载要求的前端财富
    • 对应提到十分清晰组件所要求的前端财富都在平等目录,任务鲜明且独一,对应提到明显
    • 轻易测验组件是全数独立表现和相互的微小单元,可应用 Phantom 等工具自动化测量检验

    其余,由于前端财富集中开展调节,组件化也为高阶品质优化提供了空中。举个例子落到实处组件级其余BigRender、通过数量深入分析进行资源的统中兴载等等。

    关键词


    依赖以前对背景和主题素材的分析,此番做前端的机要正是完成以下3个重大词。

    • 模块化
    • 组件化
    • 工程化

    重要歌唱家 seajs 和 handlebars

    花色布局

    ├── build // 创设相关
    │ ├── build.js
    │ ├── check-versions.js
    │ ├── dev-client.js
    │ ├── dev-server.js
    │ ├── utils.js
    │ ├── vue-loader.conf.js
    │ ├── webpack.base.conf.js
    │ ├── webpack.dev.conf.js
    │ └── webpack.prod.conf.js
    ├── comm// 打包后生成的目录
    │ ├── favicon.ico
    │ ├── index.html
    │ └── static
    ├── config // 配置相关
    │ ├── dev.env.js
    │ ├── index.js
    │ └── prod.env.js
    ├── package.json //开垦生产正视
    ├── server //服务及mock数据
    │ ├── controller
    │ └── mock
    ├── src //源代码
    │ ├── App.vue // 入口页面
    │ ├── api // 全部供给
    │ ├── assets // 字体等静态能源
    │ ├── common // 全局公用方法
    │ ├── components // 全局公用组件
    │ ├── favicon.ico
    │ ├── index.html // html模板
    │ ├── main.js // 入口js 加载组件 开始化等
    │ ├── pages // 所有页面
    │ ├── plugins // 全局工具
    │ ├── router // 路由
    │ └── store // 全局store管理
    └── static // 第三方不打包能源
    ├── async.js
    ├── css
    ├── img
    ├── jquery-1.8.3.min.js
    ├── jquery.ztree.js
    ├── md5.js
    ├── paopao.js
    ├── spark-md5.js
    ├── tinymce
    ├── upload.js
    ├── upyun-mu.js
    └── vue-style-loader

    零件化 2.0:趋于成熟

    组件化 1.0 上线后,由于轻便易用,异常快获得程序员的确定,并开头在各个事情中选拔起来。新的必要接踵而至 蜂拥而至,向来不停到 二〇一四 年初,那些品级大家称为组件化 2.0。上边介绍下首要的多少个创新。

    标题及对应施工方案


    seajs新葡亰496net,是三个前端模块化js库,功效类似于requirejs,都用来前端js的模块化和按需依靠加载的做事。功用都大概,seajs是天猫玉伯开采的,国产,所以国内使用者相当多,requirejs国外的,存在时间比seajs要久广大,海外用的可比多。 两者最大的界别在于定义模块的语法分裂,二个是AMD标准,三个是CMD规范。seajs是CMD标准,所以他的语法更像nodejs,写起来会比requirejs优雅一些。但是劣势也相当多,比方非常多第三方库都以依照英特尔标准写的,会有包容性难点,就须求本身改写模块大概利用spm工具管理。

    这里来简单讲一下src文件

    Lifecycle

    零件在高内聚的还要,往往必要暴光一些接口供外部调用,进而能够适应复杂的页面须求,比方提交订单页面需求在付出密码组件运维成功后绑定提交时的反省。Web Components、React 等都选用了生命周期事件/方法,大家也是大同小异。

    组件的生命周期:

    新葡亰496net 22多少个零部件的完好生命周期包涵:

    • init,初叶化组件根节点和配备
    • fetch,加载 css 和 js 资源
    • render,内容渲染,暗中认可的渲染内容措施是 BigRender
    • ready,进行数量绑定等操作
    • update,数据更新
    • destroy,解除全部事件监听,删除全数组件节点

    组件提供 pause、resume 方法以福利实行生命周期调节。种种阶段采纳 Promise 串行举行,异步的军管更清楚。使用自定义语义事件,在改换默许行为、组件间通讯上丰硕利用了 YUI 强大的自定义事件系列,有效减弱了开支爱戴资金财产。

    比如,页面早先化时组件的起步进度实际上也是借助生命周期达成的:

    JavaScript

    var afterLoadList = []; Y.all('[data-component]').each(function (node) { var component = new Y.mt.Component(node); // 绑定 init 生命周期事件,在 init 私下认可行为成功后实践回调 component.after('init', function (e) { // 假如布置了延期运转 if (e.config.afterLoad) { // 暂停组件生命周期 e.component.pause(); // 压入延迟运行数组 afterLoadList.push(e.component); } }); // 早先踏向生命周期 component.start(); }); Y.on('load', function () { // 在页面 load 事件发生时回涨组件生命周期 afterLoadList.forEach(function (component) { component.resume(); }); });

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    var afterLoadList = [];
    Y.all('[data-component]').each(function (node) {
        var component = new Y.mt.Component(node);
        // 绑定 init 生命周期事件,在 init 默认行为完成后执行回调
        component.after('init', function (e) {
            // 如果配置了延迟启动
            if (e.config.afterLoad) {
                // 暂停组件生命周期
                e.component.pause();
                // 压入延迟启动数组
                afterLoadList.push(e.component);
            }
        });
        // 开始进入生命周期
        component.start();
    });
     
    Y.on('load', function () {
        // 在页面 load 事件发生时恢复组件生命周期
        afterLoadList.forEach(function (component) {
            component.resume();
        });
    });

    回过头来看,引进生命周期除了带来扩充性外,更着重的是理顺了组件的逐条阶段,有利于越来越好的领会和动用。

    标题:前后端代码高耦合 全局变量

    handlebars是三个前端js模板引擎 作者在前头的稿子介绍过handlebars及轻松的用法,我们能够去看下点我

    api 和 views

    笔者们公司后台项目近来大要有贰11个api模块。随着业务的迭代,模块会非常多。所以那边依据职业模块来划分pages,何况将pages 和 api 七个模块一一对应,方便维护,如下图

    新葡亰496net 23

    aip和pages.png

    如此不管项目怎么累加,api和pages相比较好爱抚。

    Data Binding

    多少绑定是我们期盼已久的遵守,将 View 和 ViewModel 之间的互动自动化无疑会省掉工程师的大方时光。在组件化减少关切点和减低复杂度后,完结多少绑定变得进一步可能。

    大家最终兑现的数额绑定方案重要仿照效法了 Angular,通过在 html 节点上加多特定的习性注解绑定逻辑,js 扫描这么些内容并扩充对应的渲染和事件绑定。当数码产生变化时,对应的内容全方位再一次渲染。

    XHTML

    <ul class="addressList"> <li mt-bind-repeat="addr in addrList" mt-bind-html="addr.text" > </li> </ul> <script> Y.use(['mt-bind', 'mt-scope'], function () { Y.mt.bind.init(document.body); var scope = Y.one('.addressList').getScope(); // 将 scope.addrList 设置为三个数组,DOM 上将电动渲染其剧情 scope.$set('addrList', [ { text: "first address" }, { text: "second address" } ]); }); </script>

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    <ul class="addressList">
        <li
            mt-bind-repeat="addr in addrList"
            mt-bind-html="addr.text"
        >
        </li>
    </ul>
     
    <script>
    Y.use(['mt-bind', 'mt-scope'], function () {
        Y.mt.bind.init(document.body);
        var scope = Y.one('.addressList').getScope();
        // 将 scope.addrList 设置为一个数组,DOM 上将自动渲染其内容  
        scope.$set('addrList', [
            { text: "first address" },
            { text: "second address" }
        ]);
    });
    </script>

    采纳性质证明绑定逻辑的收益是足以同一时间支持后端渲染,那对于美团团购那样的偏表现型业务是不行要求的,客户能够急速看到页面内容。

    方案:模块化


    先是,将前端的公物部分提抽出来,实行模块化编制程序;
    接下来,划定功效域,差别模块间效率域不分享,那样就能够达成,公共代码模块化,防止了全局变量的麻烦。

    现阶段,流行的模块化编制程序的工具备RequireJS, SeaJS, Webpack,在那之中SeaJS是由Ali的团伙开拓的,在中原有活泼的社区。Webpack由英国人编写,体现了越来越强的品种工程化特点。
    中间,RequireJS和SeaJS分别是AMD和CMD三种规范的意味。英特尔推崇注重前置,在概念模块的时候将在注明其借助的模块;CMD推崇异步注重加载的,独有在应用某些模块的时候再去require。

    RequireJS(AMD写法)

    SeaJS(CMD写法)

    而Webpack则属于集大成者,不仅仅包容两个书写标准,况兼因接纳异步IO及类别缓存,使Webpack在增量编译上更加快。但骨子里,Webpack在其实支出中越多的是充当编写翻译者和打包者的角色,它整合React一样可以落到实处模块化开采,这些大家在前边体现。

    职分1,页面框架配置,seajs开首化

    components

    此地的components放置的都以大局公用的部分零件,如上传组件,富文本等等。

    新葡亰496net 24

    components.png

    Flux

    落到实处数量绑定后,我们只好面前遭受别的贰个难题:怎么样同步八个零件间的数码。因为有些组件的数额变化,很有希望孳生别的零件的生成。比方当修改购买数码,总金额会生成,而总金额凌驾500 后,还索要出示大数额花费提示。

    为了缓和这一个主题素材,大家引进了 Flux,使用全局信息总线的笔触展开跨组件交互。

    举例说因为相互复杂而直接让大家特别头痛的花色购买页,在动用组件 Flux 重构后,各模块之间的交互愈加清晰:

    新葡亰496net 25

    其余方面包车型客车勘误还会有为数非常的多,满含引进模板引擎 LightnCandy约束模板逻辑、支持组件放肆嵌套、扶助异步加载并活动发轫化等。

    趁着组件化 2.0 的稳步健全,基本已经得以从容应对日常开销,在效用和质感点面都上了多少个阶梯。

    难点:不好的档期的顺序结构,调控越来越多的情况

    新建html页面,引进需求的类库

    store

    vex要依靠供给去选拔,大家后台项目以来,尽管事情模块相当多,还会有权力,但事情之间的耦合度是非常低的,所以根本无需采纳vuex来存款和储蓄data,各样页面里寄存本人的data就行。当然有些数据依然供给用vuex来归并保管的,如登陆,顾客新闻,照旧用vuex管理有助于。

    组件化 3.0:重启征程

    时间的车轮滚滚前行,2015 年终,大家相见一些新的火候和挑衅:

    • 依靠 Node 的全栈开拓情势最初采纳,前后端渲染有了越来越多的可能性
    • YUI 结束维护,须求一套新的资源管理方案
    • 新专门的职业持续加码,须要找到一种组件共享的措施,幸免重新造轮子

    组合以前的执行,以及在这一进度中逐步积攒的对行业内部方案的回味,大家建议了新的组件化方案:

    • 据他们说 React 开荒页面组件,使用 NPM 实行分发,方便一起创建共享
    • 依赖 Browserify 贰回开拓,建设能源打包工具 Reduce,方便浏览器加载
    • 建设适应组件化开荒方式的工程化开荒方案 Turbo,方便程序猿将零件应用于业务支出中

    方案:组件化


    组件化创建在模块化的底蕴上,模块化是对财富的保管,组件化是对作业逻辑的管制,各自要有两样的前进方向。经调查研商,NPM,Webpack(或Browserify)和React的组合会便于我们对项目布局有二个安然无恙的认知。在这么些基础上,大家便能拾叁分便利地开创二个组织越发分明的体系。

    而针对性差别组件所具有的不仅退换的场所,通过对Flux和Redux的上学和行使,能够方便我们对于复杂组件状态管理的机制,有越来越了然。(但因为本身到底不是前面一个,且较懒,关于redux,未做过多学习,直接上手redux-form是为着快速支付管理端表单。)这里要提一下,redux-form这些强大,官方已提交比比较多例子,这里不再熬述。

    Flux

    Redux

    seajs用于做js模块管理,seajs-text能够用来加载handlebars的模版内容

    Router

    路由那几个概念最初是在后台现身的,浏览器发出央求,服务器依照央求,分析url路线,依照服务器的路由配置,再次回到相应 html 字串。大家前端路由的贯彻精神上正是检查评定 url 的变化,截获 url 地址,然后深入分析来相称路由法则,每便 hash 值的改动,会触发 hashchange 这一个事件,通过轮换 DOM 的秘籍来兑现页面包车型大巴切换,还或者有通过HTML5的三个api,pushState 和 replaceState完毕记住路由。

    React

    在组件化 2.0 的进度中,大家开采众多效果与利益和 React 重合,譬如 Data Binding、Lifecycle、前后端渲染,乃至一向借鉴的 Flux。除却,React 的函数式编制程序观念、增量更新、兼容性杰出的事件种类也让咱们极其赞佩。借着前端全栈开辟的主要关头,大家发轫思虑基于 React 举办零部件化 3.0 的建设。

    问题:IE8兼容

    配置seajs

    router-view

    <router-view> 是用来渲染路线匹配到的零件。<router-view> 仍是能够内嵌<router-view>,达成路由嵌套。

     <keep-alive>
            <router-view v-if="$route.meta.keepAlive"></router-view>
     </keep-alive>
    

    NPM Reduce

    NPM Reduce 构成了作者们新的能源管理方案,个中:

    • NPM 担负组件的公布和安装。可以认为是“分”的长河,粒度越小,重用的或者越大
    • Reduce 负担将页面能源拓宽包装。能够以为是“合”的历程,让浏览器更快地加载

    贰个天下无双的组件包:

    smart-box/ ├── package.json # 组件包元消息 ├── smart-box.jsx # React Component ├── smart-box.scss # 样式 └── test └── main.js # 测试

    1
    2
    3
    4
    5
    6
    smart-box/
    ├── package.json    # 组件包元信息
    ├── smart-box.jsx   # React Component
    ├── smart-box.scss  # 样式
    └── test
        └── main.js     # 测试

    NPM 暗许只帮助 js 文件的保管,大家对 NPM 中的 package.json 进行了扩张,扩充了 style 字段,以使打包工具 Reduce 也能够对 css 和 css 中援引的 image、font 进行甄别和拍卖:

    XHTML

    { "style": "./smart-box.scss" }

    1
    2
    3
    {
        "style": "./smart-box.scss"
    }

    假如在页面中 require 了 smart-box,经过 Reduce 打包后,js、css 以致图片、字体,都会现出在浏览器中。

    JavaScript

    var SmartBox = require('@mtfe/smart-box'); // 页面 var IndexPage = React.createClass({ render: function () { return ( <Header> <SmartBox keyword={ this.props.keyword } /> </Header> ... ); } }); module.exports = IndexPage;

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    var SmartBox = require('@mtfe/smart-box');
    // 页面
    var IndexPage = React.createClass({
        render: function () {
            return (
                <Header>
                    <SmartBox keyword={ this.props.keyword } />
                </Header>
                ...
            );
        }
    });
    module.exports = IndexPage;

    总体思路和零部件化 1.0 一模二样,却又那么分化。

    方案:React(0.14.x)


    最一齐初调研的时候,因为ie包容的忧郁,所以,调查研商了比较流行的React,Angular,Vue以及小众的Avalon。在那之中后八个都以MVVM框架(大旨是数量双向绑定)。关于包容,Angular早在1.3本子就撇下了对ie8的援助,Vue就没希图扶助ie8,可面前蒙受中国1/5的ie8客户,只万幸Avalon和React做取舍。

    MVVM

    Avalon是由去哪儿前端框架结构师“司徒正美”开采的一款基于设想DOM与本性威逼的精美、易用、高质量的前端MVVM框架,适用于各样地方,包容各类古老刁钻浏览器,吸取最新的本事成果,能高效堆砌组件与使用。它乃至协理到IE6 。

    avalon2

    包容性和个性好外,短处就是除了文书档案和十分小的论坛,你搜不到过多的素材,那对中期学习,大概是外人维护,是很不便于的。何况撰稿人重写了好些个js的中坚措施,那致使假使出错,除了找笔者和友爱改源码外,很只怕无处查询。

    最后选拔React(0.14.x),除了包容外,还大概有诸如社区活泼,能源多,最注重的是有推特(TWTR.US)做支撑。当然,它的ie8扶助供给举办多插件配置和支撑,在自家github上关于react学习的有关实践里有连锁配置文件。当中,package.json和webpack.config.js须要优秀看下,其次正是小心项目组织(因为用了react-router,项目布局相对清晰),别的包容相关能够经过上边链接举办学习。
    https://github.com/xcatliu/react-ie8
    http://www.aliued.com/?p=3240

    //basePath由服务端配置域名varbasePath=" seajs 的粗略陈设seajs.config({base:base帕特h,alias:{"jquery":"../bower_components/jquery/jquery.seejs.min.js","handlebars":"../bower_components/handlebars/handlebars.seajs.min.js"}});// 加载入口模块seajs.use("./scripts/seajs handlebars.js");````小心,jquery默许是不支持速龙规范,也不帮衬seajs,所以那边用的````jquery.seejs.min.js和handlebars.seajs.min.js````都以投机手动改过的,jquery修改的主意正是在源码中级加上几行代码,如下:````jsdefine(function(){//jquery源代码写在此中return$.noConflict();});

    最后

    后天自己只是给我们简单来讲了一下后台管理的布局和vue的简短知识,大家只要风乐趣能够去询问一下,也能够每26日交换~

    Turbo

    偏偏消除分发和包装的题目还非常不够,业务花费进度假若变得繁琐、难以 Debug、质量低下的话,恐怕不会碰着程序员应接。

    为了缓和那几个主题材料,大家在 Node 框架的底子上,提供了一多种中间件和开拓工具,稳步营造对组件友好的前端工程化方案 Turbo。主要有:

    • 援救前后端同构渲染,让顾客更早看到内容
    • 简化 Flux 流程,数据流特别清晰易维护
    • 引入 ImmutableJS,保障 Store 以外的数目不可变
    • 使用 cursor 机制,保障数据修改/获取同步
    • 支撑 Hot Module Replacement,立异开辟流自动化

    因而那么些创新,一线程序猿能够方便的行使各样零件,专注在事情自个儿上。开辟框架层面包车型地铁扶助也反过来推动了组件化的发展,咱们更乐于使用一多种组件来营造页面效果。

    主题素材:前端发送的静态文件(js,css)必要多

    handlebars修改比jquery少了那句return $.noConflict();就足以了。

    小结

    意识痛点、深入分析调查切磋、应用创新的消除难题思路在组件化开垦推行中不停使用。历经四个大学本科子的演进,组件化开辟情势有效化解了职业发展带来的复杂度进步的压力,并铸就程序猿具有小而美的工程观念,产生一同创建分享的突出氛围。无庸置疑,组件化这种“分而治之”的考虑将会长时间地影响和推动前端开垦形式。大家前日早就准备好,招待新的机遇和挑衅,用技巧的无休止革新进步程序猿的幸福感。

    1 赞 5 收藏 评论

    新葡亰496net 26

    方案:Webpack


    Webpack协理转译,打包,压缩等功用。转译能够将react的jsx语言转成js,es6写法转成es5(半数以上浏览器包容)等,同一时间,将过多文本的转译结果及依赖关系等,打包压缩到多少个文书中,只需二次呼吁,便也就是加载了四个文本及其关联,相当大地进步了后边八个页面加载速度。

    basePath能够依赖实际条件设置。使用alias设置别称后,就足以在sea中使用require()的措施获得到源码

    标题:样式要求不高,但重复性高

    seajs.use("./scripts/seajs handlebars.js");设置了页面包车型地铁js入口

    方案:Ace Admin(ie8)


    友善第一照旧后端开荒,管理端样式不是友善切磋的重大,故决定取舍模板进行支付。通过应用商量,Ace Admin和AdminLTE都以github上相比受接待的前端样式模板,在那之中Ace协助ie8,故选取前面一个用于实际开销。但AdminLTE相对美观,提供的体裁选用很多,我们也得以行使。

    QQ截图20161027190718.jpg

    入口seajs handlebars.js

    其三方库


    React的考虑是经过操作设想dom完毕钦定任务。可是,在实质上支付中,其实有过多方便人民群众的轮子并从未React化,依然供给通过外界引进的格局借助第三方库开张开垦。
    在那边,React组件的生命周期为引进三方库,提供了componentDidMount等艺术,方便对第三方库开展加载。
    这里顺便安利多少个库,如uploadify(flash版上传文件)、ZeroClipboard(点击就能够复制)、dataTables(列表前端组件)

    // 全体模块都通过 define 来定义

    代码相关


    类型代码不太好开源,但前面本身写的多少个react相关的上学例子与项目契合度较高。
    此地贴出react学习的事例链接,感兴趣的朋友能够活动下载。款待各位相互交换。

    define(function(require, exports, module) {

    $(function(){

    var Handlebars = require('handlebars');

    var tpl = require("./data.tpl");

    var demoTplc = Handlebars.compile(tpl);

    $("body").html( demoTplc("hello world"));

    });

    });

    data.tpl

    { {this} }

    加载Handlebars和tpl模板,然后直接调用Handlebars.compile进行模板编写翻译后渲染dom。推行后bady的开始和结果就改成了hello world。我们可以下载demo,展开seajs handlebars.html看下效果

    组件化方案

    从前的代码能够产生用seajs动态加载handlebars模板并渲染的作用,但并不曾兑现组件化。以后咱们来促成组件的包裹。首先组建三个文书夹,满含

    components

    - Boxes

    -index.js  //js代码管理数量和事件

    -index.css //boxes样式

    -boxes.tpl //boxes html模板

    里面包车型客车剧情作为demo,小编写的简约一些。

    boxes.tpl

    this is a boxes!

    { {this} }

    index.css

    .c-boxes{background-color:red;}.c-boxesh1{color:blue;}

    index.js

    define(function(require,exports,module){varHandlebars=require('handlebars');varbox={init:function(){returnbox;},clicked:function(){},render:function($dom,data){vartpl=require('./boxes.tpl');vartplc=Handlebars.compile(tpl);// var _clicked = clicked;$dom.html(tplc(data));box.$dom=$dom;$dom.click(function(){box.clicked&&box.clicked();});}};module.exports=box;});

    index.js稍微复杂一些,封装了一个对象,并定义了指标的点击事件的外界接口。rander方法使用了前头相似的点子,渲染了bandlerbars模板,并流入点击事件

    如此那般三个零件就曾经封装好了。

    组件的运用

    日前定义了二个boxes组件,今后大家来使用他.html页面和在此以前的页面是千篇一律的,独一的界别是入口js换来了componentization.js

    componentization.js

    define(function(require,exports,module){// 通过 require 引进重视var$=require('jquery');var博克斯es=require('boxes');$(function(){//实例化组件varbox=Boxes.init();box.render($("body"),"hello world!");box.clicked=function(){console.log("clicked");};});});

    box定义好后,使用起来非常轻松,直接调用box.render()方法就足以了,也故意还是无意个box绑定了点击事件,效果能够看demo中的 componentization.html 页面

    组件化别的的探究

    组件的模板和js都经过seajs加载和包装了,独一可惜的是css必要独自引进。 seajs也许有css引进的插件sea-css,不过试了须臾间没得逞。 所以作者的实施方案是使用gulp完毕,遍历component下的富有css文件,组合成并压缩成三个css,然后再利用的页面中集合引用这几个统一的css

    具体gulp脚本

    //css components 组合 创设职责

    gulp.task('css-concat', function () {

    gulp.src(css_components_Src)

    .pipe(concat('allComponent.css'))//合併后的公文名

    .pipe(gulp.dest(cssDst));

    });

    demo

    本文示例demo见demo-web

    文件的demo在文件夹Handlebars中

    应用方法

    Handlebars目录下进行http-server -p 8080

    usage.html =》 handlerbars的应用及模板预编写翻译

    componentization.html =》 前端开拓框架 - seajs handlebars模块化开拓

    错误管理

    比如地点未有http-server命令,请安装nodejs情状,并透过npm安装 http-server 命令:npm install -g http-server

    本文由新葡亰496net发布于新葡亰官网,转载请注明出处:新葡亰496net:致我们终将组件化的Web,前端工程

    关键词: