经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » HTML/CSS » HTML5 » 查看文章
Vue.js 源码分析(十九) 指令篇 v-html和v-text指令详解
来源:cnblogs  作者:大沙漠  时间:2019/7/5 10:28:16  对本文有异议

双大括号会将数据解释为普通文本,而非 HTML 代码。为了输出真正的 HTML,你需要使用 v-html 指令,例如:

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Document</title>
  6. <script src="https://cdn.jsdelivr.net/npm/vue@2.5.16/dist/vue.js"></script>
  7. </head>
  8. <body>
  9. <script>
  10. Vue.config.productionTip=false;
  11. Vue.config.devtools=false;
  12. </script>
  13. <div id="app">
  14. <p>{{message}}</p>
  15. <p v-html="message"></p>
  16. </div>
  17. <script>
  18. var app = new Vue({
  19. el:'#app',
  20. data:{
  21. message:'<span style="color:#f00">Hell World!</span>'
  22. }
  23. })
  24. </script>
  25. </body>
  26. </html>

渲染结果为:

<p>{{message}}</p>里的message被解释为了普通文本,而不是输出真正的 HTML,而<p v-html="message"></p>输出了真正的html

v-text和v-html类似,v-text以普通文本来插入,例如:

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Document</title>
  6. <script src="https://cdn.jsdelivr.net/npm/vue@2.5.16/dist/vue.js"></script>
  7. </head>
  8. <body>
  9. <script>
  10. Vue.config.productionTip=false;
  11. Vue.config.devtools=false;
  12. </script>
  13. <div id="app">
  14. <p v-html="message">{{message}}</p>
  15. <p v-text="hello">你好</p>
  16. </div>
  17. <script>
  18. var app = new Vue({
  19. el:'#app',
  20. data:{
  21. message:'<span style="color:#f00">Hell World!</span>',
  22. hello:"Hello world"
  23. }
  24. })
  25. </script>
  26. </body>
  27. </html>

渲染的结果为:

 源码分析


我们以第二个例子为例。

v-html和v-text都是内部指令,它们有初始化函数,分别如下:

  1. function text (el, dir) { //第9785行 v-text指令
  2. if (dir.value) {
  3. addProp(el, 'textContent', ("_s(" + (dir.value) + ")"));     //给el.prop上增加一个textContext属性
  4. }
  5. }
  6. function html (el, dir) { //第9788行 v-html指令
  7. if (dir.value) {
  8. addProp(el, 'innerHTML', ("_s(" + (dir.value) + ")"));      //给el.prop上增加一个innerHTML属性
  9. }
  10. }

parse()解析模板时会执行processAttrs()函数,如下:

  1. function processAttrs (el) { //第9526行 对剩余的属性进行分析
  2. var list = el.attrsList;
  3. var i, l, name, rawName, value, modifiers, isProp;
  4. for (i = 0, l = list.length; i < l; i++) {
  5. name = rawName = list[i].name;
  6. value = list[i].value;
  7. if (dirRE.test(name)) { //如果该属性以v-、@或:开头,表示这是Vue内部指令
  8. // mark element as dynamic
  9. el.hasBindings = true;
  10. // modifiers
  11. modifiers = parseModifiers(name);
  12. if (modifiers) {
  13. name = name.replace(modifierRE, '');
  14. }
  15. if (bindRE.test(name)) { // v-bind
  16. /*v-bind的分支*/
  17. } else if (onRE.test(name)) { // v-on
  18. /*v-on的分支*/
  19. } else { // normal directives //普通指令
  20. name = name.replace(dirRE, ''); //去掉指令前缀,比如v-model执行后等于model
  21. // parse arg
  22. var argMatch = name.match(argRE);
  23. var arg = argMatch && argMatch[1];
  24. if (arg) {
  25. name = name.slice(0, -(arg.length + 1));
  26. }
  27. addDirective(el, name, rawName, value, arg, modifiers); //执行addDirective给el增加一个directives属性,值是一个数组,例如:[{name: "model", rawName: "v-model", value: "message", arg: null, modifiers: undefined}]
  28. if ("development" !== 'production' && name === 'model') {
  29. checkForAliasModel(el, value);
  30. }
  31. }
  32. } else {
  33. /*普通属性的分支*/
  34. }
  35. }
  36. }
  37. }
  38. function addDirective ( //第6561行 指令相关,给el这个AST对象增加一个directives属性,值为该指令的信息,比如:
  39. el,
  40. name,
  41. rawName,
  42. value,
  43. arg,
  44. modifiers
  45. ) {
  46. (el.directives || (el.directives = [])).push({ name: name, rawName: rawName, value: value, arg: arg, modifiers: modifiers });
  47. el.plain = false;
  48. }

对于<p v-html="message">{{message}}</p>节点来说,他的AST对象如下:

对于<p v-text="hello">你好</p>对象来说,他的AST对象如下:

执行gendata$2()拼凑data属性时会先执行genDirectives()函数,该函数会执行v-html和v-text指令的安装函数,添加对应的prop属性,最后会转换为domProps属性,例子里的代码经过解析后生产的render函数如下:

  1. with(this){return _c('div',{attrs:{"id":"app"}},[_c('p',{domProps:{"innerHTML":_s(message)}},[_v(_s(message))]),_v(" "),_c('p',{domProps:{"textContent":_s(hello)}},[_v("你好")])])}

可以看到给两个p元素分别添加了一个domProps属性,值为对应的信息,

最后渲染成对应的真实的DOM节点后就会执行domProps模块(Vue内置的模块,当DOM元素渲染、更新、删除时做一些操作)的初始化函数,也就是updateDOMProps函数,如下:

  1. function updateDOMProps (oldVnode, vnode) { //第7102行 更新DOM对象的props
  2. if (isUndef(oldVnode.data.domProps) && isUndef(vnode.data.domProps)) { //如果oldVnode和vnode的data上都没有domProps属性
  3. return //则直接返回不做处理
  4. }
  5. var key, cur;
  6. var elm = vnode.elm; //vnode对应的DOM节点对象
  7. var oldProps = oldVnode.data.domProps || {};
  8. var props = vnode.data.domProps || {}; //vnode的domProps对象
  9. // clone observed objects, as the user probably wants to mutate it
  10. if (isDef(props.__ob__)) {
  11. props = vnode.data.domProps = extend({}, props);
  12. }
  13. for (key in oldProps) {
  14. if (isUndef(props[key])) {
  15. elm[key] = '';
  16. }
  17. }
  18. for (key in props) { //遍历props 对于<p v-html="message">{{message}}</p>这个节点来说,这里的key等于:innerHTML
  19. cur = props[key]; //获取对应的值,对于<p v-html="message">{{message}}</p>这个节点来说,cur等于:"<span style="color:#f00">Hell World!</span>"
  20. // ignore children if the node has textContent or innerHTML,
  21. // as these will throw away existing DOM nodes and cause removal errors
  22. // on subsequent patches (#3360)
  23. if (key === 'textContent' || key === 'innerHTML') { //如果key等于textContent或innerHTML,这里是对指令v-html和v-text的支持
  24. if (vnode.children) { vnode.children.length = 0; } //如果有子节点,则删除它们
  25. if (cur === oldProps[key]) { continue }
  26. // #6601 work around Chrome version <= 55 bug where single textNode
  27. // replaced by innerHTML/textContent retains its parentNode property
  28. if (elm.childNodes.length === 1) {
  29. elm.removeChild(elm.childNodes[0]);
  30. }
  31. }
  32. if (key === 'value') { //如果key等于value
  33. // store value as _value as well since
  34. // non-string values will be stringified
  35. elm._value = cur;
  36. // avoid resetting cursor position when value is the same
  37. var strCur = isUndef(cur) ? '' : String(cur);
  38. if (shouldUpdateValue(elm, strCur)) {
  39. elm.value = strCur;
  40. }
  41. } else {
  42. elm[key] = cur; //否则直接设置elm的key属性值为cur,也就是设置元素的innerHTML或textContent属性
  43. }
  44. }
  45. }

从updateDOMProp函数内看到,对于v-html或v-text指令来说,如果有子节点,会每个删除掉,所以如果一个元素绑定了v-html或v-text指令,它的子节点时将忽略掉。

总结:通过源码可以发现,对于v-html和v-text来说,Vue是通过设置元素原生的innerHTML或textContent这两个属性来实现的。

 

原文链接:http://www.cnblogs.com/greatdesert/p/11128344.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号