代码记录

2019-12-27  本文已影响0人  增商

MVVM.js

// 观察者(发布订阅) 观察者 被观察者
class Dep {
  constructor() {
    this.subs = [];
  }
  addSub(watcher) {
    this.subs.push(watcher);
  }
  notify() {
    this.subs.forEach(wather => wather.update());
  }
}
//new Watcher
class Watcher {
  constructor(vm, expr, callb) {
    this.vm = vm;
    this.expr = expr;
    this.cb = callb;
    //默认先存放老值
    this.oldValue = this.get();
  }
  get() {
    Dep.target = this;
    let value = CompileUtil.getVal(this.vm, this.expr);
    Dep.target = null;
    return value;
  }
  update() {
    //更新操作,数据变化后会调用观察者的update方法
    let newVal = CompileUtil.getVal(this.vm, this.expr);
    if (newVal != this.oldValue) {
      this.cb(newVal);
    }
  }
}

//实现数据劫持功能=>解构赋值
class Observer {
  constructor(data) {
    console.log(data);
    this.observer(data);
  }
  observer(data) {
    //依次遍历,先判断合法性,如果数据没写是空的就不用监控了
    //if object => observer 如果是对象就观察
    if (data && typeof data == "object") {
      //if object => for
      for (let key in data) {
        this.defineReactive(data, key, data[key]);
        //data数据定义key属性 值是data[key]
      }
    }
  }
  defineReactive(obj, key, value) {
    let dep = new Dep();
    this.observer(value);
    Object.defineProperty(obj, key, {
      get() {
        //创建watcher时会 取到对应的内容并且把watcher放到全局上
        Dep.target && dep.addSub(Dep.target);
        //默认get方法取它未来的value
        return value;
      },
      set: newVal => {
        //{school:{name:'imycode'}} school={}
        if (newVal != value) {
          this.observer(newVal);
          value = newVal;
          dep.notify();
          //if current time value==newVal 如果
          //当前的老值和新值一样就不要进到这个if里
          //赋值
        }
      }
    });
  }
}

//基础类
class Compiler {
  constructor(el, vm) {
    // console.log(el);
    this.vm = vm;
    //判断el属性是不是一个元素,如果不是元素那就获取他
    this.el = this.isElementNode(el) ? el : document.querySelector(el);
    // console.log(this.el); //dom节点
    let fragment = this.node2fragment(this.el);
    // console.log(fragment);
    //再内存中处理好
    //节点的内容进行替换

    //编译模板数据驱动
    this.compile(fragment);
    //然后再渲染给页面
    this.el.appendChild(fragment);
  }
  //编译文本的-看看有没{}
  isDirective(attrName) {
    return attrName.startsWith("v-");
  }
  compileText(node) {
    //{{xxx}} {{aaa}}
    let content = node.textContent;
    // console.log(content, "内容");
    if (/\{\{(.+?)\}\}/.test(content)) {
      // console.log(content); //找到所有文本元素开始填充
      CompileUtil["text"](node, content, this.vm);
    }
  }
  //编译元素的-看看有没v-model
  compileElement(node) {
    //取dom元素的属性
    let attributes = node.attributes; //类数组
    // console.log(attributes);
    //判断这些属性里有没有v-model的如果有我就找到了
    [...attributes].forEach(attr => {
      //v-model="school.name" type="text"
      // console.log(attr);
      let { name, value: expr } = attr;
      //结构  - v-model="school.name"
      // console.log(name, value);
      //判断name是不是v-开头的,是不是指令
      if (this.isDirective(name)) {
        let [, directive] = name.split("-");
        let [directiveName, eventName] = directive.split(":");
        // console.log(node, "element");
        //找到了
        CompileUtil[directiveName](node, expr, this.vm, eventName);
        //需要调用不同的指令来处理 - v-model="school.name"
      }
    });
  } //核心的编译方法
  compile(node) {
    //用来编译内存中的dom节点
    let childNodes = node.childNodes;
    // console.log(childNodes);
    [...childNodes].forEach(child => {
      //是不是元素,不是元素就是文本
      //元素有v-model 文本有{{...}}
      if (this.isElementNode(child)) {
        // console.log("element  ", child);
        this.compileElement(child);
        // 如果是元素的话再去遍历子节点
        this.compile(child);
      } else {
        // console.log("text  ", child);
        this.compileText(child);
      }
    });
  }
  //把节点移动到内存中
  node2fragment(node) {
    //把当前元素中的东西都拿到
    //一个一个拿childnode儿子节点,都拿到
    let fragment = document.createDocumentFragment();
    //创建一个文档碎片
    let firstChild;
    //node指最外层的div#app
    while ((firstChild = node.firstChild)) {
      //不停拿第一个塞到内存最后页面中的节点都没有了就为null循环自然结束
      //拿一个少一个不用担心死循环
      //appendChild具有移动性
      // console.log(firstChild);

      fragment.appendChild(firstChild);
    }
    return fragment;
  }
  isElementNode(node) {
    //是不是元素节点
    return node.nodeType === 1; //undefined===1? 不相等
  } //=>#app false
}
//编译工具
CompileUtil = {
  getVal(vm, expr) {
    //vm.$data 'school.name' 根据表达式取到最终的数据
    return expr.split(".").reduce((data, current) => {
      return data[current];
    }, vm.$data);
  },
  setValue(vm, expr, value) {
    expr.split(".").reduce((data, current, index, arr) => {
      if (index == arr.length - 1) {
        return (data[current] = value);
      }
      return data[current];
    }, vm.$data);
  },
  model(node, expr, vm) {
    //给输入框赋予value属性 node.value=xx
    // vm[expr]=vm.$data['...']错误
    let fn = this.updater["modelUpdater"];
    new Watcher(vm, expr, newVal => {
      fn(node, newVal);
    });
    node.addEventListener("input", e => {
      let value = e.target.value; //获取用户输入的内容
      this.setValue(vm, expr, value);
    });
    let value = this.getVal(vm, expr); //返给我imycode
    fn(node, value);
  }, //node节点expr表达式school.name, vm当前实例
  html(node, expr, vm) {
    // v-html="msg"
    let fn = this.updater["htmlUpdater"];
    new Watcher(vm, expr, newVal => {
      fn(node, newVal);
    });

    let value = this.getVal(vm, expr); //返给我imycode
    fn(node, value);
  },
  //遍历表达式将内容重新替换成一个完整的内容返还回去
  getContentValue(vm, expr) {
    return expr.replace(/\{\{(.+?)\}\}/g, (...args) => {
      return this.getVal(vm, args[1]);
    });
  },
  on(node, expr, vm, eventName) {
    // expr v-on:click="change" expr==change
    node.addEventListener(eventName, e => {
      // alert(1);
      vm[expr].call(vm, e); //this.change
    });
  },
  text(node, expr, vm) {
    let fn = this.updater["textUpdater"];
    //expr=>{{a}} {{b}} {{c}} => a b c
    let content = expr.replace(/\{\{(.+?)\}\}/g, (...args) => {
      //给表达式每个人都加上观察者,只要数据变了我就更新, 每个大括号
      new Watcher(vm, args[1], () => {
        fn(node, this.getContentValue(vm, expr)); //返回了一个全的字符串
      });
      return this.getVal(vm, args[1]);
    });
    //最终文本内容
    fn(node, content);
  },
  updater: {
    //把数据插入到节点中
    modelUpdater(node, value) {
      node.value = value;
    },
    htmlUpdater(node, value) {
      //xss攻击
      node.innerHTML = value;
    },
    //处理文本节点
    textUpdater(node, value) {
      node.textContent = value;
    }
  }
};
class Vue {
  constructor(options) {
    //this.$el $data $options
    this.$el = options.el;
    // console.log("options.el:  " + options.el);
    this.$data = options.data;
    let computed = options.computed;
    let methods = options.methods;
    // console.log("options.data:  " + options.data);
    //把数据全部转化成Object.defineProperty来定义
    //这个根元素存在编译模板
    if (this.$el) {
      new Observer(this.$data);

      //把vm上的取值,都代理到vm.$data上

      //{{getNewName}} reduce vm.$data.getNewName
      for (let key in computed) {
        //有依赖关系 数据变化
        Object.defineProperty(this.$data, key, {
          get: () => {
            return computed[key].call(this);
          }
        });
      }
      this.proxyVm(this.$data);
      new Compiler(this.$el, this);
    }
    for (let key in methods) {
      Object.defineProperty(this, key, {
        get() {
          return methods[key];
        }
      });
    }
  }

  proxyVm(data) {
    //取值代理可以通过vm.school取到对应的内容
    for (let key in data) {
      Object.defineProperty(this, key, {
        get() {
          return data[key]; //进行了转化操作
        },
        set(newVal) {
          data[key] = newVal;
        }
      });
    }
  }
}

html

<!DOCTYPE html>
<html lang="en">

  <head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <!-- <script src='https://cdnjs.cloudflare.com/ajax/libs/vue/2.6.6/vue.js'></script> -->
  </head>

  <body>
    <div id="app">
      <input v-model="school.name" type="text" />
      <div>{{school.name}}</div>
      <div>{{school.age}}</div>
      {{getNewName}}
      <ul>
        <li>1</li>
        <li>1</li>
      </ul>
      <input type="button" value="这是一个源码解析实验" v-on:click="change">
      <div v-html="msg"></div>
    </div>
    <script src="MVVM.js"></script>
    <script>
    var vm = new Vue({
      el: "#app",
      data: {
        school:{
          name:"imycode",
          age:10
        },
        msg:'<h1>好难!!!</h1>'
      },
      methods: {
        change(){
          this.school.name='武兴师'
        }
      },
      computed: {
        getNewName(){
          return this.school.name+'newHander'
        }
      },
    });
  </script>
  </body>

</html>
上一篇 下一篇

猜你喜欢

热点阅读