A simple python egg and jquery plugin to easily use JQuery in Django/Plone/.. websites.
Project description
jquery.pyproxy
The goal of jquery.pyproxy is to help integration of Ajax calls with JQuery in Python powered sites. The main idea is to use jquery syntax inside your Python call to update the page the users are seeing. This way, you do not have to return complex data to the Javascript client that will have to decrypt them.
IMPORTANT: the product is not yet finalized and might not work in every situations.
Installing
Add jquery.pyproxy to your buildout eggs and run the buildout, if you do not use buildout, you can easy_install jquery.pyproxy (or do as you usually do with other products).
In Plone, go to the quickinstaller and install jquery.pyproxy. That’s all, happy Plone users can now skip to ‘Simple example’.
In Django, you should also add django-appmedia to you buildout eggs and appmedia to the list of installed apps. Run buildout, then run bin/django symlinkmedia. Now in your templates, add:
<script type="text/javascript" src="{{MEDIA_URL}}/pyproxy/jquery.pyproxy.min.js"></script>
If you want to have a spinner automatically shown when a request is done, you can also add in your header:
<script type="text/javascript" src="{{MEDIA_URL}}/pyproxy/jquery.pyproxy.spinner.min.js"></script> <link rel="stylesheet" href="{{MEDIA_URL}}/pyproxy/jquery.pyproxy.spinner.css" type="text/css" media="screen">
You need to have a pyproxy_spinner.gif image available at the root of the site. If not (or if your media files are hosted on a different subdomain), you should declare a javascrit variable called pyproxy_spinner_location that points to the URL of the file. The declaration should be done before including the jquery.pyproxy.spinner.js file.
If you do not want to use app_media, you can download the javascript library from github (https://github.com/vincent-psarga/jquery.pyproxy/blob/master/jquery/pyproxy/media/jquery.pyproxy.min.js - please ensure to match the egg version and js version) and install it to your media folder. You can then include it to your templates.
Simple example
A simple example should be easier to explain the idea behind jquery.pyproxy. You do not want user to have to reload the full page when submitting a comment to your blog, so you will use an Ajax call. Here’s how you do this with jquery.pyproxy.
First, you create a view that will manage the information sent by the user (here with django, but the principle is the same with Plone):
from jquery.pyproxy.jq_django import JQueryProxy, jquery # The @jquery decorator handles the transformation of your results # into JSON so we can decode it on client side. @jquery def ajax_add_comment(request): # The JQuery proxy object helps us to manipulate the page the user sees. jq = JQueryProxy() # The data/form sent with Ajax just apear like a classical POST form. form = request.POST #we do some validation of the form. ... if errors: # We display an error message. jq('#my_error_message').show() return jq ... # We display a success message. jq('#my_success_message').show() # If the JQueryProxy object is not returned, nothing happens. return jq
and finally, bind a jquery.pyproxy action to the button used to submit a form:
$(document).ready(function() { $('#submit_comment').pyproxy('click', '/ajax_add_comment', '#comment_form'); });
Using the JQueryPyproxy object
First, you need to import the correct JQueryProxy object:
if you use Plone: from jquery.pyproxy.plone import JQueryProxy
if you use Django: from jquery.pyproxy.jq_django import JQueryProxy
Both are based on the same object, defined in the base.py module:
>>> from jquery.pyproxy.base import JQueryProxy
The main idea behind this project was to be able to use the same syntax than with jQuery, so you can more or less copy-paste some samples on the web or some existing code. Of course, as we are using Python, we can not use the dollar sign as a identifier:
>>> $ = JQueryPyproxy() Traceback (most recent call last): ... SyntaxError: invalid syntax
So in our examples we declare our object as jq, which is the classic way in Plone to name the jQuery object:
>>> jq = JQueryProxy()
Currently, JQueryProxy supports all manipulation API of JQuery 1.4 and all the effect API.
>>> sorted(jq.grammar.keys()) ['addClass', 'after', 'animate', 'append', 'appendTo', 'attr', 'before', 'css', 'detach', 'empty', 'fadeIn', 'fadeOut', 'fadeTo', 'find', 'hide', 'html', 'insertAfter', 'insertBefore', 'parent', 'prepend', 'prependTo', 'remove', 'removeAttr', 'removeClass', 'replaceAll', 'replaceWith', 'show', 'slideDown', 'slideToggle', 'slideUp', 'text', 'toggle', 'toggleClass', 'unwrap', 'wrap', 'wrapAll', 'wrapInner']
So if you know how to use them in jQuery, you know how to use them with pyproxy, for example:
>>> jq('#error_msg').html('Errors have been found, please correct them') >>> jq('#error_email').show()
The way the jQuery methods are declared are matching the API of jQuery (except for the callbacks, see the ‘Limitations’ part). So if you use incorrect arguments, you will get errors in the Python code (which should help you a lot when debugging, at least you should have server logs):
>>> jq('.to_slide').slideDown() Traceback (most recent call last): ... TypeError: Method 'slideDown' takes exactly 1 arguments >>> jq('.empty').empty('Spanish argument is like Spanish inquisition: no one expects it') Traceback (most recent call last): ... TypeError: Method 'empty' does not take any argument >>> jq('.to_fade').fadeTo('something', 'wrong type') Traceback (most recent call last): ... TypeError: Argument 2 of method fadeTo must be: <type 'int'>
If you need to process a list of selectors, you can use the batch method of the JQueryObject. It takes five arguments:
a list of selectors
the method to apply
the list of arguments for this method
a prefix to add before each selector (optional)
a substituion list on which the selector will be applied (optional)
That can be usefull for example when you have a list of error to display:
>>> my_errors = ['email', 'title', 'text'] >>> jq.batch(my_errors, 'addClass', ['error'], substitution='#%s_error') >>> jq.list_calls()[-3:] ["jq('#email_error').addClass('error')", "jq('#title_error').addClass('error')", "jq('#text_error').addClass('error')"]
That is equivalent to:
>>> for err in my_errors: ... jq('#' + err + '_error').addClass('error') >>> jq.list_calls()[-3:] ["jq('#email_error').addClass('error')", "jq('#title_error').addClass('error')", "jq('#text_error').addClass('error')"]
You can also chain the jQuery calls as you would do on the Javascript side:
>>> jq('.nice_divs').css({'width': '200px'}).fadeIn(10) >>> jq.list_calls()[-1] "jq('.nice_divs').css({'width': '200px'}).fadeIn(10)"
Note that we don’t check (yet) if the call returns something that can be chained.
When you need to have a clear overview of which calls have been done by the jq object, you can use the list_calls method:
>>> jq.list_calls() ["jq('#error_msg').html('Errors have been found, please correct them')", "jq('#error_email').show()", "jq('.to_slide').slideDown()", "jq('.empty').empty()", "jq('.to_fade').fadeTo()", "jq('#email_error').addClass('error')", "jq('#title_error').addClass('error')", "jq('#text_error').addClass('error')", "jq('#email_error').addClass('error')", "jq('#title_error').addClass('error')", "jq('#text_error').addClass('error')", "jq('.nice_divs').css({'width': '200px'}).fadeIn(10)"]
You can see that even failing calls are stored here (but not the parameters). This is due to the fact that we are in the doctests. In real-life use case, as a exception is raised your code will stop after the problem has been found.
Extending JQuery proxy
If you want to be able to use jQuery methods that are not known by default, you have to extend the list of methods known by JQueryPyproxy. In our example, we’ll consider that you want to use a showDialog method from an extra-plugin. By default, it does not works (which is logical as pyproxy does not have a clue of which jQuery plugins you are using):
>>> jq('.bla').showDialog() Traceback (most recent call last): ... AttributeError: 'JQueryCommand' object has no attribute 'showDialog'
To be able to use the method, you need to define the list of extra methods you want to use and the parameters expected. Here we only define the showDialog method, taking three parameters (the first one is a string or unicode, the second one an int, a string or a unicode and the last one a optional dictionnary):
>>> from types import NoneType >>> my_plugin = {'showDialog': [[str, unicode], ... [str, unicode, int], ... [dict, NoneType]]}
Then, you can use the extend_grammar method so your methods are recognized:
>>> jq.extend_grammar(my_plugin) >>> 'showDialog' in jq.grammar True >>> jq.grammar['showDialog'] [[<type 'str'>, <type 'unicode'>], [<type 'str'>, <type 'unicode'>, <type 'int'>], [<type 'dict'>, <type 'NoneType'>]] >>> jq('#my_dialog').showDialog('some text', 42, dict(opt1 = 2, opt2 = False))
And of course it respects the grammar you defined:
>>> jq('.bla').showDialog() Traceback (most recent call last): ... TypeError: Method 'showDialog' takes between 2 and 3 arguments >>> jq('.bla').showDialog(1, 2, 3) Traceback (most recent call last): ... TypeError: Argument 1 of method showDialog must have one of the following types: [<type 'str'>, <type 'unicode'>] >>> jq('.grep').showDialog('blabla', 'fich')
If you need to use custom methods in all your Ajax views, it will be painfull to extend the grammar every time. You have some options to solve this.
1) if you use the source code of jquery.pyproxy: add a file called my_plugin.py in jquery.pyproxy/jquery/pyproxy/plugins. In this file, describe your plugin with the dictionnary as explained before. This dictionnary must be called grammar.
2) you do not use the source, just the egg. Create a new Python class, subclassing JQueryProxy. Declare a base_grammar property in this object that describes your grammar:
>>> class MyJQueryProxy(JQueryProxy): ... base_grammar = {'showDialog': [[str, unicode], ... [str, unicode, int], ... [dict, NoneType]]} >>> my_jq = MyJQueryProxy() >>> my_jq('.bla').showDialog('a', 2)
and in your views, use this class instead of the JQueryProxy class.
3) integrate your grammar in the next release. In any case, do not hesitate to submit the grammars you defined so it can be integrated in the next release of jquery.pyproxy.
Limitations
There is currently two (at least) major limitations.
First, you can not store your selector, like this:
>>> jq = JQueryProxy() >>> divs = jq('.nice_divs') >>> divs.css({'width': '200px'}) >>> divs.fadeIn(10)
It seems to work fine, but if we have a close looks to what has been stored by the jq object, we can see that only the last call was saved and the call to ‘css({width’: ‘200px’})’ will never be executed:
>>> jq.list_calls() ["jq('.nice_divs').fadeIn(10)"]
The proper way to write this code is:
>>> jq = JQueryProxy() >>> jq('.nice_divs').css({'width': '200px'}) >>> jq('.nice_divs').fadeIn(10)
Now if we look at what has ben stored by the object, we see all wanted calls:
>>> jq.list_calls() ["jq('.nice_divs').css({'width': '200px'})", "jq('.nice_divs').fadeIn(10)"]
The second one is that the callback are not handled, so you can not use something like this:
>>> jq('.animated').show(10, 'my_callback') Traceback (most recent call last): ... TypeError: Method 'show' takes between 0 and 1 arguments
Even if the jQuery doc tells that the show method can take multiple arguments (which is true, but not here).
Is there some samples available ?
If you use Plone, add the following to one available configure.zcml file (the one from your theme from example):
<include package="jquery.pyproxy.samples.plone" />
Restart the instance and then open http://localhost:8080/your_plone_site/pyproxy_samples.
If you use Django, some samples will be added later.
Testing the module
There are tests embedded in this package to ensure it works correctly. To run the tests on the python side, you can run:
bin/instance test -m jquery.pyproxy (for Plone users) bin/django test pyproxy (for Django users with a buildout) ./manage.py test pyproxy (for Django users without buildout)
There is also qUnit tests to ensure the jQuery library works correclty. For the moment it is only available for Plone users. First, you have to load the ‘tests.zcml’ file from jquery.pyproxy. For example in the main configure.zcml of a product you develop:
<include package="jquery.pyproxy" file="tests.zcml" />
Then, in the ZMI, go to the portal_setup, then the import tab. Select jquery.pyproxy tests in the list, select the Skins tools step and then click on Import selected steps. In the portal_skins tool, you should see a new folder called pyproxy_tests. Now open http://localhost:8080/your_plone_site/pyproxy_tests and you will see the qUnit tests running.
I use Python but not Django or Plone
You should use Django.
If this solution is not acceptable, you can still update the @jquery decorator to work with your framework. The only thing this decorator does is to transform the JQueryProxy object returned by the function into JSON. To make the transformation, this code is enough:
>>> import simplejson as json >>> jq_to_json = json.dumps(jq.json_serializable()) >>> jq_to_json '[{"args": [{"width": "200px"}], "call": "css", "selector": ".nice_divs"}, {"args": [10], "call": "fadeIn", "selector": ".nice_divs"}, {"args": [], "call": "show", "selector": ".animated"}]'
Then, the jq_to_json object must be returned according to your framework system (for example for Plone we just return it, for Django we wrap it into a HttpResponse object).
If you ported the @jquery decorator to any framework, please let me know so it can be integrated in the next release.
Compatibility
Tested with:
jQuery 1.2 and 1.4
Python 2.4 and 2.6
Firefox
Chrome
Safari
IE
Roadmap
What should be coming for the next releases.
0.5
tests for the Plone specifics. [done]
tests for the Django specifics (there should only be the @jquery decorator). [done]
0.6
refactor the jQuery plugin, so it takes only one argument which will be a dictionnary of options.
include some utility to extract data from a URL (the opposite of the native ‘params’ method).
allow debug mode for a specific call instead of having a global setting.
add custom events when request starts/ends.
use the events to show the spinner.
1.0
allow storing the selector and do multiple calls on it.
later on
allow callbacks.
Changelog
0.4.3 (2013-06-04)
Point to http://github.com/zestsoftware/jquery.pyproxy [maurits]
added Django specific tests. [vincent]
finished Plone specific tests. [vincent]
0.4.2 (2013-02-25)
Fix issue with Diazo wrapping the responses with HTML tags. [vincent]
0.4.1 (2011-09-19)
Updated the minified version. [vincent]
0.4 (2011-09-19)
added find() and parent() support. [vincent]
added support for chained calls. [vincent]
Added support for ‘this’. [vincent]
0.3.1 (2011-05-09)
removed the ‘debug’ fonction that was sometimes comflicting with other products. [vincent]
0.3 (2011-02-25)
you can now specify a callback without specifying a form to send. [vincent]
added tests for the jQuery plugin. [vincent]
removed the ‘eval()’ calls in process_call. Also removed arg_to_string not needed anymore and disabled ‘clean_string’ which is also not needed anymore. [vincent]
extracted sub-methods from the jquery plugin to easy testing. All are prefixed with pyproxy to avoid name conflicts. [vincent]
added roadmap. [vincent]
added extra JS/CSS file to show a spinner when a request is done. The spinner image was generated using ajaxload.info/ [vincent]
finally added tests for the base python code. Tests for the jquery plugin and customized Plone.Django object to come. [vincent+maurits]
IMPORTANT: the ‘grammar’ attribute has been renamed ‘base_grammar’ attribute. If you created sub-classes of the JQueryProxy object, update them. Using the ‘grammar’ attribute caused a problem as it was not initialized in the __init__ method, the value was spread to every instances of the JQueryProxy object. [vincent]
added a ‘batch’ method on the JueryProxy object to process multiple entries. It should be used to replace show_errors and hide_errors. [vincent]
deprecated jq.show_errors and jq.hide_errors as it was based on Plone CSS classes. [vincent]
moved ‘clean_string’, ‘package_contents’ and ‘custom_endswith’ to utils. [vincent]
small bugfix in Plone version to hide the previous portal messages. [vincent]
updated __init__.py to declare the namespace to avoid problems with setuptools 0.7. [vincent]
0.2 (2010-10-22)
renamed django.py in jq_django.py as it was giving conflicts when importing HttpResponse. [vincent]
moved jquery.pyproxy.js in a folder called ‘media’ so it can be used easily by django user with django-appmedia. Also added some registration for plone users so jquery.pyproxy appears in the quick_installer and automatically adds the JS file to the jsregistry. [vincent]
bugfix: fadeOut and fadeIn did not have correct names in plugins/jq_effects. [vincent]
Added ‘PreventDefault’ in make_call - using ‘return false’ works for links but is not enough for submit buttons. [vincent]
0.1 (2010-04-19)
bugfix in base: package_contents did not work with python 2.4. [vincent]
renamed jquerypyproxy in jquery.pyproxy [vincent]
bugfix in base: the grammar was not passed to the objects created. [vincent]
bugfix in base: str.endswith(tuple) only works in python 2.5+, added a custom endswith. [vincent]
bugfix in plone.py: typo in the class name. [vincent]
added append in grammar. [vincent]
added basic javascript calls. [vincent]
added JQueryProxy Python class and decorators for plone and Django. [vincent]
added log file + python egg. [vincent]
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.