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