Vue.js源碼全方位深刻解析--學習筆記

模板中的插入變量是如何渲染到DOM上的?

initMixin(Vue)->_init->$options-> $mount()當執行該掛載方法時DOM變化css

爲何能夠經過this訪問到data裏面的數據?

initstate(vm)->initData()->proxy(vm,_data,key)代理函數
因此咱們也能夠同過this._data.dataName獲取到數據
圖片描述vue

$mount的實現

$mount->處理e(編譯,轉化成render函數)->mountComponent()->updateComponent()->渲染Wathernode

vm._render的實現

_render->從vm.options拿到render->render.call(vm._renderProxy,vm.$createElement)->initProxy->hasHandler判斷元素若是不在target上,則會報錯warnNonPresent-> 返回vnodewebpack

虛擬Dom

  1. VNodeData定義在flow/vnode.js (建立虛擬DomTree)
  2. create-element–>
  3. 參數重載->
  4. _createElement->
  5. 對data校驗(若是是響應式的 return create EmptyVnode() (vnode.js))->
  6. 對children作normalizetionChildren(normalize-chiildren.js) 多維數組變一維數組->
  7. 對tag進行判斷(字符串仍是組件)->
  8. 建立Dom

  1. _update定義在src/core/instance/lifecycle.js (渲染web

  2. vm.__patch__vuex

  3. patch數組

  4. createPatchFunction ( 內部定義了一系列的輔助方法,最終返回了一個 patch 方法,這個方法就賦值給了 vm._update 函數裏調用的安全

    vm.__patch__

    )babel

    函數柯里化app

    1. createElm (經過虛擬節點建立真實的 DOM 並插入到它的父節點中)
    2. createChildren
    3. invokeCreateHooks

組件化

createCompment

  • createElement
  • _createElement(對Tag判斷)
  • createComponent

圖片描述

patch

總體流程
重要屬性
  • activeInstance
  • vm.$vnode
  • vm._vnode
嵌套組件插入順序

Vuex

圖片描述

Vue組件

  • 巧用Vue標籤is屬性,解決模板標籤出現bug問題
  • 子組件定義data,必須是一個函數
  • ref 操做dom
  • 父組件經過屬性向子組件傳遞數據

動畫

transition

經過自動操縱transition中的元素的class實現

圖片描述

同時使用過渡和動畫

  • 經過設置type=「transition」(過渡)來設置根據過渡仍是動畫顯示時長
  • 經過appear實現頁面初試動畫
<link rel="stylesheet" href="./animate.css">
    <script src="./vue.min.js"></script>
    <style>
      .fade-enter,.fade-leave-to{
        opacity: 0;
      }
      .fade-enter-active,
      .fade-leave-active{
        transition: opacity 3s;
      }
    </style>
</head>
<body>
    <div id="app">
        <!-- type="transition" 放在transition裏面指定指行的時間以animate或者transition爲準-->
        <transition
         :duration="{enter: 5000, leave: 10000}"
         name="fade"
         appear
         enter-active-class="animated swing fade-enter-active"
         leave-active-class="animated shake fade-leave-active"
         appear-active-class="animated swing"
        >
        <div v-show="show">hello meijing</div>
        </transition>
        <button @click="handleClick">切換</button>
    </div>

    <script>
    

    var vm = new Vue({
        el: '#app',
        data: {
            show: true
        },
        methods: {
            handleClick: function(){
                this.show = !this.show
            }
        }
    })
    </script>

Js 動畫與 Velocity.js 的結合

<link rel="stylesheet" href="./animate.css">
    <script src="./vue.min.js"></script>
    <script src="./velocity.min.js"></script>
</head>
<body>
    <div id="app">
        <transition
         name="fade"
         @before-enter="handleBeforeEnter"
         @enter="handleEnter"
         @after-enter="handleAfterEnter"
        >
        <!-- <transition
         name="fade"
         @before-leave="handleBeforeEnter"
         @leave="handleEnter"
         @after-leave="handleAfterEnter"
        > -->
        <div v-show="show">hello meijing</div>
        </transition>
        <button @click="handleClick">切換</button>
    </div>

    <script>
    

    var vm = new Vue({
        el: '#app',
        data: {
            show: true
        },
        methods: {
            handleClick: function(){
                this.show = !this.show
            },
            handleBeforeEnter: function(el){
                el.style.opacity = 0;
            },
            handleEnter: function(el, done){
                Velocity(el, {opacity: 1}, {duration: 1000, complete: done})
            },
            handleAfterEnter: function(el){
                console.log("動畫結束")
            }
        }
    })
    </script>

Router

  • hash路由並不適合SEO
    • mode: ‘history’
  • base 基路徑
  • router路由樣式
    • linkActiveClass 部分匹配
    • linkExactActiveClass 徹底匹配
  • historyApiFallback 路徑映射關係
    • 要包含webpack裏的publicPath
  • scrollBehavior 記錄滾動行爲
  • parseQuery,stringifyQuery 參數
  • fallback 若是頁面不支持history路由,自動切換Hash方式
  • this.$route獲取當前的路由信息,可是並非全部的信息都有,因此可使用meta屬性
  • 同一組件內,不一樣路由有不一樣router-view可使用components替換component,而後給router-view來解決
  • 路由守衛(導航狗子) 能夠用來驗證參數
    • beforeEach
    • beforeResolve
    • afterEach 跳轉以後
  • 路由配置狗子
    • beforeEnter
  • 在組件定義狗子
    • beforeRouteEnter
    • beforeRouteUpdate
    • beforeRouteLeave (大表單確認提醒,安全性)
    • 在next以前拿不到this
    • next(vm => {console.log(vm.id)})
  • 異步加載,在進入路由中import引入組件
    • 須要插件 babel-plugin-syntax-dynamic-import

Vuex

  • 注意目錄結構的劃分
  • mapState
    • ...mapState({*** : (state) => state.**})
  • mapGetters
    • ...mapState(['**'])
    • mapActions
    • mapMutation
  • mutation只有兩個參數,修改多個數據要用對象,使用this.$store.commit(‘方法名’,{})來觸發
  • action和mutation同樣,不過主要用來作異步修改方法。使用this.$store.dispatch(‘方法名’,{參數})來觸發
  • 模塊化 modules
    • namespaced
  • 異步加載模塊
    • registerModule
    • unregisterModule
  • 熱加載
    -store.watch((state) => {},()=>{})
    當第一個方法返回值有變化的時候纔會調用第二個方法
  • store.subscribe(mutation,state)=>{}) 拿到全部mutation的變化,每次變化調用回調函數
  • subscribeAction
  • plugins 在vue初始化的時候定義

SSR

圖片描述

相關文章
相關標籤/搜索