0

我正在使用 Angular 4 +webpack。我在 webpack.config.vendor.js 的 nonTreeShakableModules const 中添加了一个 jQuery 插件:

const path = require('path');
const webpack = require('webpack');
const ExtractTextPlugin = require('extract-text-webpack-plugin');
const merge = require('webpack-merge');

const treeShakableModules = [
    '@angular/animations',
    '@angular/common',
    '@angular/compiler',
    '@angular/core',
    '@angular/forms',
    '@angular/http',
    '@angular/platform-browser',
    '@angular/platform-browser-dynamic',
    '@angular/router',
    'font-awesome/css/font-awesome.css',
    'zone.js',

];
const nonTreeShakableModules = [
    'bootstrap',
    'bootstrap/dist/css/bootstrap.css',
    'es6-promise',
    'es6-shim',
    'event-source-polyfill',
    'jquery',
    'virtual-keyboard'              //HERE
];

启动应用程序时出现此错误:

NodeInvocationException:预渲染失败,因为错误:错误: jQuery 需要一个带有文档的窗口

如果我刷新页面 2-3 次,错误就消失了。谢谢你的帮助!

4

1 回答 1

2

正如您在上一个问题的评论中所说,您无法window通过服务器端预渲染在服务器端运行依赖于和其他一些情况(如会话存储)的 javascript 代码。

诸如 ASP.NET Core Angular Web 模板之类的模板带有启用服务器端渲染的功能。这适用于不需要会话存储、身份验证或访问浏览器组件或 dom-tree 的应用程序。

您必须通过从.asp-prerender-module="ClientApp/dist/app.module.server.ts"Index.cshtml

代替

<my-app asp-prerender-module="ClientApp/dist/app.module.server.ts"></my-app>

<my-app></my-app>

当然替换my-app为您的应用程序的选择器,通常app在模板中。

或者,您必须有条件地运行代码,如此 GitHub 问题中所指出的:

// boot-client.ts file 
import 'ngx-charts';

// some.component.ts
import { isBrowser } from 'angular2-universal';
import * as $ from 'jquery';

// inside ngOnInit 
if (isBrowser) { 
    $('body').hide();  // or whatever call you need to make
}

以避免在服务器端预渲染上运行此类代码。

于 2017-10-30T20:00:49.320 回答