Vue SSR 組件加載問題
Node 端渲染提示 window/document 沒有定義
業(yè)務(wù)場景
首先來看一個簡單的 Vue 組件 test.vue
<template> <div> <h2>clientHeight: {{ clientHeight }} px </h2> </div> </template> <script type="text/babel"> export default { data(){ return { } }, computed :{ clientHeight() { return document.body.clientHeight; } }, mounted(){ } } </script>
上面 test.vue 組件通過 Vue computed 屬性 clientHeight 直接獲取 document 的文檔高度,這段代碼在前端渲染是不會報錯的,也能拿到正確的值。但如果把這個組件放到 SSR(Server Side Render) 模式下, 就會報如下錯誤:
ReferenceError: document is not defined
解決方案
通過 typeof 判斷是否是存在 document 對象, 如果存在則執(zhí)行后面代碼。 這種方式雖然能解決問題, 但在 Webpack 構(gòu)建壓縮時, 不會執(zhí)行的代碼不會被剔除,也會打包到 js 文件中去, 因為這個是在運(yùn)行期才知道結(jié)果的, 所以在 Webpack 構(gòu)建方案中,不建議使用 typeof 方式判斷。而是使用 Webpack 提供的 webpack.DefinePlugin 插件定義常量解決。
clientHeight() { return typeof document === 'object' ? document.body.clientHeight : ''; }
使用 Webpack 提供的 webpack.DefinePlugin 插件定義常量解決。 這里直接使用 easywebpack https:// github.com/hubcarl/easy webpack 內(nèi)置的全局 Webpack 常量 EASY_ENV_IS_BROWSER http:// hubcarl.github.io/easyw ebpack/webpack/env 進(jìn)行 判斷。 這樣在構(gòu)建壓縮期間, 如果是 Node 模式構(gòu)建, EASY_ENV_IS_BROWSER 會被替換為 false,如果是 Browser 模式構(gòu)建, EASY_ENV_IS_BROWSER 會被替換為 true,最后構(gòu)建后代碼也就是變成了 true 或者 false 的常量。 因為這個是構(gòu)建期間執(zhí)行的,壓縮插件剔除永遠(yuǎn)不會被執(zhí)行的代碼, 也就是
dead_code clientHeight() { return EASY_ENV_IS_BROWSER ? document.body.clientHeight : ''; }
NPM Vue 組件 SSR 支持
針對上面這種自己寫的代碼,我們可以通過這種方式解決,因為可以直接修改。但如果我們引入的一個 npm Vue 插件想進(jìn)行SSR渲染, 但這個插件里面使用了 window/docment 等瀏覽器對象, 并沒有對 SSR 模式進(jìn)行兼容,這個時候該如何解決呢?
一般我們通過 通過 v-if 來決定是否渲染該組件 和 Vue 只在前端掛載組件解決問題 可以解決。
通過 v-if 來決定是否渲染該組件
<template> <div v-if="isBrowser"> <Loading></Loading> </div> </template> <script type="text/babel"> export default { componets:{ Loading: () =>import('vue-loading'); } data(){ return { isBrowser: EASY_ENV_IS_BROWSER } }, mounted(){ } } </script>
Vue 只在前端掛載組件解決問題
<template> <div> <Loading></Loading> </div> </template> <script type="text/babel"> export default { data(){ return { } }, beforeMount() { // 只會在瀏覽器執(zhí)行 this.$options.components.Loading = () =>import('vue-loading'); }, mounted(){ } } </script>
loading 組件因為沒有注冊, 在 SSR 模式, <Loading></Loading> 會被原樣輸出到 HTML 中,不會報錯且不能被瀏覽器識別, 在顯示時不會有內(nèi)容。當(dāng) SSR 直出 HTML 后,瀏覽器模式中執(zhí)行 beforeMount 掛載組件, 從而達(dá)到解決服務(wù)端渲染報錯的問題
總結(jié)
以上所述是小編給大家介紹的Vue SSR 組件加載問題,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
- CentOS7 上利用 jenkins 實(shí)現(xiàn)自動部署
- 在CentOS7上搭建Jenkins+Maven+Git持續(xù)集成環(huán)境的方法
- Centos7+Docker+Jenkins+ASP.NET Core 2.0自動化發(fā)布與部署的實(shí)現(xiàn)
- centos7 安裝Jenkins詳細(xì)介紹
- 詳解Vue基于 Nuxt.js 實(shí)現(xiàn)服務(wù)端渲染(SSR)
- 基于vue-ssr服務(wù)端渲染入門詳解
- 簡述pm2常用命令集合及配置文件說明
- pm2發(fā)布node配置文件ecosystem.json詳解
- 如何使用pm2快速將項目部署到遠(yuǎn)程服務(wù)器
- 詳解從買域名到使用pm2部署node.js項目全過程
- 利用pm2部署多個node.js項目的配置教程
- pm2 部署 node的三種方法示例
- pm2啟動ssr失敗的解決方法
相關(guān)文章
vue2.0的計算屬性computed和watch的區(qū)別及各自?使用場景解讀
這篇文章主要介紹了vue2.0的計算屬性computed和watch的區(qū)別及各自?使用場景,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-01-01vue 使用iView組件中的Table實(shí)現(xiàn)定時自動滾動效果
要在css中設(shè)置table的高度,使數(shù)據(jù)過多時出現(xiàn)滾動條,將縱向設(shè)置為overflow-y: auto;橫向設(shè)置隱藏 overflow-x: hidden,接下來通過本文介紹vue使用iView組件中的Table實(shí)現(xiàn)定時自動滾動效果,需要的朋友可以參考下2024-05-05淺談webpack SplitChunksPlugin實(shí)用指南
這篇文章主要介紹了淺談webpack SplitChunksPlugin實(shí)用指南,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-09-09