Vue SSR 组件加载问题


Posted in Javascript onMay 02, 2018

Node 端渲染提示 window/document 没有定义

Vue SSR 组件加载问题

业务场景

首先来看一个简单的 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 插件定义常量解决。 这里直接使用 easywebpack     https:// github.com/hubcarl/easy    webpack   内置的全局 Webpack 常量 EASY_ENV_IS_BROWSER  http:// hubcarl.github.io/easyw ebpack/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

总结

以上所述是小编给大家介绍的Vue SSR 组件加载问题,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
js 事件小结 表格区别
Aug 13 Javascript
jQuery设置与获取HTML,文本和值的简单实例
Feb 26 Javascript
jQuery表单域属性过滤器用法分析
Feb 10 Javascript
jQuery实现菜单感应鼠标滑动动画效果的方法
Feb 28 Javascript
JavaScript中数据结构与算法(二):队列
Jun 19 Javascript
javascript生成大小写字母
Jul 03 Javascript
js实现文本框宽度自适应文本宽度的方法
Aug 13 Javascript
JavaScript 数组- Array的方法总结(推荐)
Jul 21 Javascript
JS模拟实现方法重载示例
Aug 03 Javascript
js获取form表单所有数据的简单方法
Aug 18 Javascript
Vue项目中使用better-scroll实现一个轮播图自动播放功能
Dec 03 Javascript
a标签调用js的方法总结
Sep 05 Javascript
基于jquery实现左右上下移动效果
May 02 #jQuery
关于Vue在ie10下空白页的debug小结
May 02 #Javascript
解析Json字符串的三种方法日常常用
May 02 #Javascript
使用vue-cli创建项目的图文教程(新手入门篇)
May 02 #Javascript
vue中的模态对话框组件实现过程
May 01 #Javascript
vue如何通过id从列表页跳转到对应的详情页
May 01 #Javascript
使用Vue的slot插槽分发父组件内容实现高度复用、更加灵活的组件(推荐)
May 01 #Javascript
You might like
漫荒推荐:画风超赞的国风漫画推荐 超长假期不无聊
2020/03/08 国漫
PHP中的按位与和按位或操作示例
2014/01/27 PHP
ecshop后台编辑器替换成ueditor编辑器
2015/03/03 PHP
PHP中使用Memache作为进程锁的操作类分享
2015/03/30 PHP
PHP命令空间namespace及use的用法小结
2017/11/27 PHP
laravel5.5安装jwt-auth 生成token令牌的示例
2019/10/24 PHP
jQuery代码优化 遍历篇
2011/11/01 Javascript
javascript html5实现表单验证
2016/03/01 Javascript
基于ajax与msmq技术的消息推送功能实现代码
2016/12/26 Javascript
一篇看懂vuejs的状态管理神器 vuex状态管理模式
2017/04/20 Javascript
layui form.render('select', 'test2') 更新渲染的方法
2019/09/27 Javascript
Vue的data、computed、watch源码浅谈
2020/04/04 Javascript
Python 检查数组元素是否存在类似PHP isset()方法
2014/10/14 Python
python连接字符串的方法小结
2015/07/13 Python
Python实现针对中文排序的方法
2017/05/09 Python
深入浅析Python中list的复制及深拷贝与浅拷贝
2018/09/03 Python
使用pandas把某一列的字符值转换为数字的实例
2019/01/29 Python
Python3 实现文件批量重命名示例代码
2019/06/03 Python
大家都说好用的Python命令行库click的使用
2019/11/07 Python
CSS3与动画有关的属性transition、animation、transform对比(史上最全版)
2017/08/18 HTML / CSS
基于HTML5的WebGL经典3D虚拟机房漫游动画
2017/11/15 HTML / CSS
世界上最大的折扣香水店:FragranceNet.com
2016/10/26 全球购物
奥地利网上书店:Weltbild
2017/07/14 全球购物
来自美国主售篮球鞋的零售商店:KICKSUSA
2017/11/28 全球购物
Osklen官方在线商店:巴西服装品牌
2019/04/25 全球购物
Ooni英国官网:披萨烤箱
2020/05/31 全球购物
英文自荐信
2013/12/15 职场文书
鉴定评语大全
2014/05/05 职场文书
公司2014年度工作总结
2014/12/10 职场文书
幼儿园辞职信
2015/05/13 职场文书
驳回起诉民事裁定书
2015/05/19 职场文书
企业愿景口号
2015/12/25 职场文书
小学班主任培训心得体会
2016/01/07 职场文书
2019客服个人年终工作总结范文
2019/07/08 职场文书
Css预编语言及区别详解
2021/04/25 HTML / CSS
java中重写父类方法加不加@Override详解
2021/06/21 Java/Android