slightly prettier annotation links (#677)
[wolnelektury.git] / apps / catalogue / fields.py
1 # -*- coding: utf-8 -*-
2 # This file is part of Wolnelektury, licensed under GNU Affero GPLv3 or later.
3 # Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
4 #
5 import datetime
6
7 from django.conf import settings
8 from django.db import models
9 from django.db.models import signals
10 from django import forms
11 from django.forms.widgets import flatatt
12 from django.forms.util import smart_unicode
13 from django.utils import simplejson as json
14 from django.utils.html import escape
15 from django.utils.safestring import mark_safe
16 from django.utils.translation import ugettext_lazy as _
17 from south.modelsinspector import add_introspection_rules
18
19
20 class JSONEncoder(json.JSONEncoder):
21     def default(self, obj):
22         if isinstance(obj, datetime.datetime):
23             return obj.strftime('%Y-%m-%d %H:%M:%S')
24         elif isinstance(obj, datetime.date):
25             return obj.strftime('%Y-%m-%d')
26         elif isinstance(obj, datetime.time):
27             return obj.strftime('%H:%M:%S')
28         return json.JSONEncoder.default(self, obj)
29
30
31 def dumps(data):
32     return JSONEncoder().encode(data)
33
34
35 def loads(str):
36     return json.loads(str, encoding=settings.DEFAULT_CHARSET)
37
38
39 class JSONFormField(forms.CharField):
40     widget = forms.Textarea
41
42     def clean(self, value):
43         try:
44             loads(value)
45             return value
46         except ValueError, e:
47             raise forms.ValidationError(_('Enter a valid JSON value. Error: %s') % e)
48
49
50 class JSONField(models.TextField):
51     def formfield(self, **kwargs):
52         defaults = {'form_class': JSONFormField}
53         defaults.update(kwargs)
54         return super(JSONField, self).formfield(**defaults)
55
56     def db_type(self):
57         return 'text'
58
59     def get_internal_type(self):
60         return 'TextField'
61
62     def contribute_to_class(self, cls, name):
63         super(JSONField, self).contribute_to_class(cls, name)
64
65         def get_value(model_instance):
66             return loads(getattr(model_instance, self.attname, None))
67         setattr(cls, 'get_%s_value' % self.name, get_value)
68
69         def set_value(model_instance, json):
70             return setattr(model_instance, self.attname, dumps(json))
71         setattr(cls, 'set_%s_value' % self.name, set_value)
72
73 add_introspection_rules([], ["^catalogue\.fields\.JSONField"])
74
75
76 class JQueryAutoCompleteWidget(forms.TextInput):
77     def __init__(self, source, options=None, *args, **kwargs):
78         self.source = source
79         self.options = None
80         if options:
81             self.options = dumps(options)
82         super(JQueryAutoCompleteWidget, self).__init__(*args, **kwargs)
83
84     def render_js(self, field_id):
85         source = "'%s'" % escape(self.source)
86         options = ''
87         if self.options:
88             options += ', %s' % self.options
89
90         return u'$(\'#%s\').autocomplete(%s%s).result(autocomplete_result_handler);' % (field_id, source, options)
91
92     def render(self, name, value=None, attrs=None):
93         final_attrs = self.build_attrs(attrs, name=name)
94         if value:
95             final_attrs['value'] = smart_unicode(value)
96
97         if not self.attrs.has_key('id'):
98             final_attrs['id'] = 'id_%s' % name
99
100         html = u'''<input type="text" %(attrs)s/>
101             <script type="text/javascript">//<!--
102             %(js)s//--></script>
103             ''' % {
104                 'attrs' : flatatt(final_attrs),
105                 'js' : self.render_js(final_attrs['id']),
106             }
107
108         return mark_safe(html)
109
110
111 class JQueryAutoCompleteField(forms.CharField):
112     def __init__(self, source, options=None, *args, **kwargs):
113         if 'widget' not in kwargs:
114             kwargs['widget'] = JQueryAutoCompleteWidget(source, options)
115
116         super(JQueryAutoCompleteField, self).__init__(*args, **kwargs)
117
118 try:
119     # check for south
120     from south.modelsinspector import add_introspection_rules
121
122     add_introspection_rules([
123     (
124         [JSONField], # Class(es) these apply to
125         [], # Positional arguments (not used)
126         {}, # Keyword argument
127     ), ], ["^catalogue\.fields\.JSONField"])
128 except ImportError:
129     pass