Vue SSR 组件加载:Node 端渲染提示 window/document 没有定义

摘要:
document.body.clientHeight:'';}使用Webpack提供的webpack.DefinePlugin插件定义常量解决。这里直接使用easywebpackhttps://github.com/hubcarl/easywebpack内置的全局Webpack常量EASY_ENV_IS_BROWSERhttp://hubcarl.github.io/easywebpack/webpack/env进行判断。document.body.clientHeight:'';}NPMVue组件SSR支持针对上面这种自己写的代码,我们可以通过这种方式解决,因为可以直接修改。但如果我们引入的一个npmVue插件想进行SSR渲染,但这个插件里面使用了window/docment等浏览器对象,并没有对SSR模式进行兼容,这个时候该如何解决呢?一般我们通过通过v-if来决定是否渲染该组件和Vue只在前端挂载组件解决问题可以解决。

业务场景

  • 首先来看一个简单的 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 对象, 如果存在则执行后面代码。 这种方式虽然能解决问题, 但在 Webpack 构建压缩时, 不会执行的代码不会被剔除,也会打包到 js 文件中去, 因为这个是在运行期才知道结果的, 所以在 Webpack 构建方案中,不建议使用 typeof 方式判断。而是使用 Webpack 提供的 webpack.DefinePlugin 插件定义常量解决。
clientHeight() {
   return typeof document === 'object' ? document.body.clientHeight : '';
}
  • 使用 Webpack 提供的 webpack.DefinePlugin 插件定义常量解决。 这里直接使用 easywebpackhttps://github.com/hubcarl/easywebpack内置的全局 Webpack 常量 EASY_ENV_IS_BROWSERhttp://hubcarl.github.io/easywebpack/webpack/env进行判断。 这样在构建压缩期间, 如果是 Node 模式构建, EASY_ENV_IS_BROWSER 会被替换为 false,如果是 Browser 模式构建, EASY_ENV_IS_BROWSER 会被替换为 true,最后构建后代码也就是变成了 true 或者 false 的常量。 因为这个是构建期间执行的,压缩插件剔除永远不会被执行的代码, 也就是 dead_code
clientHeight() {
   return EASY_ENV_IS_BROWSER ? document.body.clientHeight : '';
}

NPM Vue 组件 SSR 支持

针对上面这种自己写的代码,我们可以通过这种方式解决,因为可以直接修改。但如果我们引入的一个 npm Vue 插件想进行SSR渲染, 但这个插件里面使用了 window/docment 等浏览器对象, 并没有对 SSR 模式进行兼容,这个时候该如何解决呢?

一般我们通过通过 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() {
      // 只会在浏览器执行  
      this.$options.components.Loading = () =>import('vue-loading');
    },
    mounted(){
    }
  }
</script>

loading组件因为没有注册, 在 SSR 模式,<Loading></Loading>会被原样输出到 HTML 中,不会报错且不能被浏览器识别, 在显示时不会有内容。当 SSR 直出 HTML 后,浏览器模式中执行beforeMount挂载组件, 从而达到解决服务端渲染报错的问题

https://github.com/hubcarl/egg-vue-webpack-boilerplate/blob/master/app/web/page/dynamic/dynamic.vue​github.com/hubcarl/egg-vue-webpack-boilerplate/blo

免责声明:文章转载自《Vue SSR 组件加载:Node 端渲染提示 window/document 没有定义》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇转 EasyUi日期控件datebox设置,只显示年月,也只能选择年月sqlalchemy——基本操作下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

vue中如何实现pdf文件预览?

  今天产品提出一个优化的需求,就是之前我们做的图片展示就是一个img标签搞定,由于我们做的是海外后台管理系统,那边的人上传的文件时pdf格式,vue本事是不支持这种格式文件展示的,于是就google搜索,发现有iframe、embed、vueshowpdf(测试了不咋好用)、pdf等,本文说一下pdf插件的使用过程。 说明:iframe标签这种,对于有...

人人都能学会的webpack5搭建vue3项目(二)配置Vue

webpack5 搭建vue3 项目 (二) 安装vue以及vue-loader yarn add vue@nextyarn add vue-loader@next thread-loader -D 配置webpack.config.js const VueLoaderPlugin = require('vue-loader/dist/plugin...

vue项目---实现点击加入购物车

分析:使用Vuex状态管理。 点击加入购物车,将这个商品组成一个对象,放到store里面的car[]里。 1.在Goodsinfo.vue中,给加入购物车注册点击函数,addToShopCar  2.在methods中添加这个方法    this.ballFlag=!this.ballFlag;//这是之前控制小球的显示与隐藏   //首先,要拼接出一个...

vue 解决jsonp跨域

在Vue中使用jsonp 参考链接:https://blog.csdn.net/m0_38134431/article/details/87930647 在vue中使用vue-jsonp 参考链接:https://www.jianshu.com/p/10901e13d794 参考链接:https://www.jianshu.com/p/e4379e6690...

Vue-脚手架(一)

Vue脚手架:Vue-Cli,可方便快捷地开发完整地系统 通过 @vue/cli 实现的交互式的项目脚手架。 通过 @vue/cli + @vue/cli-service-global 实现的零配置原型开发。 一个运行时依赖 (@vue/cli-service),该依赖: 可升级; 基于 webpack 构建,并带有合理的默认配置; 可以通...

Vue生命周期

  昨天简单介绍了一下生命周期是什么,以及生命周期的几个阶段,今天具体介绍一下生命周期各个阶段都是干什么的叭!   beforeCreate(创建前):    当前生命周期函数主要做初始化工作,可以创建一个loading。   Created(创建后*****):      1.当前生命周期函数中我们可以访问到vm身上所有的属性和方法    2.当前生命周...