Reputation: 809
I need to have multiple versions of a javascript library on the same page. How can I accomplish this, short of manually refactoring one version to avoid naming conflicts?
There are many examples of how to do this with Jquery (example). This seems to rely on jQuery goodness, however. How can I do this for an arbitrary script?
More detail: I'm using d3.js, and I'm plugging in visualizations others have made using d3. The issue is, one of the vizzes requires one version of d3, the other requires a newer version. Both of these vizzes are supposed to be available on the same page - the user swaps which viz is displayed by clicking a thumbnail, and then js is used to hide one viz and build the other. So, it seems like swapping the script rather than loading both in a no-conflict style could also be an option.
Upvotes: 18
Views: 12380
Reputation: 1
I've been testing this and while the selected answer may work for most scenarios, the .on function which related to the d3.event object can't seem to be rewritten.
Upvotes: 0
Reputation: 41
I had a similar problem like yours, some visualization needed d3 v3 while others needed v4. This is how I achieved it.
write the following code at the top.
function graphicviz(id) {
require.config({
paths: {
d3: "https://d3js.org/d3.v4.min"
}
});
require(["d3"], function(d3) {
//... D3 Code Here...
});
}
//Call The Function
graphicviz(id);
Upvotes: 1
Reputation: 16222
If you take a look at the main d3 source file: https://github.com/mbostock/d3/blob/master/d3.js
you see it begins:
d3 = function() {
var d3 = {
version: "3.1.5"
};
//.....
So d3 is just an object. I'm not sure if this is the best method, but I think the following would work:
d3versionX = d3;
d3 = d3versionX
where X is your default version for the visualization when the page loadsSee this jsbin for a working example. The relevant code:
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/4.10.0/d3.min.js"></script>
<script>
d3version4 = d3
window.d3 = null
</script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/3.5.17/d3.min.js"></script>
<script>
d3version3 = d3
window.d3 = null
// test it worked
console.log('v3', d3version3.version)
console.log('v4', d3version4.version)
</script>
Upvotes: 27
Reputation: 16184
From d3 version d3.v4 the header of the script looks a bit different. You can change the identifier global.d3 to global.d3v4, such that it looks like this
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) :
typeof define === 'function' && define.amd ? define(['exports'], factory) :
(factory((global.d3v4 = global.d3v4 || {})));
}(this, (function (exports) { 'use strict';
var version = "4.8.0";
...
Now you can call d3
functions by d3v4
.
Upvotes: 0
Reputation: 23482
You need to use this general principle
var x = {
version: 1,
alert1: function() {
alert("1hi1");
},
alert2: function() {
alert("1hi2");
}
};
var y = x;
var x = {
version: 2,
alert1: function() {
alert("2hi1");
},
alert2: function() {
alert("2hi2");
}
};
y.alert1();
x.alert1();
on jsfiddle
jquery offers its noconflict
method and many libraries offer the same (not necessarily by name) method. But you can do it yourself by referencing my example, depending on the complexity of the script you are loading.
Here is how to inject the 2 different versions and use the above principle to assign them to different variables.
<div id="version1"></div>
<div id="version2"></div>
var script1 = document.createElement("script"),
script2 = document.createElement("script"),
oldD3;
function noConflict() {
oldD3 = d3;
console.log("loaded old");
script2.type = 'text/javascript';
script2.src = "http://d3js.org/d3.v3.min.js";
script2.addEventListener("load", ready, false);
document.head.appendChild(script2);
}
function ready() {
console.log("loaded new");
console.log(d3, oldD3);
document.getElementById("version1").textContent = oldD3.version;
document.getElementById("version2").textContent = d3.version;
}
script1.type = 'text/javascript';
script1.src = "http://d3js.org/d3.v2.min.js";
script1.addEventListener("load", noConflict, false);
document.head.appendChild(script1);
on jsfiddle
Upvotes: 0