手写vue双向数据绑定

x33g5p2x  于2022-03-05 转载在 Vue.js  
字(5.1k)|赞(0)|评价(0)|浏览(681)

一、写在前面
学习前端不得不学vue,学习vue不得不学习他的响应式原理,下面为我实现响应式的思路和代码,注释写在代码中。
二、实现

  1. const CommandUtils = {
  2. setValue(vm, attr, newValue) {
  3. attr.split('.').reduce((data, currentAttr, index, arr) => {
  4. if (index === arr.length - 1) {
  5. data[currentAttr] = newValue;
  6. }
  7. return data[currentAttr];
  8. }, vm.$data)
  9. },
  10. getContent(vm, value) {
  11. let reg = /\{\{(.+?)\}\}/gi;
  12. let val = value.replace(reg, (...args) => {
  13. return this.getValue(vm, args[1]);
  14. });
  15. return val;
  16. },
  17. getValue: function (vm, value) {
  18. return value.split(".").reduce((data, key) => {
  19. return data[key.trim()]
  20. }, vm.$data)
  21. },
  22. model: function (vm, node, value) { //vm为options, node为节点,value为 data中属性
  23. const val = this.getValue(vm, value)
  24. new Watcher(vm, value, (newValue, oldValue) => {
  25. node.value = newValue
  26. })
  27. node.addEventListener("input", (e) => {
  28. const newValue = e.target.value
  29. this.setValue(vm, value, newValue);
  30. })
  31. node.value = val
  32. },
  33. html: function (vm, node, value) {
  34. new Watcher(vm, value, (newValue, oldValue) => {
  35. node.innerHTML = newValue
  36. })
  37. const val = this.getValue(vm, value)
  38. node.innerHTML = val
  39. },
  40. text: function (vm, node, value) {
  41. new Watcher(vm, value, (newValue, oldValue) => {
  42. node.textContent = newValue
  43. })
  44. const val = this.getValue(vm, value)
  45. node.textContent = val
  46. },
  47. content: function (vm, node) {
  48. const reg = /\{\{(.+?)\}\}/gi
  49. const content = node.textContent
  50. const val = content.replace(reg, (...args) => {
  51. new Watcher(vm, args[1], (newValue, oldValue) => {
  52. node.textContent = this.getContent(vm, content);
  53. });
  54. return this.getValue(vm, args[1])
  55. })
  56. node.textContent = val
  57. },
  58. on: function (node, value, vm, type) {
  59. node.addEventListener(type, (e) => {
  60. vm.$methods[value].call(vm, e)
  61. })
  62. }
  63. }
  64. export default class minVue {
  65. constructor(options) {
  66. if (this.isElement(options.el)) {
  67. this.$el = options.el
  68. } else {
  69. this.$el = document.querySelector(options.el)
  70. }
  71. this.$data = options.data
  72. this.proxyData();
  73. this.$methods = options.methods;
  74. new Observer(this.$data)
  75. if (this.$el) {
  76. // 将模板进行编译
  77. new Compiler(this)
  78. }
  79. }
  80. // 判断是否是标签元素
  81. isElement(node) {
  82. return node.nodeType === 1
  83. }
  84. // 将data中的数据放到vue实例上
  85. proxyData() {
  86. for (let key in this.$data) {
  87. Object.defineProperty(this, key, {
  88. get: () => {
  89. return this.$data[key]
  90. }
  91. })
  92. }
  93. }
  94. }
  95. // 封装对模板进行编译的类
  96. class Compiler {
  97. constructor(vm) {
  98. this.$vm = vm
  99. this.fragment = this.fragmentTemplate()
  100. this.buildTemplate(this.fragment)
  101. this.$vm.$el.appendChild(this.fragment)
  102. }
  103. fragmentTemplate() {
  104. const fragment = document.createDocumentFragment()
  105. const parentNode = this.$vm.$el
  106. let childNode = parentNode.firstChild
  107. while (childNode) {
  108. fragment.appendChild(childNode)
  109. childNode = parentNode.firstChild
  110. }
  111. return fragment
  112. }
  113. // 编译模板
  114. buildTemplate(fragment) {
  115. const fragmentChildNos = [...fragment.childNodes]
  116. fragmentChildNos.forEach(node => {
  117. // 元素节点类型
  118. if (this.$vm.isElement(node)) {
  119. this.buildNode(node)
  120. this.buildTemplate(node)
  121. } else {
  122. // 文本节点
  123. this.buildText(node)
  124. }
  125. })
  126. }
  127. // 编译node节点
  128. buildNode(node) {
  129. const attrs = node.attributes
  130. const newAttrs = [...attrs]
  131. newAttrs.forEach(item => {
  132. const {
  133. name,
  134. value
  135. } = item //name 为 v-model value 为 title
  136. if (name.startsWith("v-")) {
  137. const command = name.split("-")[1] //model
  138. const typeEvent = command.split(":")
  139. if (typeEvent.length === 1) {
  140. CommandUtils[command](this.$vm, node, value)
  141. } else {
  142. CommandUtils[typeEvent[0]](node, value, this.$vm, typeEvent[1])
  143. }
  144. }
  145. })
  146. }
  147. // 编译文本节点
  148. buildText(node) {
  149. const reg = /\{\{.+?\}\}/gi
  150. if (reg.test(node.textContent)) {
  151. CommandUtils["content"](this.$vm, node)
  152. }
  153. }
  154. }
  155. // 对全部的属性进行监听
  156. class Observer {
  157. constructor(obj) {
  158. this.observer(obj)
  159. }
  160. observer(obj) {
  161. if (obj && typeof obj === "object") {
  162. Reflect.ownKeys(obj).forEach(key => {
  163. this.defineReactive(obj, key, obj[key])
  164. })
  165. }
  166. }
  167. defineReactive(obj, attr, value) {
  168. this.observer(value)
  169. let dep = new Dep()
  170. Object.defineProperty(obj, attr, {
  171. get() {
  172. Dep.target && dep.addWatchs(Dep.target)
  173. return value
  174. },
  175. set: (newValue) => {
  176. if (value !== newValue) {
  177. this.observer(newValue);
  178. value = newValue
  179. dep.notify()
  180. }
  181. }
  182. })
  183. }
  184. }
  185. // 保存依赖的类
  186. class Dep {
  187. constructor() {
  188. this.subs = []
  189. }
  190. addWatchs(watcher) {
  191. this.subs.push(watcher)
  192. }
  193. notify() {
  194. this.subs.forEach(watch => watch.update())
  195. }
  196. }
  197. class Watcher {
  198. constructor(vm, attr, cb) {
  199. this.vm = vm
  200. this.attr = attr
  201. this.cb = cb
  202. this.oldValue = this.getOldValue()
  203. }
  204. getOldValue() {
  205. Dep.target = this
  206. let val = CommandUtils.getValue(this.vm, this.attr)
  207. Dep.target = null
  208. return val
  209. }
  210. update() {
  211. const newValue = CommandUtils.getValue(this.vm, this.attr)
  212. if (this.oldValue !== newValue) {
  213. this.cb(newValue, this.oldValue)
  214. this.oldValue = newValue
  215. }
  216. }
  217. }

html代码展示

  1. <!DOCTYPE html>
  2. <html lang="cn">
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta http-equiv="X-UA-Compatible" content="IE=edge">
  6. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  7. <title>Document</title>
  8. </head>
  9. <body>
  10. <div id="app">
  11. <h1>大家好,我是{{ name }}</h1>
  12. <h1>大家好,我的年龄是{{ age }}</h1>
  13. <input type="text" v-model="time.H">
  14. <input type="text" v-model="time.M">
  15. <input type="text" v-model="time.S">
  16. <input type="text" v-model="time.S">
  17. <h1>{{ time.S }}</h1>
  18. <button v-on:click="myFn">点击</button>
  19. </div>
  20. <script type="module">
  21. import minVue from './minVue.js'
  22. new minVue({
  23. el:"#app",
  24. data: {
  25. name: "dmc",
  26. age: 20,
  27. time: {
  28. H:"小时",
  29. M: "分钟",
  30. S: "秒钟"
  31. }
  32. },
  33. methods: {
  34. myFn:() => {
  35. console.log("hhhhh")
  36. }
  37. },
  38. })
  39. </script>
  40. </body>
  41. </html>

相关文章

最新文章

更多