Opened 15 years ago

Last modified 13 years ago

#12882 closed

jQuery.noConflict() in admin breaks site specific code with jQuery — at Version 18

Reported by: krejcik@… Owned by: Jannis Leidel
Component: contrib.admin Version: dev
Severity: Normal Keywords: jQuery admin
Cc: ales.zoulek@…, yed@…, rob@…, semente@…, michael.c.strickland@…, Gabriel Hurley Triage Stage: Accepted
Has patch: yes Needs documentation: no
Needs tests: yes Patch needs improvement: yes
Easy pickings: no UI/UX: no

Description (last modified by Jannis Leidel)

JQuery javascipt included by django admin calls noConflict method ( http://api.jquery.com/jQuery.noConflict/ ). This method removes variable $ shortcut from the global namespace.
Site specific code can include an own jQuery code to enrich admin pages. In this case, commonly used variable $ is undefined and page is broken.

Change History (20)

comment:1 by Jannis Leidel, 15 years ago

Triage Stage: UnreviewedAccepted

comment:2 by anonymous, 15 years ago

Cc: ales.zoulek@… added

comment:3 by Jiri Suchan, 15 years ago

Cc: yed@… added

comment:4 by Jannis Leidel, 15 years ago

I'm not sure what you are saying with ".. causing problem with strip of jQuery plugins after redefinition". Mind elaborating?

comment:5 by Jannis Leidel, 15 years ago

Owner: changed from nobody to Jannis Leidel
Status: newassigned

comment:6 by Ales Zoulek, 15 years ago

jezdez: Admin's jquery.min.js may overide some site-specific javascript. And since there are tons of jQuery plugins, versions and possible combinations, (and I suspect a lot of more customized admin sites uses jQuery) site should be always able to enforce it's jQuery version (and thus be responsible for and core disfunctionalities it may cause).

At the moment, overwriting default jquery is possible either by:

  • rewring ModelAdmin.media property (in every admin option class, or by monkeypatching)
  • copy'n'pasting templates/admin/change_list.html and changing only "extrahead" block

Extending "extrahead" block in base.html or site_base.html is not possible since change_list.html writes it's media (including jquery) AFTER {{block.super}}.

Possible solutions would be one of:

  • Including all admin javascripts in base.html (What is the dynamic contruction of list of internal JS files to include on every page good for, anyway? :) )
  • Introduce {% block customjavascript %} block tag in admin/base.html that would go after {% block extrahead %} since there's no easy way to put custom tags at the end of extrahead block.

Workaround:

  • Extend {% block blockbots %} tag in custom base_site.html and put custom jQuery there.

comment:7 by krejcik, 15 years ago

about plugins, see symbolic example
<head>

<script JQuery by Site>
<script Plugin 1 (by site)>

<script JQuery by Django>
<script Plugin 2 (by Django)> (eq. actions.min.js)

<head>

In example above JQuery by Django redefine jquery instance, so Plugin 1 is not defined on new jQuery instance. For example it can be solved by ommiting first jQuery and putting Plugin 1 include after Plugin 2. (But jquery by django is included in media part which is last in head, so it needs some work to override it (especially if it can be generic solution for whole project and not only one template))

In fact it is not main topic of this ticket, but i would like to show more issues with jQuery include by Django. I my opinion is not a bad idea use this great library, but O guess it should be done in more customizable way and with respect about site which can use own jquery (maybe in worst case in different version) with it's own jQuery plugins. Present situation must be hacked by ugly workarounds in client code.

comment:8 by krejcik, 15 years ago

And finally my workaround for noConflict issue is

jQuery.noConflict = function() { return this; }; {# HACK - Django calls it #}

in right place in html document.

in reply to:  7 comment:9 by Jannis Leidel, 15 years ago

Replying to krejcik:

In fact it is not main topic of this ticket, but i would like to show more issues with jQuery include by Django. I my opinion is not a bad idea use this great library, but O guess it should be done in more customizable way and with respect about site which can use own jquery (maybe in worst case in different version) with it's own jQuery plugins. Present situation must be hacked by ugly workarounds in client code.

Absolutely, that's why we are here, fixing that bug :)

comment:10 by Rob Hudson, 15 years ago

Cc: rob@… added

It is possible to completely encapsulate Django's jQuery version and its plugins, but I'm curious if that's the best thing to do here. If it's known that the Django admin has jQuery available via jQuery in the global namespace, custom admin site changes can code to that. I, for one, like knowing I can use Django's version of jQuery for some custom widgets and such.

However, if we do want to encapsulate things so different versions of jQuery can co-exist, it is possible following this skeleton code:

(function(window, document, version, callback) {
	var j, d;
	var loaded = false;
	if (!(j = window.jQuery) || version > j.fn.jquery || callback(j)) {
		var script = document.createElement("script");
		script.type = "text/javascript";
		script.src = "/path/to/jquery.js";
		script.onload = script.onreadystatechange = function() {
			if (!loaded && (!(d = this.readyState) || d == "loaded" || d == "complete")) {
				callback((j = window.jQuery).noConflict(1), loaded = true);
				j(script).remove();
			}
		};
		document.documentElement.childNodes[0].appendChild(script);
	}
})(window, document, "1.4", function($, jquery_loaded) {
	// plugin code goes here.
});

What this code does is checks if there is an existing version of jQuery loaded and checks it meets a given minimum version and if so, uses that jQuery. If it doesn't exist, or doesn't meet the minimum version, it dynamically adds a script tag to load its own jQuery and uses that. The version loaded is encapsulated within the anonymous function and will not conflict with any other Javascript libraries (e.g. other versions of jQuery, prototype, etc.).

It seems a bit overkill to me, but if it's the best option, feel free to use it.

-Rob

comment:11 by krejcik, 15 years ago

Because jQuery has stable API, which is almost backward compatible, I agree that solving different version seems to be overkill.
Personaly I see problem only in noConflict call and in jQuery instance replacement by another JS include (which cause lost of previously defined jQuery plugins as desciribed in a previous post).

by Jannis Leidel, 15 years ago

Attachment: 12882.1.diff added

Pass true to noConflict as mentioned on http://api.jquery.com/jQuery.noConflict/

comment:12 by Jannis Leidel, 15 years ago

In the attached patch I'm referring to the part of jQuery's documentation:

"If necessary, we can free up the jQuery name as well by passing true as an argument to the method. This is rarely necessary, and if we must do this (for example, if we need to use multiple versions of the jQuery library on the same page), we need to consider that most plug-ins rely on the presence of the jQuery variable and may not operate correctly in this situation."

Question is, can we test this somehow? Does it solve your issue, krejcik?

by Jannis Leidel, 15 years ago

Attachment: 12882.2.diff added

The collapse widget needs jQuery, too

comment:13 by taylormarshall, 15 years ago

"In this case, commonly used variable $ is undefined and page is broken."

krejcik: can't this be solved by just doing wrapping your code in a function? Like so:

(function($) {
   // your jQuery code goes here.
})(jQuery);

I do this very commonly in my own code, just to ensure that if someone decides to use another library or screw with the $ variable (not that uncommon on large projects), my scripts can still be reused just fine when jQuery is in noConflict mode.

My apologies if I am missing something about why this wouldn't work.

comment:14 by krejcik, 15 years ago

taylormarshall: yes it can be used in this way. but I guess main problem is in jQuery plugins. (after second jQuery include, previosly included plugins are lost

comment:15 by Guilherme Gondim (semente) <semente@…>, 15 years ago

Cc: semente@… added

comment:16 by Guilherme Gondim (semente) <semente@…>, 15 years ago

Cc: semente@… added; semente@… removed

comment:17 by Gabriel Hurley, 15 years ago

Cc: Gabriel Hurley added

comment:18 by Jannis Leidel, 15 years ago

Description: modified (diff)
Summary: jQuery.noConflict() in admin brokes site specific code with jQueryjQuery.noConflict() in admin breaks site specific code with jQuery
Note: See TracTickets for help on using tickets.
Back to Top