16

我在一个多页项目中使用 RequireJS,它的 Javascript 文件夹结构看起来有点像这样(你如何在 Markdown 中再次制作那些花哨的目录树?):

common.js
lib/
-- jquery-1.9.1.min.js
-- modernizr-2.6.2.min.js
-- underscore-amd.min.js
page/
-- index.js
-- start.js
-- checkout.js

无论如何,common.js这是我设置配置参数的主要脚本文件。这就是它的样子:

common.js 文件

// Configure RequireJS
requirejs.config({
    baseUrl: "assets/js/lib",
    paths: {
        page: '../page',
        jquery: [
            '//ajax.googleapis.com/ajax/libs/jquery/1.9.1/jquery.min',
            //If the CDN location fails, load from this location
            'jquery-1.9.1.min'
        ],
        modernizr: [
            '//cdnjs.cloudflare.com/ajax/libs/modernizr/2.6.2/modernizr.min',
            //If the CDN location fails, load from this location
            'modernizr-2.6.2.min'
        ],
        underscore: [
            'underscore-amd.min'
        ]
    }
});

所有页面调用都按预期工作(加载了 CDN 位置),但我无法理解缩小部分。优化器r.js只是拒绝合作,Error: paths fallback not supported in optimizer. Please provide a build config path override for underscore即使 Underscore 没有指定 CDN,也会抛出一个。

构建.js 文件

{
    appDir: '../www',
    baseUrl: 'js/lib',
    dir: '../www-built',
    mainConfigFile: '../www/assets/js/common.js',
    modules: [
        //First set up the common build layer.
        {
            //module names are relative to baseUrl
            name: '../common',
            include: ['jquery',
                      'modernizr',
                      'underscore'
            ]
        },
        // Now the page scripts
        {
            //module names are relative to baseUrl/paths config
            name: '../page-index',
            include: ['page/index'],
            exclude: ['../common']
        },
    ],
    paths: {
        jquery: "empty",
        modernizr: "empty"
    },
    optimize: "uglify2",
    removeCombined: true
}

请帮我弄清楚如何构建这个项目来创建common.js脚本以及各个页面脚本。

(注意:我的build.js文件结构基于此示例

更新!

我已经更新了问题以包含正确的语法empty:(感谢 Travis!),现在构建运行没有错误。但是,我的 JS 文件没有连接或丑化。CSS 文件在导入点,所以发生了一些事情。完整build.js文件如下(请原谅我,但她个子很高):

{
    appDir: '../www',
    baseUrl: 'assets/js', // relative to appDir
    dir: '../www-built',
    mainConfigFile: '../www/assets/js/common.js',
    modules: [
        //First set up the common build layer.
        {
            //module names are relative to baseUrl
            name: 'common',
            //List common dependencies here. Only need to list
            //top level dependencies, "include" will find
            //nested dependencies.
            include: ['jquery',
                      'modernizr',
                      'underscore',
                      'bootstrap'
            ]
        },

        //Now set up a build layer for each page, but exclude
        //the common one. "exclude" will exclude nested
        //the nested, built dependencies from "common". Any
        //"exclude" that includes built modules should be
        //listed before the build layer that wants to exclude it.
        //"include" the appropriate "app/main*" module since by default
        //it will not get added to the build since it is loaded by a nested
        //require in the page*.js files.
        {
            //module names are relative to baseUrl/paths config
            name: 'pages/home',
            include: ['pages/home'],
            exclude: ['common']
        },
        {
            //module names are relative to baseUrl/paths config
            name: 'pages/start',
            include: ['pages/start'],
            exclude: ['common']
        },
        {
            //module names are relative to baseUrl/paths config
            name: 'pages/checkout',
            include: ['pages/checkout'],
            exclude: ['common']
        },
    ],
    paths: {
        jquery: "empty:",
        modernizr: "empty:"
//        underscore: "empty:"
    },
    optimize: "uglify2",
    optimizeCss: "standard",
    removeCombined: true,
    preserveLicenseComments: false
}

最终更新(耶!它正在工作)

感谢下面的特拉维斯,一切都很好!(文件被缩小和连接)。由于他的解决方案托管在 Dropbox 上,并且将来可能会丢失(谁知道 amirite?),我将总结他所做的修复:

1. 不要在一个文件中混用define和调用。require

我有几个文件,我像这样混合它们:

页面/start.js

 define(['jquery','underscore'],function($,_) {
      ...
 });

 require(['jquery','page/start'], function($, Y) { // BAD!
      ...
 });

解决方法是这样做:

页面/start.js

 require(['jquery','app/start_helper'], function($, Y) {
      ...
 });

应用程序/start_helper.js

 define(['jquery','underscore'],function($,_) {
      ...
 });

2."empty:"不是"empty"

这是一个棘手的问题,尽管 RequireJS 文档提到了它们。

3. 因为

什么样的列表只有 2 个要点?


Awesomelicious - 现在它就像一个魅力:)

4

1 回答 1

7

看起来 requireJs 认为您正在尝试为下划线提供路径后备,因为您将其路径值指定为数组。试着把它作为一个字符串来代替。

paths: {
    underscore: 'underscore-amd.min'
}

另请注意,空路径的正确符号empty:(带有':') not empty

最后,作为一个不相关的旁注,您可以随时查看lodash,恕我直言,它是一种更可定制、跨浏览器兼容、更快的下划线替换,具有相同的 API,它托管在 cdnjs 上并且符合 AMD 标准

更新

跟进评论中的对话,这是我对您项目的更新版本:https ://dl.dropboxusercontent.com/u/21823728/so_rjs.tar.gz

正如评论中提到的,您的问题似乎是define您在需要这些模块的同一文件中使用模块,并且我认为这导致 r.js 认为这些模块没有主入口点。约定是将模块定义与需要这些模块的文件分开。

请注意,现在在您的页面模块中曾经存在的每个模块现在都存储在该app文件夹下。您的页面模块现在包含这些模块。当我重新运行(使用 uglify2 作为优化器)时,一切似乎都按预期工作。assets/jsdefinerequirer.js

此外,我从你的 build.js 文件中删除了一些冗余(你只需要在你的 mainConfigFile 中指定 baseUrl,如果这也是你用于构建配置的那个)。最后,如果您想将 jQuery 和 Bootstrap 全局附加到每个页面,您可能需要考虑使用shim 配置。否则,您应该在需要时将它们明确列为依赖项。

最后,作为最后一条建议,您可能希望减少require每个文件中的数量。对于大多数这些页面,您可以将所有内容包装在一个 require 调用中,然后将不同的逻辑分离到函数中,以节省事件队列上的一些空间以及一些必须冗余调用require函数的循环。

于 2013-08-16T11:17:33.063 回答