4f0a8a898c61031ca08be9506fb4837cc656adb9
[wolnelektury.git] / apps / catalogue / views.py
1 # -*- coding: utf-8 -*-
2 from django.template import RequestContext
3 from django.shortcuts import render_to_response, get_object_or_404
4 from django.http import HttpResponse, HttpResponseRedirect, Http404
5 from django.core.urlresolvers import reverse
6 from django.db.models import Q
7 from django.contrib.auth.decorators import login_required
8 from django.utils.datastructures import SortedDict
9 from django.views.decorators.http import require_POST
10 from django.contrib import auth
11 from django.contrib.auth.forms import UserCreationForm, AuthenticationForm
12 from django.utils import simplejson
13 from django.utils.functional import Promise
14 from django.utils.encoding import force_unicode
15
16 from catalogue import models
17 from catalogue import forms
18 from catalogue.utils import split_tags
19 from newtagging import views as newtagging_views
20
21
22 class LazyEncoder(simplejson.JSONEncoder):
23     def default(self, obj):
24         if isinstance(obj, Promise):
25             return force_unicode(obj)
26         return obj
27
28
29 def search(request):
30     query = request.GET.get('q', '')
31     tags = request.GET.get('tags', '')
32     if tags == '':
33         tags = []
34     
35     try:
36         tag_list = models.Tag.get_tag_list(tags)
37         tag = models.Tag.objects.get(name=query)
38     except models.Tag.DoesNotExist:
39         try:
40             book = models.Book.objects.get(title=query)
41             return HttpResponseRedirect(book.get_absolute_url())
42         except models.Book.DoesNotExist:
43             return HttpResponseRedirect(reverse('catalogue.views.main_page'))
44     else:
45         tag_list.append(tag)
46         return HttpResponseRedirect(reverse('catalogue.views.tagged_object_list', 
47             kwargs={'tags': '/'.join(tag.slug for tag in tag_list)}
48         ))
49
50
51 def tags_starting_with(request):
52     try:
53         prefix = request.GET['q']
54         if len(prefix) < 2:
55             raise KeyError
56             
57         books = models.Book.objects.filter(title__icontains=prefix)
58         tags = models.Tag.objects.filter(name__icontains=prefix)
59         if request.user.is_authenticated():
60             tags = tags.filter(~Q(category='set') | Q(user=request.user))
61         else:
62             tags = tags.filter(~Q(category='set'))
63         
64         completions = [book.title for book in books] + [tag.name for tag in tags]
65
66         return HttpResponse('\n'.join(completions))    
67     
68     except KeyError:
69         return HttpResponse('')
70
71
72 def main_page(request):    
73     if request.user.is_authenticated():
74         shelves = models.Tag.objects.filter(category='set', user=request.user)
75         new_set_form = forms.NewSetForm()
76     extra_where = 'NOT catalogue_tag.category = "set"'
77     tags = models.Tag.objects.usage_for_model(models.Book, counts=True, extra={'where': [extra_where]})
78     fragment_tags = models.Tag.objects.usage_for_model(models.Fragment, counts=True,
79         extra={'where': ['catalogue_tag.category = "theme"'] + [extra_where]})
80     categories = split_tags(tags)
81     
82     form = forms.SearchForm()
83     return render_to_response('catalogue/main_page.html', locals(),
84         context_instance=RequestContext(request))
85
86
87 def book_list(request):
88     books = models.Book.objects.all()
89     form = forms.SearchForm()
90     
91     books_by_first_letter = SortedDict()
92     for book in books:
93         books_by_first_letter.setdefault(book.title[0], []).append(book)
94     
95     return render_to_response('catalogue/book_list.html', locals(),
96         context_instance=RequestContext(request))
97
98
99 def tagged_object_list(request, tags=''):
100     # Prevent DoS attacks on our database
101     if len(tags.split('/')) > 6:
102         raise Http404
103         
104     try:
105         tags = models.Tag.get_tag_list(tags)
106     except models.Tag.DoesNotExist:
107         raise Http404
108     
109     model = models.Book
110     shelf_is_set = (len(tags) == 1 and tags[0].category == 'set')
111     theme_is_set = any(tag.category == 'theme' for tag in tags)
112     if theme_is_set:
113         model = models.Fragment
114
115     extra_where = 'NOT catalogue_tag.category = "set"'
116     related_tags = models.Tag.objects.related_for_model(tags, model, counts=True, extra={'where': [extra_where]})
117     categories = split_tags(related_tags)
118
119     if not theme_is_set:
120         model=models.Book.objects.filter(parent=None)
121     
122     return newtagging_views.tagged_object_list(
123         request,
124         tag_model=models.Tag,
125         queryset_or_model=model,
126         tags=tags,
127         template_name='catalogue/tagged_object_list.html',
128         extra_context = {'categories': categories, 'shelf_is_set': shelf_is_set },
129     )
130
131
132 def book_detail(request, slug):
133     book = get_object_or_404(models.Book, slug=slug)
134     tags = list(book.tags.filter(~Q(category='set')))
135     categories = split_tags(tags)
136     book_children = book.children.all().order_by('parent_number')
137     
138     form = forms.SearchForm()
139     return render_to_response('catalogue/book_detail.html', locals(),
140         context_instance=RequestContext(request))
141
142
143 def book_text(request, slug):
144     book = get_object_or_404(models.Book, slug=slug)
145     book_themes = {}
146     for fragment in book.fragments.all():
147         for theme in fragment.tags.filter(category='theme'):
148             book_themes.setdefault(theme, []).append(fragment)
149     
150     book_themes = book_themes.items()
151     book_themes.sort(key=lambda s: s[0].sort_key)
152     return render_to_response('catalogue/book_text.html', locals(),
153         context_instance=RequestContext(request))
154
155
156 def logout_then_redirect(request):
157     auth.logout(request)
158     return HttpResponseRedirect(request.GET.get('next', '/'))
159
160
161 @require_POST
162 def register(request):
163     registration_form = UserCreationForm(request.POST, prefix='registration')
164     if registration_form.is_valid():
165         user = registration_form.save()
166         user = auth.authenticate(
167             username=registration_form.cleaned_data['username'], 
168             password=registration_form.cleaned_data['password1']
169         )
170         auth.login(request, user)
171         response_data = {'success': True, 'errors': {}}
172     else:
173         response_data = {'success': False, 'errors': registration_form.errors}
174     return HttpResponse(LazyEncoder(ensure_ascii=False).encode(response_data))
175
176
177 @require_POST
178 def login(request):
179     form = AuthenticationForm(data=request.POST, prefix='login')
180     if form.is_valid():
181         auth.login(request, form.get_user())
182         response_data = {'success': True, 'errors': {}}
183     else:
184         response_data = {'success': False, 'errors': form.errors}
185     return HttpResponse(LazyEncoder(ensure_ascii=False).encode(response_data))
186
187
188 def book_sets(request, slug):
189     book = get_object_or_404(models.Book, slug=slug)
190     user_sets = models.Tag.objects.filter(category='set', user=request.user)
191     book_sets = book.tags.filter(category='set', user=request.user)
192     
193     if not request.user.is_authenticated():
194         return HttpResponse('<p>Aby zarządzać swoimi półkami, musisz się zalogować.</p>')
195     
196     if request.method == 'POST':
197         form = forms.ObjectSetsForm(book, request.user, request.POST)
198         if form.is_valid():
199             book.tags = ([models.Tag.objects.get(pk=id) for id in form.cleaned_data['set_ids']] +
200                 list(book.tags.filter(~Q(category='set') | ~Q(user=request.user))))
201             if request.is_ajax():
202                 return HttpResponse('<p>Półki zostały zapisane.</p>')
203             else:
204                 return HttpResponseRedirect('/')
205     else:
206         form = forms.ObjectSetsForm(book, request.user)
207         new_set_form = forms.NewSetForm()
208     
209     return render_to_response('catalogue/book_sets.html', locals(),
210         context_instance=RequestContext(request))
211
212
213 def fragment_sets(request, id):
214     fragment = get_object_or_404(models.Fragment, pk=id)
215     user_sets = models.Tag.objects.filter(category='set', user=request.user)
216     fragment_sets = fragment.tags.filter(category='set', user=request.user)
217
218     if not request.user.is_authenticated():
219         return HttpResponse('<p>Aby zarządzać swoimi półkami, musisz się zalogować.</p>')
220
221     if request.method == 'POST':
222         form = forms.ObjectSetsForm(fragment, request.user, request.POST)
223         if form.is_valid():
224             fragment.tags = ([models.Tag.objects.get(pk=id) for id in form.cleaned_data['set_ids']] +
225                 list(fragment.tags.filter(~Q(category='set') | ~Q(user=request.user))))
226             if request.is_ajax():
227                 return HttpResponse('<p>Półki zostały zapisane.</p>')
228             else:
229                 return HttpResponseRedirect('/')
230     else:
231         form = forms.ObjectSetsForm(fragment, request.user)
232         new_set_form = forms.NewSetForm()
233
234     return render_to_response('catalogue/fragment_sets.html', locals(),
235         context_instance=RequestContext(request))
236
237
238 @login_required
239 @require_POST
240 def new_set(request):
241     new_set_form = forms.NewSetForm(request.POST)
242     if new_set_form.is_valid():
243         new_set = new_set_form.save(request.user)
244         
245         if request.is_ajax():
246             return HttpResponse(u'<p>Półka <strong>%s</strong> została utworzona</p>' % new_set)
247         else:
248             return HttpResponseRedirect('/')
249     
250     return render_to_response('catalogue/book_sets.html', locals(),
251             context_instance=RequestContext(request))
252
253
254 @login_required
255 @require_POST
256 def delete_shelf(request, slug):
257     user_set = get_object_or_404(models.Tag, slug=slug, category='set', user=request.user)
258     user_set.delete()
259     
260     if request.is_ajax():
261         return HttpResponse(u'<p>Półka <strong>%s</strong> została usunięta</p>' % user_set.name)
262     else:
263         return HttpResponseRedirect('/')
264
265
266 @login_required
267 def user_shelves(request):
268     shelves = models.Tag.objects.filter(category='set', user=request.user)
269     new_set_form = forms.NewSetForm()
270     return render_to_response('catalogue/user_shelves.html', locals(),
271             context_instance=RequestContext(request))
272