18

我正在编写一个脚本,该脚本旨在嵌入到 3rd 方网站上以向它们添加功能。我最近删除了我相当混乱的自定义加载程序代码,并开始用 requirejs 替换它。可以选择为我加载的库之一(取决于传入的一些参数)是 jQuery。

这很好用,直到我的脚本包含在 jQuery 已经存在的页面上,在这种情况下,似乎发生的是一些插件在中途加载,requirejs 在页面的版本上加载 jQuery,并且插件立即中断。

要求客户重写他们的页面只是为了使用这个脚本是不可能的,所以我想做的是检测jQuery是否已经加载,如果是,跳过通过requirejs加载它,只使用已经加载的一个(当他们使用更旧版本的 jQuery 时,这可能会让我遇到奇怪的边缘情况和错误,但我没有太多选择)。

我想我会写一个新模块,首先查看是否加载了 jQuery,如果是,则导出 jQuery 对象,如果不是,则加载它,然后执行导出。但是,我似乎受到了阻碍,因为模块的定义函数似乎需要同步才能工作,所以我不能去加载另一个异步脚本,然后将导出填充到 requirejs 中。

我在文档中遗漏了什么吗?我正在尝试的事情是不可能的吗?

4

3 回答 3

26

我在一个类似的项目中遇到了同样的问题,将 require.js 用于打算在 3rd 方网站上加载的库。你可以在这里看到我的方法,但这里是简化版本:

// check for existing jQuery
var jQuery = window.jQuery,
    // check for old versions of jQuery
    oldjQuery = jQuery && !!jQuery.fn.jquery.match(/^1\.[0-4](\.|$)/),
    localJqueryPath = libPath + 'jquery/jquery-1.7.2.min',
    paths = {},
    noConflict;

// check for jQuery 
if (!jQuery || oldjQuery) {
    // load if it's not available or doesn't meet min standards
    paths.jquery = localJqueryPath;
    noConflict = !!oldjQuery;
} else {
    // register the current jQuery
    define('jquery', [], function() { return jQuery; });
}

// set up require
require.config({
    paths: paths 
});

// load stuff
require(['jquery', ... ], function($, ...) {

    // deal with jQuery versions if necessary
    if (noConflict) $.noConflict(true);

    // etc

});

如您所见,这会查找 jQuery,然后将“jquery”模块定义为现有库的包装器,或者(如果没有 jQuery 或现有 jQuery 是旧版本)加载特定于库的 jQuery noConflict.

这工作得很好。唯一的缺点是您在require()脚本中动态调用,这使得r.js有效使用优化器变得更加困难。

于 2012-09-04T17:05:53.427 回答
3

我有一个类似的问题..我的脚本重新加载了 jQuery 两次..使用了本文评论中的解决方案

像魅力一样工作!

(有效的评论是:

“我把它放在我的配置文件中,在 requirejs.config(...) 之后和 requirej(["app"]) 之前,它起作用了")

于 2014-02-25T23:12:29.113 回答
0

通过一些玩弄 srini 的解决方案,我能够获得一种“条件”手段,仅当网页尚未加载 jquery 时才在 requirejs 中加载它。在我的 main.js 顶部附近(我也在加载角度):

requirejs.config({
    "baseUrl": "./app/",
    "paths": {
        "jquery"            : "https://code.jquery.com/jquery-3.6.0.min.js",
        "angular"           : "https://ajax.googleapis.com/ajax/libs/angularjs/1.4.5/angular.min"
    },
    "shim": {
        "jquery": {
            "exports" : "$"
        },
        "angular": {
            "exports" : 'angular'
        }
    }
})

// don't load jquery if it is already loaded
if(typeof(jQuery) == 'function') {
    // define 'jquery' so requirejs knows how to get it
    define('jquery', [], function() {
        return jQuery;
    });
} else {
    // require (preload) jquery so it's available in requirejs
    requirejs(['jquery'], function( $ ) {
        return $; // optional
    });
}

我用 jquery 的本地副本确认,我在顶部添加了一个“console.log()”,它只被加载一次。我还在我的 html 文件中使用脚本标记进行了测试,没有在服务方法和控制器方法中引用“$”,并且都使用或不使用脚本标记。

于 2021-09-09T18:41:29.247 回答