1 # This file is part of FNP-Redakcja, licensed under GNU Affero GPLv3 or later.
2 # Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
4 from hashlib import sha1
5 from os import makedirs
8 from django.conf import settings
9 from django.contrib.auth.decorators import permission_required
10 from django.http import HttpResponse, HttpResponseRedirect, Http404, JsonResponse
11 from django.shortcuts import get_object_or_404, render
12 from django.views.decorators.csrf import csrf_exempt
13 from django.views.decorators.http import require_POST
14 from lxml import etree
15 from librarian import RDFNS, DCNS
16 from librarian.cover import make_cover
17 from librarian.dcparser import BookInfo
18 from documents.helpers import active_tab
19 from documents.models import Book, Chunk
20 from cover.models import Image
21 from cover import forms
22 from cover.utils import get_import_data
25 PREVIEW_SIZE = (212, 300)
28 def preview(request, book, chunk=None, rev=None):
29 """Creates a cover image.
31 If chunk and rev number are given, use version from given revision.
32 If rev is not given, use publishable version.
35 chunk = Chunk.get(book, chunk)
36 except Chunk.DoesNotExist:
39 if chunk.book.cover and rev is None and not request.GET.get('width') and not request.GET.get('height'):
40 return HttpResponseRedirect(chunk.book.cover.url)
44 revision = chunk.at_revision(rev)
45 except Chunk.change_model.DoesNotExist:
48 revision = chunk.publishable()
51 xml = revision.materialize().encode('utf-8')
54 info = BookInfo.from_bytes(xml)
55 except Exception as e:
57 return HttpResponseRedirect(os.path.join(settings.STATIC_URL, "img/sample_cover.png"))
58 width = request.GET.get('width')
59 width = int(width) if width else None
60 height=request.GET.get('height')
61 height = int(height) if height else None
63 if not (height or width):
64 width, height = PREVIEW_SIZE
66 cover_class = request.GET.get('cover_class', 'default')
69 if chunk.book.project is not None:
70 if chunk.book.project.logo_mono or chunk.book.project.logo:
71 kwargs['cover_logo'] = (chunk.book.project.logo_mono or chunk.book.project.logo).path
72 cover = make_cover(info, cover_class=cover_class, width=width, height=height, **kwargs)
73 response = HttpResponse(content_type=cover.mime_type())
74 img = cover.final_image()
75 img.save(response, cover.format)
77 if 'download' in request.GET:
78 response['Content-Disposition'] = 'attachment; filename=%s.jpg' % chunk.book.slug
85 def preview_from_xml(request):
86 xml = request.POST['xml']
88 info = BookInfo.from_bytes(xml.encode('utf-8'))
89 except Exception as e:
91 return HttpResponse(os.path.join(settings.STATIC_URL, "img/sample_cover.png"))
92 coverid = sha1(etree.tostring(info.to_etree())).hexdigest()
93 cover = make_cover(info)
95 cover_dir = 'cover/preview'
97 makedirs(os.path.join(settings.MEDIA_ROOT, cover_dir))
100 fname = os.path.join(cover_dir, "%s.%s" % (coverid, cover.ext()))
101 img = cover.image().resize(PREVIEW_SIZE, PIL.Image.ANTIALIAS)
102 img.save(os.path.join(settings.MEDIA_ROOT, fname))
103 return HttpResponse(os.path.join(settings.MEDIA_URL, fname))
107 def image(request, pk):
108 img = get_object_or_404(Image, pk=pk)
110 if not request.accepts('text/html') and request.accepts('application/json') or request.GET.get('format') == 'json':
111 return JsonResponse({
113 'author': img.author,
114 'license_name': img.license_name,
115 'license_url': img.license_url,
116 'source_url': img.source_url,
117 'attribution': img.attribution,
118 'cut_left': img.cut_left,
119 'cut_right': img.cut_right,
120 'cut_top': img.cut_top,
121 'cut_bottom': img.cut_bottom,
122 'file': img.file.url,
123 'use_file': img.use_file.url,
126 if request.user.has_perm('cover.change_image'):
127 if request.method == "POST":
128 form = forms.ImageEditForm(request.POST, request.FILES, instance=img)
131 return HttpResponseRedirect(img.get_absolute_url())
133 form = forms.ImageEditForm(instance=img)
136 form = forms.ReadonlyImageEditForm(instance=img)
139 return render(request, "cover/image_detail.html", {
140 "object": Image.objects.get(id=img.id),
142 "editable": editable,
146 def image_file(request, pk):
147 img = get_object_or_404(Image, pk=pk)
148 return HttpResponseRedirect(img.file.url)
152 def image_list(request):
153 return render(request, "cover/image_list.html", {
154 'object_list': Image.objects.all().order_by('-id'),
155 'can_add': request.user.has_perm('cover.add_image'),
159 @permission_required('cover.add_image')
161 def add_image(request):
163 if request.method == 'POST':
164 if request.POST.get('form_id') == 'import':
165 ff = forms.ImportForm(request.POST)
167 form = forms.ImageAddForm(ff.cleaned_data)
169 form = forms.ImageAddForm(request.POST, request.FILES)
172 return HttpResponseRedirect(obj.get_absolute_url())
174 form = forms.ImageAddForm()
176 ff = forms.ImportForm()
177 return render(request, 'cover/add_image.html', {
182 @permission_required('cover.add_image')
183 def quick_import(request, pk):
184 url = request.POST.get('url')
185 if url.startswith('%s://%s/' % (
187 request.get_host())):
188 cover_id = url.rstrip('/').rsplit('/', 1)[-1]
189 cover = Image.objects.get(pk=cover_id)
191 data = get_import_data(url)
192 same = Image.objects.filter(source_url=data['source_url'])
193 if not same.exists():
194 same = Image.objects.filter(download_url=data['download_url'])
198 form = forms.ImageAddForm(data)
202 # We have a cover. Now let's commit.
203 book = Book.objects.get(pk=pk)
205 text = chunk.head.materialize()
207 root = etree.fromstring(text)
208 rdf = root.find('.//' + RDFNS('Description'))
209 for tag in 'url', 'attribution', 'source':
210 for elem in rdf.findall('.//' + DCNS('relation.coverImage.%s' % tag)):
212 e = etree.Element(DCNS('relation.coverImage.url'))
213 e.text = request.build_absolute_uri(cover.use_file.url)
216 e = etree.Element(DCNS('relation.coverImage.attribution'))
219 e.text += cover.title + ', '
221 e.text += cover.author + ', '
222 e.text += cover.license_name
225 e = etree.Element(DCNS('relation.coverImage.source'))
226 e.text = cover.get_full_url()
230 xml = etree.tostring(root, encoding='unicode')
235 publishable=chunk.head.publishable,
237 return HttpResponseRedirect(book.get_absolute_url())