我试图使用grunt将SPA应用程序(requirejs,durandal 2,knockout)构建到单个main-build.js文件中,并且我正在使用'文' durandal用来加载我的观点的插件。
在开发中,我成功使用了' text'根据构建durandal应用程序的标准方式动态加载视图。不同之处在于我需要为视图做一些服务器端模板,因此它们实际上是动态生成的。
考虑到这一点,我想使用r.js将应用模型,视图模型和服务(通过grunt-durandal插件)打包到一个文件中,但不要打包视图(.html并且仍然根据需要动态加载它们。
在我的咕噜配置中,我使用了inlineText: false
选项 - 我已经检查过它会抑制文本!*'构建中的模块。但是当我运行应用程序时,我得到了:
Uncaught TypeError: undefined is not a function
内的 text.load
位于以下行:
var parsed = text.parseName(name),
nonStripName = parsed.moduleName +
(parsed.ext ? '.' + parsed.ext : ''),
url = req.toUrl(nonStripName), // EXCEPTION THROWN HERE
正在加载的模块名称似乎是正确的(它是一个' text!*' one)但除此之外我不知道如何继续调试此问题。我做错了什么?
我的咕噜声是:
/*global module, require */
module.exports = function (grunt) {
'use strict';
// library that allows config objects to be merged together
var mixIn = require('mout/object/mixIn');
var requireConfig = {
baseUrl: 'App/',
paths: {
'jquery': '../Scripts/jquery-2.1.0',
'knockout': '../Scripts/knockout-3.1.0',
'text': '../Scripts/text',
'durandal': '../Scripts/durandal',
'plugins': '../Scripts/durandal/plugins',
'transitions': '../Scripts/durandal/transitions',
}
};
grunt.initConfig({
pkg: grunt.file.readJSON('package.json'),
jshint: {
options: {
"-W099": true, // allowed mixed tabs and spaces
"-W004": true, // needed to avoid errors where TS fakes inheritance
ignores: [
'App/main-built.js' // ingore the built/compacted file
]
},
all: [ // run jshint on these files
'gruntfile.js',
'App/**/*.js'
]
},
// TODO: could add jasmine here to do JS testing
clean: {
build: ['build/*']
},
copy: {
scripts: {
src: 'Scripts/**/**',
dest: 'build/'
}
},
durandal: {
main: {
src: [
'App/**/*.*',
'!App/main-built.js', // ignore the built file!
'Scripts/durandal/**/*.js'
],
options: {
name: '../Scripts/almond-custom',
baseUrl: requireConfig.baseUrl,
mainPath: 'App/main',
paths: mixIn(
{},
requireConfig.paths,
{ 'almond': '../Scripts/almond-custom.js' }),
exclude: [],
inlineText: false, // prevent bundling of .html (since we are dynamically generating these for content)
optimize: 'none', // turn off optimisations - uglify will run seperately by grunt to do this
out: 'build/app/main-built.js'
}
}
},
uglify: {
options: {
banner: '/*! <%= pkg.name %> <%= grunt.template.today("yyyy-mm-dd") %> \n' +
'* Copyright (c) <%= grunt.template.today("yyyy") %> Kieran Benton \n' +
'*/\n'
},
build: {
src: 'build/App/main.js',
dest: 'build/App/main-built.js'
}
}
}
);
// loading plugin(s)
grunt.loadNpmTasks('grunt-contrib-clean');
grunt.loadNpmTasks('grunt-contrib-connect');
grunt.loadNpmTasks('grunt-contrib-copy');
grunt.loadNpmTasks('grunt-contrib-jasmine');
grunt.loadNpmTasks('grunt-contrib-jshint');
grunt.loadNpmTasks('grunt-contrib-uglify');
grunt.loadNpmTasks('grunt-contrib-watch');
grunt.loadNpmTasks('grunt-open');
grunt.loadNpmTasks('grunt-durandal');
// only one grunt task
grunt.registerTask('build', [
'jshint',
'clean',
'copy',
'durandal:main']);
};
答案 0 :(得分:10)
杏仁不支持动态加载。它不实现require.toUrl
并且不支持异步加载器插件。 RequireJS和Almond的创建者James Burke回答了同样的问题here。
要解决此问题,您可以将RequireJS包含在捆绑包中而不是Almond中。查看示例here。您必须在r.js配置的require.js
部分中为paths
创建别名,因为require
是一个特殊的保留依赖项名称。然后,在配置的name
字段中指定此别名,而不是Almond。你会得到这样的东西:
options: {
name: 'requireLib', // use the alias
baseUrl: requireConfig.baseUrl,
mainPath: 'App/main',
paths: mixIn(
{},
requireConfig.paths,
{ 'requireLib': '../Scripts/require.js' }), // declare the alias
exclude: [],
inlineText: false,
optimize: 'none',
out: 'build/app/main-built.js'
}
答案 1 :(得分:0)
只是想为这个问题增加一些额外的背景,因为我一直回到它寻求指导,最终解决了我的问题。我使用Durandal w / TypeScript和grunt进行优化。我在文本插件方面遇到了一些例外,例如
<div data-bind="compose: { model: someObject, view: '../../components/something/something.html' }"></div>
&#13;
我最终删除了相对路径并执行以下操作:
<div data-bind="compose: { model: someObject, view: 'components/something/something.html' }"></div>
&#13;
通过这种方式,我能够通过优化的构建工作获得所有功能。