zoukankan      html  css  js  c++  java
  • 【js】vue 2.5.1 源码学习(二) 策略合并

     一、  整体思路
        1 首先是代码的大体构造,先判断引入代码的环境,即对应amd 和cmd的处理
        2 vue_init 需要借助 initMinxin    ==>>>       // 初始化选项1: 规范 2: 合并策略。
        3 mergeOptions 选项合并 一个或者多个对象合并,并且生成一个新的对象
             ==>  resloveConstructorOptions 返回vm的optios 判断是否是vue对象有可能是vue子类。不一定指向options。
                    parent 自定义的options的选项     child 用户自定义的options
             ==>   checkComonpents // 规范的检测 validataComponentsName(key)
                    检测component的名称是否规范(不能使用内置标签slot component 关于html的标签名称 svg的子类的名称)
                  (规范:组件的名称必须是字母或中横线,必须由字母开头)
                    isbuiltInTag 检测是都是内置标签 isReservedTag是否是html的标签
                    ===>   makeMap 检测key是否在makeMap里面
            ==> mergeField 默认策略 // 以默认为优先。用户定义为覆盖
                  defaultStrat(parent,chhild)  合并策略   child === undefined ? parent : child;
           ==>  var config = {    // 配置对象
                      // 内置对象自定义策略
                     optionMergeStrategies:{}
                   }
                 var strats = config.optionMergeStrategies;
        4 Vue.options = {} vue的全局api
      1 //  大体思路  
      2 // 1 首先是代码的大体构造,先判断引入代码的环境,即对应amd 和cmd的处理
      3 // 2 vue_init  需要借助  initMinxin  // 初始化选项1: 规范 2: 合并策略。
      4 // 3 mergeOptions  选项合并 一个或者多个对象合并,并且生成一个新的对象 
      5     // resloveConstructorOptions  返回vm的optios  判断是否是vue对象又k
      6     // 有可能是vue子类。不一定指向options。     
      7     //  parent 自定义的options的选项  
      8     //  child  用户自定义的options  
      9     /*  checkComonpents  // 规范的检测 validataComponentsName(key)
     10         检测component的名称是否规范(不能使用内置标签slot component 关于html的标签名称  svg的子类的名称)自称)
     11        (规范:组件的名称必须是字母或中横线,必须由字母开头) 
     12         isbuiltInTag 检测是都是内置标签  isReservedTag是否是html的标签
     13         makeMap  检测key是否在makeMap里面 
     14     */   
     15     /*  mergeField 默认策略 // 以默认为优先。用户定义为覆盖  
     16         defaultStrat(parent,chhild)  
     17         child === undefined ? parent : child;
     18     */
     19    /*
     20      var config = {
     21          optionMergeStrategies:{}
     22      }
     23      var strats = config.optionMergeStrategies;
     24    */
     25 // 4 Vue.options = {} vue的全局api   
     26 
     27 (function(global,factory){
     28     // 兼容 cmd  
     29     typeof exports === 'object'  && module !== 'undefined' ? module.exports = factory():   
     30     // Amd
     31     typeof define  === 'function' && define.amd ?  define(factory) : global.Vue = factory();
     32 })(this,function(){
     33     var uip = 0;
     34     function warn(string){
     35         console.error('Vue Wran:' + string)
     36     }
     37 
     38     function resolveConstructorOptions(Con){
     39         var options = Con.options;
     40         // 判断是否为vm的实例 或者是子类
     41           return options
     42     }
     43     var hasOwnPropeerty = Object.prototype.hasOwnProperty
     44     function hasOwn(obj , key){
     45         return hasOwnPropeerty.call(obj,key)
     46     }
     47     function makeMap(str, expectsLoweraseC){
     48         if(expectsLoweraseC){
     49             str = str.toLowerCase()
     50         }
     51         var map = Object.create(null)
     52         var list = str.split(',')
     53         for(var i = 0 ; i < list.length; i++){
     54             map[list[i]] = true
     55         }
     56         return function(key){
     57             return map[key]
     58         
     59         }
     60     }
     61     var  isbuiltInTag = makeMap('slot,component',true)
     62     var isHTMLTag = makeMap(
     63         'html,body,base,head,link,meta,style,title,' +
     64         'address,article,aside,footer,header,h1,h2,h3,h4,h5,h6,hgroup,nav,section,' +
     65         'div,dd,dl,dt,figcaption,figure,picture,hr,img,li,main,ol,p,pre,ul,' +
     66         'a,b,abbr,bdi,bdo,br,cite,code,data,dfn,em,i,kbd,mark,q,rp,rt,rtc,ruby,' +
     67         's,samp,small,span,strong,sub,sup,time,u,var,wbr,area,audio,map,track,video,' +
     68         'embed,object,param,source,canvas,script,noscript,del,ins,' +
     69         'caption,col,colgroup,table,thead,tbody,td,th,tr,' +
     70         'button,datalist,fieldset,form,input,label,legend,meter,optgroup,option,' +
     71         'output,progress,select,textarea,' +
     72         'details,dialog,menu,menuitem,summary,' +
     73         'content,element,shadow,template,blockquote,iframe,tfoot'
     74     );
     75     var isSVG = makeMap(
     76         'svg,animate,circle,clippath,cursor,defs,desc,ellipse,filter,font-face,' +
     77         'foreignObject,g,glyph,image,line,marker,mask,missing-glyph,path,pattern,' +
     78         'polygon,polyline,rect,switch,symbol,text,textpath,tspan,use,view',
     79         true
     80     );
     81     var isReservedTag = function(key){
     82         return isHTMLTag(key) || isSVG(key) 
     83     }
     84     function validataComponentName(key){
     85         //检测component 的自定义名称是否合格 
     86         // 只能是字母开头或下划线,必须是字母开头
     87         if(!(/^[a-zA-Z][w-]*$/g.test(key))){
     88            warn('组件的名称必须是字母或中横线,必须由字母开头')
     89         }
     90         // 1. 不能为内置对象,2.不能是html ,和avg的内部标签
     91         if( isbuiltInTag(key) || isReservedTag(key)){
     92             warn('不能为html标签或者avg的内部标签')
     93         } 
     94     }
     95     function checkComonpents(child){
     96         for(var key in child.component){
     97             validataComponentName(key)
     98         }
     99     }
    100    // 配置对象
    101     var config = {
    102         // 自定义的策略
    103         optionMergeStrategies:{}
    104     }
    105     var strats = config.optionMergeStrategies
    106     strats.el = function(parent,child , key , vm){
    107        
    108           if(!vm){
    109               warn('选项'+key+'只能在vue实例用使用')
    110           }
    111           return defaultStrat(parent,child , key , vm)
    112     }
    113     function defaultStrat(parent,child , key , vm){
    114         return child === undefined ? parent :child ;
    115     }
    116     function mergeOptions(parent,child,vm){
    117         var options = {}
    118         // 检测是component 是否是合法的  
    119      
    120         checkComonpents(child)
    121         
    122         // console.log(parent, child)
    123         for(key in parent){
    124             magerField(key)
    125         }
    126         for(key in child){
    127             if(!hasOwn(parent ,key)){  // parent 中循环过地方不进行循环
    128                 magerField(key)  // ----忘记写
    129             }
    130               
    131         }
    132         // 默认合并策略
    133         function magerField(key){  
    134             // 自定义策略  默认策略 
    135             // console.log(key)
    136             var result = strats[key] || defaultStrat        // ---忘记写
    137             options[key] = result(parent[key],child[key] , key , vm)
    138         }
    139         console.log(options)
    140         return options
    141     }
    142     function initMinxin(options){
    143         Vue.prototype._init = function(options){
    144             var vm = this 
    145             // 记录生成的vue实例对象 
    146             vm._uip =  uip++ //   //-------忘记写
    147           
    148              mergeOptions(resolveConstructorOptions(vm.constructor),options,vm)
    149         }
    150     }
    151     function Vue(options){
    152         // 安全机制
    153         if(!(this instanceof Vue)){     //-------忘记写
    154             warn('Vue是一个构造函数,必须是由new关键字调用')  
    155         }
    156         this._init(options)
    157     }
    158     initMinxin()  //  初始化选项1: 规范 2: 合并策略。
    159     Vue.options = {
    160         components: {},
    161         directives:{},
    162         _bash: Vue
    163     }
    164     return Vue
    165 })
     1 <body>
     2     <div id="app">
     3         <huml></huml>
     4     </div>
     5     <script src="vue.js"></script>
     6     <!-- <script src="vue2.5.1.js"></script> -->
     7     <script type="text/javascript">
     8         var componentA = {
     9             el: "#app"
    10         }
    11         var vm = new Vue({
    12             el:"#app",
    13             data: {
    14                 message: "hello Vue",
    15                 key: "wodow"
    16             },
    17             components:{
    18                 huml: componentA
    19             }
    20             
    21         })
    22         console.log(vm.$options)
    23     </script>
    24 </body>

    以上仅为个人学习上的笔记,如有问题,欢迎评论。

  • 相关阅读:
    Alpha 冲刺 (10/10)
    Alpha 冲刺 (9/10)
    Alpha 冲刺 (8/10)
    Alpha 冲刺 (7/10)
    Alpha 冲刺 (6/10)
    Alpha 冲刺 (5/10)
    18软工实践-团队现场编程实战(抽奖系统)
    Alpha 冲刺 (4/10)
    BETA(4)
    BETA(3)
  • 原文地址:https://www.cnblogs.com/yeujuan/p/10959708.html
Copyright © 2011-2022 走看看