【js】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>

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

posted @ 2019-06-01 15:14  逆风-鬼刀  阅读(149)  评论(0编辑  收藏  举报