Vue.js 源碼分析(十九) 指令篇 v-html和v-text指令詳解

来源:https://www.cnblogs.com/greatdesert/archive/2019/07/05/11128344.html
-Advertisement-
Play Games

雙大括弧會將數據解釋為普通文本,而非 HTML 代碼。為了輸出真正的 HTML,你需要使用 v-html 指令,例如: 渲染結果為: <p>{{message}}</p>里的message被解釋為了普通文本,而不是輸出真正的 HTML,而<p v-html="message"></p>輸出了真正的h ...


雙大括弧會將數據解釋為普通文本,而非 HTML 代碼。為了輸出真正的 HTML,你需要使用 v-html 指令,例如:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Document</title>
    <script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/vue.js"></script>
</head>
<body>
    <script>
        Vue.config.productionTip=false;
        Vue.config.devtools=false;
    </script>
    <div id="app">
        <p>{{message}}</p>
        <p v-html="message"></p>
    </div>
    <script>
        var app = new Vue({
            el:'#app',
            data:{
                message:'<span style="color:#f00">Hell World!</span>'
            }
        })
    </script>
</body>
</html>

渲染結果為:

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

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

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Document</title>
    <script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/vue.js"></script>
</head>
<body>
    <script>
        Vue.config.productionTip=false;
        Vue.config.devtools=false;
    </script>
    <div id="app">
        <p v-html="message">{{message}}</p>
        <p v-text="hello">你好</p>
    </div>
    <script>
        var app = new Vue({
            el:'#app',
            data:{
                message:'<span style="color:#f00">Hell World!</span>',
                hello:"Hello world"
            }
        })
    </script>
</body>
</html>

渲染的結果為:

 源碼分析


我們以第二個例子為例。

v-html和v-text都是內部指令,它們有初始化函數,分別如下:

function text (el, dir) {       //第9785行 v-text指令
  if (dir.value) {
    addProp(el, 'textContent', ("_s(" + (dir.value) + ")"));     //給el.prop上增加一個textContext屬性
  }
}
function html (el, dir) {       //第9788行 v-html指令
  if (dir.value) {
    addProp(el, 'innerHTML', ("_s(" + (dir.value) + ")"));      //給el.prop上增加一個innerHTML屬性
  }
}

parse()解析模板時會執行processAttrs()函數,如下:

function processAttrs (el) {  //第9526行 對剩餘的屬性進行分析
  var list = el.attrsList; 
  var i, l, name, rawName, value, modifiers, isProp;
  for (i = 0, l = list.length; i < l; i++) {
    name = rawName = list[i].name;
    value = list[i].value;
    if (dirRE.test(name)) {           //如果該屬性以v-、@或:開頭,表示這是Vue內部指令
      // mark element as dynamic
      el.hasBindings = true;
      // modifiers
      modifiers = parseModifiers(name);
      if (modifiers) {
        name = name.replace(modifierRE, '');
      }
      if (bindRE.test(name)) { // v-bind        
        /*v-bind的分支*/
      } else if (onRE.test(name)) { // v-on
        /*v-on的分支*/
      } else { // normal directives               //普通指令
        name = name.replace(dirRE, '');               //去掉指令首碼,比如v-model執行後等於model
        // parse arg
        var argMatch = name.match(argRE);
        var arg = argMatch && argMatch[1];
        if (arg) {
          name = name.slice(0, -(arg.length + 1));
        }
        addDirective(el, name, rawName, value, arg, modifiers);     //執行addDirective給el增加一個directives屬性,值是一個數組,例如:[{name: "model", rawName: "v-model", value: "message", arg: null, modifiers: undefined}]
        if ("development" !== 'production' && name === 'model') {
          checkForAliasModel(el, value);
        }
      }
    } else {
      /*普通屬性的分支*/
      }
    }
  }
}

function addDirective (     //第6561行  指令相關,給el這個AST對象增加一個directives屬性,值為該指令的信息,比如:
  el,
  name,
  rawName,
  value,
  arg,
  modifiers
) {
  (el.directives || (el.directives = [])).push({ name: name, rawName: rawName, value: value, arg: arg, modifiers: modifiers });
  el.plain = false;
}

對於<p v-html="message">{{message}}</p>節點來說,他的AST對象如下:

對於<p v-text="hello">你好</p>對象來說,他的AST對象如下:

執行gendata$2()拼湊data屬性時會先執行genDirectives()函數,該函數會執行v-html和v-text指令的安裝函數,添加對應的prop屬性,最後會轉換為domProps屬性,例子里的代碼經過解析後生產的render函數如下:

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函數,如下:

function updateDOMProps (oldVnode, vnode) {       //第7102行 更新DOM對象的props
  if (isUndef(oldVnode.data.domProps) && isUndef(vnode.data.domProps)) {    //如果oldVnode和vnode的data上都沒有domProps屬性
    return                                                                    //則直接返回不做處理
  }
  var key, cur;
  var elm = vnode.elm;                                                      //vnode對應的DOM節點對象
  var oldProps = oldVnode.data.domProps || {};
  var props = vnode.data.domProps || {};                                    //vnode的domProps對象
  // clone observed objects, as the user probably wants to mutate it
  if (isDef(props.__ob__)) {
    props = vnode.data.domProps = extend({}, props);
  }

  for (key in oldProps) {
    if (isUndef(props[key])) {
      elm[key] = '';
    }
  }
  for (key in props) {                                                    //遍歷props 對於<p v-html="message">{{message}}</p>這個節點來說,這裡的key等於:innerHTML 
    cur = props[key];                                                         //獲取對應的值,對於<p v-html="message">{{message}}</p>這個節點來說,cur等於:"<span style="color:#f00">Hell World!</span>" 
    // ignore children if the node has textContent or innerHTML,
    // as these will throw away existing DOM nodes and cause removal errors
    // on subsequent patches (#3360)
    if (key === 'textContent' || key === 'innerHTML') {                   //如果key等於textContent或innerHTML,這裡是對指令v-html和v-text的支持
      if (vnode.children) { vnode.children.length = 0; }                      //如果有子節點,則刪除它們
      if (cur === oldProps[key]) { continue }                       
      // #6601 work around Chrome version <= 55 bug where single textNode
      // replaced by innerHTML/textContent retains its parentNode property
      if (elm.childNodes.length === 1) {
        elm.removeChild(elm.childNodes[0]);
      }
    }

    if (key === 'value') {                                                //如果key等於value
      // store value as _value as well since
      // non-string values will be stringified
      elm._value = cur;
      // avoid resetting cursor position when value is the same
      var strCur = isUndef(cur) ? '' : String(cur);
      if (shouldUpdateValue(elm, strCur)) {
        elm.value = strCur;
      }
    } else {
      elm[key] = cur;                                                     //否則直接設置elm的key屬性值為cur,也就是設置元素的innerHTML或textContent屬性
    }
  }
}

從updateDOMProp函數內看到,對於v-html或v-text指令來說,如果有子節點,會每個刪除掉,所以如果一個元素綁定了v-html或v-text指令,它的子節點時將忽略掉。

總結:通過源碼可以發現,對於v-html和v-text來說,Vue是通過設置元素原生的innerHTML或textContent這兩個屬性來實現的。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 2019-07-03T01:39:51.691242+08:00 ...
  • vue學習(1) vue-cli 項目搭建 一、windows環境 1. 下載node.js安裝包 官網:https://nodejs.org/en/download/ 選擇LTS下載 2. 安裝 下載完成後點擊安裝包安裝到自己指定的文件夾 windows + r 打開cmd,在命令行中輸入node ...
  • 由於不同瀏覽器,不同版本性能不一,且控制台本質是是套用了一大堆 "eval" ,沙盒化程度高,所以需使用 "node" 環境測試來提高準確性 比第二種方法更簡潔 倒敘簡潔版 兩個分號之間的表達式為 true 會一直執行直到 判斷為 false (i = 0) 正序簡潔版 當 i 大於等於數組長度或a ...
  • 摘要: 玩轉npm。 作者:前端小智 原文: "13 個 npm 快速開發技巧" "Fundebug" 經授權轉載,版權歸原作者所有。 為了保證的可讀性,本文采用意譯而非直譯。 每天,數以百萬計的開發人員使用 或 來構建項目。運行 或`npx create response app`等命令幾乎構建J ...
  • display對象可以獲取所有顯示屏此處演示程式啟動是投放新視窗至另一屏幕 import { app, BrowserWindow } from 'electron' const electron = require('electron') //獲取electron對象 let newWindow ...
  • 寫了個評分組件,效果如下 組件Rate.js 組件樣式 Rate.less 背景圖 調用 <Rate number={10} def={5} /> number:為評分總數,預設為5 def:為評分數,預設為0 ...
  • 1. vue項目打包採坑 1.1. vue運行報錯error:Cannot assign to read only property 'exports' of object ' ' 這個錯誤我是在打包完部署到nginx上才會報的,在本地環境可以正常運行,真坑; 網上的資料說的報錯原因是export和 ...
  • 官網地址:http://www.bacubacu.com/colresizable/ 這裡值得註意的是,如果是動態加入的列,則需要先清理調用插件生成的class,id和div之後再重新調用才會有作用。 至於為何動態載入的列沒有效果呢。首先,我想到了可能是方法載入在了動態生成列之前,所以我便手動在生成 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...