Fixed complex books (books with hasPart).
[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     return newtagging_views.tagged_object_list(
120         request,
121         tag_model=models.Tag,
122         queryset_or_model=model,
123         tags=tags,
124         template_name='catalogue/tagged_object_list.html',
125         extra_context = {'categories': categories, 'shelf_is_set': shelf_is_set },
126     )
127
128
129 def book_detail(request, slug):
130     book = get_object_or_404(models.Book, slug=slug)
131     tags = list(book.tags.filter(~Q(category='set')))
132     categories = split_tags(tags)
133     book_children = book.children.all().order_by('parent_number')
134     
135     form = forms.SearchForm()
136     return render_to_response('catalogue/book_detail.html', locals(),
137         context_instance=RequestContext(request))
138
139
140 def book_text(request, slug):
141     book = get_object_or_404(models.Book, slug=slug)
142     
143     return render_to_response('catalogue/book_text.html', locals(),
144         context_instance=RequestContext(request))
145
146
147 def logout_then_redirect(request):
148     auth.logout(request)
149     return HttpResponseRedirect(request.GET.get('next', '/'))
150
151
152 @require_POST
153 def register(request):
154     registration_form = UserCreationForm(request.POST, prefix='registration')
155     if registration_form.is_valid():
156         user = registration_form.save()
157         user = auth.authenticate(
158             username=registration_form.cleaned_data['username'], 
159             password=registration_form.cleaned_data['password1']
160         )
161         auth.login(request, user)
162         response_data = {'success': True, 'errors': {}}
163     else:
164         response_data = {'success': False, 'errors': registration_form.errors}
165     return HttpResponse(LazyEncoder(ensure_ascii=False).encode(response_data))
166
167
168 @require_POST
169 def login(request):
170     form = AuthenticationForm(data=request.POST, prefix='login')
171     if form.is_valid():
172         auth.login(request, form.get_user())
173         response_data = {'success': True, 'errors': {}}
174     else:
175         response_data = {'success': False, 'errors': form.errors}
176     return HttpResponse(LazyEncoder(ensure_ascii=False).encode(response_data))
177
178
179 def book_sets(request, slug):
180     book = get_object_or_404(models.Book, slug=slug)
181     user_sets = models.Tag.objects.filter(category='set', user=request.user)
182     book_sets = book.tags.filter(category='set', user=request.user)
183     
184     if not request.user.is_authenticated():
185         return HttpResponse('<p>Aby zarządzać swoimi półkami, musisz się zalogować.</p>')
186     
187     if request.method == 'POST':
188         form = forms.ObjectSetsForm(book, request.user, request.POST)
189         if form.is_valid():
190             book.tags = ([models.Tag.objects.get(pk=id) for id in form.cleaned_data['set_ids']] +
191                 list(book.tags.filter(~Q(category='set') | ~Q(user=request.user))))
192             if request.is_ajax():
193                 return HttpResponse('<p>Półki zostały zapisane.</p>')
194             else:
195                 return HttpResponseRedirect('/')
196     else:
197         form = forms.ObjectSetsForm(book, request.user)
198         new_set_form = forms.NewSetForm()
199     
200     return render_to_response('catalogue/book_sets.html', locals(),
201         context_instance=RequestContext(request))
202
203
204 def fragment_sets(request, id):
205     fragment = get_object_or_404(models.Fragment, pk=id)
206     user_sets = models.Tag.objects.filter(category='set', user=request.user)
207     fragment_sets = fragment.tags.filter(category='set', user=request.user)
208
209     if not request.user.is_authenticated():
210         return HttpResponse('<p>Aby zarządzać swoimi półkami, musisz się zalogować.</p>')
211
212     if request.method == 'POST':
213         form = forms.ObjectSetsForm(fragment, request.user, request.POST)
214         if form.is_valid():
215             fragment.tags = ([models.Tag.objects.get(pk=id) for id in form.cleaned_data['set_ids']] +
216                 list(fragment.tags.filter(~Q(category='set') | ~Q(user=request.user))))
217             if request.is_ajax():
218                 return HttpResponse('<p>Półki zostały zapisane.</p>')
219             else:
220                 return HttpResponseRedirect('/')
221     else:
222         form = forms.ObjectSetsForm(fragment, request.user)
223         new_set_form = forms.NewSetForm()
224
225     return render_to_response('catalogue/fragment_sets.html', locals(),
226         context_instance=RequestContext(request))
227
228
229 @login_required
230 @require_POST
231 def new_set(request):
232     new_set_form = forms.NewSetForm(request.POST)
233     if new_set_form.is_valid():
234         new_set = new_set_form.save(request.user)
235         
236         if request.is_ajax():
237             return HttpResponse(u'<p>Półka <strong>%s</strong> została utworzona</p>' % new_set)
238         else:
239             return HttpResponseRedirect('/')
240     
241     return render_to_response('catalogue/book_sets.html', locals(),
242             context_instance=RequestContext(request))
243
244
245 @login_required
246 @require_POST
247 def delete_shelf(request, slug):
248     user_set = get_object_or_404(models.Tag, slug=slug, category='set', user=request.user)
249     user_set.delete()
250     
251     if request.is_ajax():
252         return HttpResponse(u'<p>Półka <strong>%s</strong> została usunięta</p>' % user_set.name)
253     else:
254         return HttpResponseRedirect('/')
255
256
257 @login_required
258 def user_shelves(request):
259     shelves = models.Tag.objects.filter(category='set', user=request.user)
260     new_set_form = forms.NewSetForm()
261     return render_to_response('catalogue/user_shelves.html', locals(),
262             context_instance=RequestContext(request))
263