欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue動態(tài)加載異步組件的方法

 更新時間:2018年11月21日 15:09:41   作者:DanielDemi  
這篇文章主要介紹了Vue動態(tài)加載異步組件的方法,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

背景:

目前我們項目都是按組件劃分的,然后各個組件之間封裝成產品。目前都是采用iframe直接嵌套頁面。項目中我們還是會碰到一些通用的組件跟業(yè)務之間有通信,這種情況下iframe并不是最好的選擇,iframe存在跨域的問題,當然是postMessage還是可以通信的,但也并非是最好的。目前有這么一個場景:門戶需要制作通用的首頁和數據概覽頁面,首頁和數據概覽頁面通過小部件來自由拼接。業(yè)務組件在制作的時候只需要提供各個模塊小部件的url就可以了,可是如果小部件之間還存在聯系呢?那么iframe是不好的。目前采用Vue動態(tài)加載異步組件的方式來實現小組件之間的通信。當然門戶也要提供一個通信的基線:Vue事件總線(空的Vue實例對象)。

內容:

使用過vue的都應該知道vue的動態(tài)加載組件components:

Vue通過is來綁定需要加載的組件。那么我們現在需要的就是如何打包組件,如果通過復制業(yè)務組件內部的代碼,那么這種就需要把依賴全部找齊,并復制過去(很多情況下會漏下某個圖片或css等),這種方式是比較low的,不方便維護。因此我們需要通過webpack來打包單個vue文件成js,這邊一個vue打包成一個js,不需壓代碼分割,css分離。因為component加載時只需要加載一個文件即可。打包文件配置如下:

首先在package.json加入打包命令:

"scripts": {
  ...
  "build-outCMP": "node build/build-out-components.js"
 },

Build-out-components.js文件:

'use strict'
require('./check-versions')()

process.env.NODE_ENV = 'production'

const ora = require('ora')
const path = require('path')
const chalk = require('chalk')
const webpack = require('webpack')
const webpackConfig = require('./webpack.out-components.prod.conf')

const spinner = ora('building for sync-components...')
spinner.start()

webpack(webpackConfig, function (err, stats) {
 spinner.stop()
 if (err) throw err
 process.stdout.write(stats.toString({
  colors: true,
  modules: false,
  children: false,
  chunks: false,
  chunkModules: false
 }) + '\n\n')

 if (stats.hasErrors()) {
  console.log(chalk.red(' Build failed with errors.\n'))
  process.exit(1)
 }

 console.log(chalk.cyan(' Build complete.\n'))
 console.log(chalk.yellow(
  ' Tip: built files are meant to be served over an HTTP server.\n' +
  ' Opening index.html over file:// won\'t work.\n'
 ))
})

webpack.out-components.prod.conf.js文件配置如下

const webpack = require('webpack');
const path = require('path');
const utils = require('./utils');
const OptimizeCSSPlugin = require('optimize-css-assets-webpack-plugin')
const {entry, mkdirsSync} = require('./out-components-tools')

function resolve(dir) {
 return path.join(__dirname, '..', dir)
}

mkdirsSync(resolve('/static/outComponents'))

module.exports = {
 entry: entry,
 output: {
  path: resolve('/static/outComponents'),
  filename: '[name].js',
 },
 resolve: {
  extensions: ['.js', '.vue', '.json'],
  alias: {
   'vue$': 'vue/dist/vue.esm.js',
   '@': resolve('src'),
  }
 },
 externals: {
  vue: 'vue',
  axios: 'axios'
 },
 module: {
  rules: [
   {
    test: /\.vue$/,
    loader: 'vue-loader',
    options: {
     esModule: false, // vue-loader v13 更新 默認值為 true v12及之前版本為 false, 此項配置影響 vue 自身異步組件寫法以及 webpack 打包結果
     loaders: utils.cssLoaders({
      sourceMap: true,
      extract: false     // css 不做提取
     }),
     transformToRequire: {
      video: 'src',
      source: 'src',
      img: 'src',
      image: 'xlink:href'
     }
    }
   },
   {
    test: /\.js$/,
    loader: 'babel-loader',
    include: [resolve('src'), resolve('test')]
   },
   {
    test: /\.(png|jpe?g|gif|svg)(\?.*)?$/,
    loader: 'url-loader',
    options: {
     limit: 10000,
     name: utils.assetsPath('img/[name].[hash:7].[ext]')
    }
   },
   {
    test: /\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/,
    loader: 'url-loader',
    options: {
     limit: 10000,
     name: utils.assetsPath('media/[name].[hash:7].[ext]')
    }
   },
   {
    test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/,
    loader: 'url-loader',
    options: {
     limit: 10000,
     name: utils.assetsPath('fonts/[name].[hash:7].[ext]')
    }
   }
  ]
 },
 plugins: [
  new webpack.DefinePlugin({
   'process.env.NODE_ENV': '"production"'
  }),
  // UglifyJs do not support ES6+, you can also use babel-minify for better treeshaking: https://github.com/babel/minify
  new webpack.optimize.UglifyJsPlugin({
   compress: false,
   sourceMap: true
  }),
  // Compress extracted CSS. We are using this plugin so that possible
  // duplicated CSS from different components can be deduped.
  new OptimizeCSSPlugin({
   cssProcessorOptions: {
    safe: true
   }
  })
 ]
};

out-components-tools.js文件配置如下:

const glob = require('glob')
const fs = require('fs');
const path = require('path');
// 遍歷要打包的組件
let entry = {}
var moduleSrcArray = glob.sync('./src/out-components/*')
for(var x in moduleSrcArray){
 let fileName = (moduleSrcArray[x].split('/')[3]).slice(0, -4)
 entry[fileName] = moduleSrcArray[x]
}

// 清理文件
function mkdirsSync(dirname) {
 if (fs.existsSync(dirname)) {
  deleteall(dirname)
  return true;
 } else {
  if (mkdirsSync(path.dirname(dirname))) {
   fs.mkdirSync(dirname);
   return true;
  }
 }
}
// 刪除文件下的文件
function deleteall(path) {
 var files = [];
 if(fs.existsSync(path)) {
  files = fs.readdirSync(path);
  files.forEach(function(file, index) {
   var curPath = path + "/" + file;
   if(fs.statSync(curPath).isDirectory()) { // recurse
    deleteall(curPath);
   } else { // delete file
    fs.unlinkSync(curPath);
   }
  });
 }
};

exports.entry = entry
exports.mkdirsSync = mkdirsSync

build-out-components是打包的入口文件,webpack.out-components.prod.conf.js是webpack打包的配置文件,out-components-tools.js是工具庫,這邊是打包的entry自動獲?。J為src/out-components),還有自動刪除之前打包的文件。

目前的文件目錄為

通過打包生產文件:

在static下outComponents文件夾內的js文件。(最終打包需要打包到dist下面,這邊做測試先打包在static文件下,方便后續(xù)動態(tài)組件ajax獲取組件使用)

門戶的小部件是通過配置url,和調整布局來生產的。因此業(yè)務組件至此已經完成了。只需要提供對門戶暴露的url即可。
接下來就是門戶這邊加載動態(tài)組件的實現了。門戶這邊就相對簡單了。看如下圖配置:

門戶通過component的動態(tài)組件來實現加載異步組件,通過ajax請求剛才打包的url,然后實例化函數new Function來賦值給mode(new Function之所以分成2部,是因此效驗規(guī)則的問題,可忽略)。這樣就實現了動態(tài)加載異步組件了。門戶和業(yè)務組件可以各個開發(fā),任何業(yè)務開發(fā)數據概覽,門戶都不需要改代碼,只需要界面上配置url即可。這個異步加載組件已經結束了。這邊門戶需要封裝一封實現異步組件。父級只需要傳入url即可。這邊還有個可以優(yōu)化的是,可以把mode優(yōu)先緩存,那么不需要每次都去加載請求。如下:

我們可以看到在門戶的一個數據概覽頁面上加載了多個異步組件,那么異步組件之間也是可能存在通信的,這樣該如何做呢?因為現在已經不是iframe嵌套了,可以通過監(jiān)聽一個組件,然調用另一個組件的方法,這樣確實可以實現平級組件間的通信,但這樣勢必不可取的,因為一旦這樣做了門戶必須要根據業(yè)務來輔助,修改代碼來實現功能。因此這邊借用門戶來生成vue事件總線(空的vue實例)來實現。

門戶代碼如下: 在this.$root上掛在一個事件總線:

 created () {
    if (!this.$root.eventBus) {
     this.$root.eventBus = new Vue()
    }
   }

然后業(yè)務組件之間就可以根據自己的業(yè)務實現通信:

組件一和組件二代碼如下:

 <template>
   <div class="test1">
    這是一個外部組件a1
    <hello-word></hello-word>
   </div>
    </template>
  
  <script>
  import helloWord from '../components/HelloWorld'
  export default {
   data () {
    return {
     i: 0
    }
   },
   components: {
    helloWord
   },
   mounted () {
    setInterval(() => {
     this.i++
     if (this.i < 10) {
      this.test()
     }
    }, 1000)
   },
   methods: {
    test () {
     this.$root.eventBus.$emit('childEvent', this.i)
    }
   }
  }
  </script>
 <template>
   <div class="test1">
    這也是外部組件哦
    <div >
     這是a1傳來的{{a1}}
    </div>
   </div>
  </template>
  
  <script>
  export default {
   data () {
    return {
     a1: 0
    }
   },
   created () {
    this.$root.eventBus.$on('childEvent', this.change)
   },
   methods: {
    change (i) {
     this.a1 = i
    }
   }
  }
  </script>

業(yè)務組件就可以根據this.$root.eventBus和vue上的事件傳遞($emit, $on)來實現相互的通信。

總結:本篇主要借助vue的動態(tài)組件和webpack打包單文件來實現動態(tài)加載異步組件,通過vue的事件總線掛載在this.$root上來實現平級組件之間的通信。

拓展方向:這個方式不僅僅可以應用在門戶單個頁面上的小部件上,同樣如果某個項目中的頁面文件需要復用時,不想通過代碼的復制,同樣可以再那個文件配置打包單文件配置,打包出的文件在領一個項目中動態(tài)加載出來即可。這種模式與通用組件的install模式是有點類似的,只是這個單文件vue不是通用的,但同樣可以達到打包復用頁面。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • vue中關于checkbox使用的問題

    vue中關于checkbox使用的問題

    這篇文章主要介紹了vue中關于checkbox使用的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • vue+vuex+axio從后臺獲取數據存入vuex實現組件之間共享數據

    vue+vuex+axio從后臺獲取數據存入vuex實現組件之間共享數據

    這篇文章主要介紹了vue+vuex+axio從后臺獲取數據存入vuex,組件之間共享數據,非常具有實用價值,需要的朋友可以參考下
    2017-04-04
  • vue 使用monaco實現代碼高亮

    vue 使用monaco實現代碼高亮

    這篇文章主要介紹了vue 使用monaco實現代碼高亮的方法,幫助大家更好的理解和學習使用vue框架,感興趣的朋友可以了解下
    2021-03-03
  • 淺談angular4.0中路由傳遞參數、獲取參數最nice的寫法

    淺談angular4.0中路由傳遞參數、獲取參數最nice的寫法

    下面小編就為大家分享一篇淺談angular4.0中路由傳遞參數、獲取參數最nice的寫法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-03-03
  • create?vite?實例源碼解析

    create?vite?實例源碼解析

    這篇文章主要為大家介紹了create?vite?實例源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-11-11
  • vue工程全局設置ajax的等待動效的方法

    vue工程全局設置ajax的等待動效的方法

    這篇文章主要介紹了vue工程全局設置ajax的等待動效的方法,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-02-02
  • 解決vue 退出動畫無效的問題

    解決vue 退出動畫無效的問題

    這篇文章主要介紹了解決vue 退出動畫無效的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-08-08
  • 在 Vue 應用中使用 Netlify 表單功能的方法詳解

    在 Vue 應用中使用 Netlify 表單功能的方法詳解

    Netlify 帶有內置表單處理功能,可以用來存儲表單數據,下載 csv 文件,同時可以在接收到新的提交時發(fā)送郵件通知或者通過配置 webhook 發(fā)送請求。這篇文章主要介紹了在 Vue 應用中使用 Netlify 表單功能,需要的朋友可以參考下
    2019-06-06
  • Vue3在history模式下如何通過vite打包部署白屏

    Vue3在history模式下如何通過vite打包部署白屏

    這篇文章主要介紹了Vue3在history模式下如何通過vite打包部署白屏問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-07-07
  • Vue實現鼠標懸浮隱藏與顯示圖片效果@mouseenter和@mouseleave事件詳解

    Vue實現鼠標懸浮隱藏與顯示圖片效果@mouseenter和@mouseleave事件詳解

    在所做的Vue項目中,有時候需要在鼠標移動文字框的時候顯示一些詳細信息,下面這篇文章主要給大家介紹了關于Vue實現鼠標懸浮隱藏與顯示圖片效果@mouseenter和@mouseleave事件的相關資料,需要的朋友可以參考下
    2022-11-11

最新評論